clang 23.0.0git
Wasm.cpp
Go to the documentation of this file.
1//===----------------- Wasm.cpp - Wasm Interpreter --------------*- C++ -*-===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8//
9// This file implements interpreter support for code execution in WebAssembly.
10//
11//===----------------------------------------------------------------------===//
12
13#include "Wasm.h"
14
15#include "llvm/Support/FileSystem.h"
16#include "llvm/Support/Path.h"
17#include <llvm/IR/LegacyPassManager.h>
18#include <llvm/IR/Module.h>
19#include <llvm/MC/TargetRegistry.h>
20#include <llvm/Target/TargetMachine.h>
21
23
24#include <string>
25
26namespace lld {
27enum Flavor {
29 Gnu, // -flavor gnu
30 MinGW, // -flavor gnu MinGW
31 WinLink, // -flavor link
32 Darwin, // -flavor darwin
33 Wasm, // -flavor wasm
34};
35
36using Driver = bool (*)(llvm::ArrayRef<const char *>, llvm::raw_ostream &,
37 llvm::raw_ostream &, bool, bool);
38
43
44struct Result {
47};
48
49Result lldMain(llvm::ArrayRef<const char *> args, llvm::raw_ostream &stdoutOS,
50 llvm::raw_ostream &stderrOS, llvm::ArrayRef<DriverDef> drivers);
51
52namespace wasm {
53bool link(llvm::ArrayRef<const char *> args, llvm::raw_ostream &stdoutOS,
54 llvm::raw_ostream &stderrOS, bool exitEarly, bool disableOutput);
55} // namespace wasm
56} // namespace lld
57
58#include <dlfcn.h>
59
60namespace clang {
61
63 llvm::ErrorAsOutParameter EAO(&Err);
64
65 if (Err)
66 return;
67
68 if (auto EC =
69 llvm::sys::fs::createUniqueDirectory("clang-wasm-exec-", TempDir))
70 Err = llvm::make_error<llvm::StringError>(
71 "Failed to create temporary directory for Wasm executor: " +
72 EC.message(),
73 llvm::inconvertibleErrorCode());
74}
75
77
79 std::string ErrorString;
80
81 const llvm::Target *Target = llvm::TargetRegistry::lookupTarget(
82 PTU.TheModule->getTargetTriple(), ErrorString);
83 if (!Target) {
84 return llvm::make_error<llvm::StringError>("Failed to create Wasm Target: ",
85 llvm::inconvertibleErrorCode());
86 }
87
88 llvm::TargetOptions TO = llvm::TargetOptions();
89 llvm::TargetMachine *TargetMachine = Target->createTargetMachine(
90 PTU.TheModule->getTargetTriple(), "", "", TO, llvm::Reloc::Model::PIC_);
91 PTU.TheModule->setDataLayout(TargetMachine->createDataLayout());
92
93 llvm::SmallString<256> ObjectFileName(TempDir);
94 llvm::sys::path::append(ObjectFileName, PTU.TheModule->getName() + ".o");
95
96 llvm::SmallString<256> BinaryFileName(TempDir);
97 llvm::sys::path::append(BinaryFileName, PTU.TheModule->getName() + ".wasm");
98
99 std::error_code EC;
100 llvm::raw_fd_ostream ObjectFileOutput(ObjectFileName, EC);
101
102 if (EC)
103 return llvm::errorCodeToError(EC);
104
105 llvm::legacy::PassManager PM;
106 if (TargetMachine->addPassesToEmitFile(PM, ObjectFileOutput, nullptr,
107 llvm::CodeGenFileType::ObjectFile)) {
108 return llvm::make_error<llvm::StringError>(
109 "Wasm backend cannot produce object.", llvm::inconvertibleErrorCode());
110 }
111
112 if (!PM.run(*PTU.TheModule)) {
113
114 return llvm::make_error<llvm::StringError>("Failed to emit Wasm object.",
115 llvm::inconvertibleErrorCode());
116 }
117
118 ObjectFileOutput.close();
119
120 std::vector<const char *> LinkerArgs = {"wasm-ld",
121 "-shared",
122 "--import-memory",
123 "--experimental-pic",
124 "--stack-first",
125 "--allow-undefined",
126 ObjectFileName.c_str(),
127 "-o",
128 BinaryFileName.c_str()};
129
130 const lld::DriverDef WasmDriver = {lld::Flavor::Wasm, &lld::wasm::link};
131 std::vector<lld::DriverDef> WasmDriverArgs;
132 WasmDriverArgs.push_back(WasmDriver);
134 lld::lldMain(LinkerArgs, llvm::outs(), llvm::errs(), WasmDriverArgs);
135
136 if (Result.retCode)
137 return llvm::make_error<llvm::StringError>(
138 "Failed to link incremental module", llvm::inconvertibleErrorCode());
139
140 void *LoadedLibModule =
141 dlopen(BinaryFileName.c_str(), RTLD_NOW | RTLD_GLOBAL);
142 if (LoadedLibModule == nullptr) {
143 llvm::errs() << dlerror() << '\n';
144 return llvm::make_error<llvm::StringError>(
145 "Failed to load incremental module", llvm::inconvertibleErrorCode());
146 }
147
148 return llvm::Error::success();
149}
150
152 return llvm::make_error<llvm::StringError>("Not implemented yet",
153 llvm::inconvertibleErrorCode());
154}
155
157 // This seems to be automatically done when using dlopen()
158 return llvm::Error::success();
159}
160
162 // Can't call cleanUp through IncrementalExecutor as it
163 // tries to deinitialize JIT which hasn't been initialized
164 return llvm::Error::success();
165}
166
169 SymbolNameKind NameKind) const {
170 void *Sym = dlsym(RTLD_DEFAULT, Name.str().c_str());
171 if (!Sym) {
172 return llvm::make_error<llvm::StringError>("dlsym failed for symbol: " +
173 Name.str(),
174 llvm::inconvertibleErrorCode());
175 }
176
177 return llvm::orc::ExecutorAddr::fromPtr(Sym);
178}
179
180llvm::Error WasmIncrementalExecutor::LoadDynamicLibrary(const char *name) {
181 void *handle = dlopen(name, RTLD_NOW | RTLD_GLOBAL);
182 if (!handle) {
183 llvm::errs() << dlerror() << '\n';
184 return llvm::make_error<llvm::StringError>("Failed to load dynamic library",
185 llvm::inconvertibleErrorCode());
186 }
187 return llvm::Error::success();
188}
189} // namespace clang
llvm::Expected< llvm::orc::ExecutorAddr > getSymbolAddress(llvm::StringRef Name, SymbolNameKind NameKind) const override
Definition Wasm.cpp:168
llvm::Error addModule(PartialTranslationUnit &PTU) override
Definition Wasm.cpp:78
llvm::Error runCtors() const override
Definition Wasm.cpp:156
WasmIncrementalExecutor(llvm::Error &Err)
Definition Wasm.cpp:62
llvm::Error LoadDynamicLibrary(const char *name) override
Definition Wasm.cpp:180
llvm::Error removeModule(PartialTranslationUnit &PTU) override
Definition Wasm.cpp:151
llvm::Error cleanUp() override
Definition Wasm.cpp:161
#define bool
Definition gpuintrin.h:32
The JSON file list parser is used to communicate input to InstallAPI.
@ Result
The result type of a method or function.
Definition TypeBase.h:905
bool link(llvm::ArrayRef< const char * > args, llvm::raw_ostream &stdoutOS, llvm::raw_ostream &stderrOS, bool exitEarly, bool disableOutput)
Definition Wasm.cpp:26
Flavor
Definition Wasm.cpp:27
@ Darwin
Definition Wasm.cpp:32
@ MinGW
Definition Wasm.cpp:30
@ Invalid
Definition Wasm.cpp:28
@ WinLink
Definition Wasm.cpp:31
@ Gnu
Definition Wasm.cpp:29
@ Wasm
Definition Wasm.cpp:33
Result lldMain(llvm::ArrayRef< const char * > args, llvm::raw_ostream &stdoutOS, llvm::raw_ostream &stderrOS, llvm::ArrayRef< DriverDef > drivers)
bool(*)(llvm::ArrayRef< const char * >, llvm::raw_ostream &, llvm::raw_ostream &, bool, bool) Driver
Definition Wasm.cpp:36
The class keeps track of various objects created as part of processing incremental inputs.
std::unique_ptr< llvm::Module > TheModule
The llvm IR produced for the input.
Driver d
Definition Wasm.cpp:41
Flavor f
Definition Wasm.cpp:40
bool canRunAgain
Definition Wasm.cpp:46
int retCode
Definition Wasm.cpp:45