24#include "llvm/ADT/SmallString.h"
25#include "llvm/ADT/SmallVector.h"
26#include "llvm/ADT/StringRef.h"
27#include "llvm/ADT/Twine.h"
29#include "llvm/ExecutionEngine/JITLink/JITLinkMemoryManager.h"
30#include "llvm/ExecutionEngine/Orc/Debugging/DebuggerSupport.h"
31#include "llvm/ExecutionEngine/Orc/EPCDynamicLibrarySearchGenerator.h"
32#include "llvm/ExecutionEngine/Orc/ExecutionUtils.h"
33#include "llvm/ExecutionEngine/Orc/ExecutorProcessControl.h"
34#include "llvm/ExecutionEngine/Orc/JITTargetMachineBuilder.h"
35#include "llvm/ExecutionEngine/Orc/LLJIT.h"
36#include "llvm/ExecutionEngine/Orc/MapperJITLinkMemoryManager.h"
37#include "llvm/ExecutionEngine/Orc/Shared/OrcRTBridge.h"
38#include "llvm/ExecutionEngine/Orc/Shared/SimpleRemoteEPCUtils.h"
39#include "llvm/ExecutionEngine/Orc/SimpleRemoteEPC.h"
41#include "llvm/Support/Error.h"
42#include "llvm/Support/FileSystem.h"
43#include "llvm/Support/FormatVariadic.h"
44#include "llvm/Support/Path.h"
45#include "llvm/Support/raw_ostream.h"
47#include "llvm/TargetParser/Host.h"
58#include <netinet/in.h>
59#include <sys/socket.h>
68 if (TT.getTriple() == llvm::sys::getProcessTriple())
70 return llvm::orc::JITTargetMachineBuilder::detectHost();
73 return llvm::orc::JITTargetMachineBuilder(TT);
78 auto JITBuilder = std::make_unique<llvm::orc::LLJITBuilder>();
79 JITBuilder->setJITTargetMachineBuilder(std::move(JTMB));
80 JITBuilder->setPrePlatformSetup([](llvm::orc::LLJIT &J) {
83 consumeError(llvm::orc::enableDebuggerSupport(J));
84 return llvm::Error::success();
86 return std::move(JITBuilder);
91 unsigned SlabAllocateSize) {
92 llvm::orc::SharedMemoryMapper::SymbolAddrs SAs;
93 if (
auto Err = SREPC.getBootstrapSymbols(
95 llvm::orc::rt::ExecutorSharedMemoryMapperServiceInstanceName},
97 llvm::orc::rt::ExecutorSharedMemoryMapperServiceReserveWrapperName},
100 ExecutorSharedMemoryMapperServiceInitializeWrapperName},
103 ExecutorSharedMemoryMapperServiceDeinitializeWrapperName},
106 ExecutorSharedMemoryMapperServiceReleaseWrapperName}}))
107 return std::move(Err);
110 if (llvm::Triple(llvm::sys::getProcessTriple()).isOSWindows())
111 SlabSize = 1024 * 1024;
113 SlabSize = 1024 * 1024 * 1024;
115 if (SlabAllocateSize > 0)
116 SlabSize = SlabAllocateSize;
118 return llvm::orc::MapperJITLinkMemoryManager::CreateWithMapper<
119 llvm::orc::SharedMemoryMapper>(SlabSize, SREPC, SAs);
123 std::pair<std::unique_ptr<llvm::orc::SimpleRemoteEPC>, uint32_t>>
125 unsigned SlabAllocateSize,
std::function<
void()> CustomizeFork) {
128 return llvm::make_error<llvm::StringError>(
129 "-" + ExecutablePath +
" not supported on non-unix platforms",
130 llvm::inconvertibleErrorCode());
131#elif !LLVM_ENABLE_THREADS
133 return llvm::make_error<llvm::StringError>(
134 "-" + ExecutablePath +
135 " requires threads, but LLVM was built with "
136 "LLVM_ENABLE_THREADS=Off",
137 llvm::inconvertibleErrorCode());
140 if (!llvm::sys::fs::can_execute(ExecutablePath))
141 return llvm::make_error<llvm::StringError>(
142 llvm::formatv(
"Specified executor invalid: {0}", ExecutablePath),
143 llvm::inconvertibleErrorCode());
145 constexpr int ReadEnd = 0;
146 constexpr int WriteEnd = 1;
155 if (pipe(ToExecutor) != 0 || pipe(FromExecutor) != 0)
156 return llvm::make_error<llvm::StringError>(
157 "Unable to create pipe for executor", llvm::inconvertibleErrorCode());
165 close(ToExecutor[WriteEnd]);
166 close(FromExecutor[ReadEnd]);
172 std::unique_ptr<char[]> ExecutorPath, FDSpecifier;
174 ExecutorPath = std::make_unique<char[]>(ExecutablePath.size() + 1);
175 strcpy(ExecutorPath.get(), ExecutablePath.data());
177 std::string FDSpecifierStr(
"filedescs=");
178 FDSpecifierStr += llvm::utostr(ToExecutor[ReadEnd]);
179 FDSpecifierStr +=
',';
180 FDSpecifierStr += llvm::utostr(FromExecutor[WriteEnd]);
181 FDSpecifier = std::make_unique<char[]>(FDSpecifierStr.size() + 1);
182 strcpy(FDSpecifier.get(), FDSpecifierStr.c_str());
185 char *
const Args[] = {ExecutorPath.get(), FDSpecifier.get(),
nullptr};
186 int RC = execvp(ExecutorPath.get(), Args);
188 llvm::errs() <<
"unable to launch out-of-process executor \""
189 << ExecutorPath.get() <<
"\"\n";
196 close(ToExecutor[ReadEnd]);
197 close(FromExecutor[WriteEnd]);
199 llvm::orc::SimpleRemoteEPC::Setup S = llvm::orc::SimpleRemoteEPC::Setup();
201 S.CreateMemoryManager =
202 [SlabAllocateSize](llvm::orc::SimpleRemoteEPC &EPC) {
207 llvm::orc::SimpleRemoteEPC::Create<llvm::orc::FDSimpleRemoteEPCTransport>(
208 std::make_unique<llvm::orc::DynamicThreadPoolTaskDispatcher>(
210 std::move(S), FromExecutor[ReadEnd], ToExecutor[WriteEnd]);
212 return EPCOrErr.takeError();
213 return std::make_pair(std::move(*EPCOrErr), ChildPID);
217#if LLVM_ON_UNIX && LLVM_ENABLE_THREADS
220 std::string PortStr) {
223 Hints.ai_family = AF_INET;
224 Hints.ai_socktype = SOCK_STREAM;
225 Hints.ai_flags = AI_NUMERICSERV;
227 if (
int EC = getaddrinfo(
Host.c_str(), PortStr.c_str(), &Hints, &AI))
228 return llvm::make_error<llvm::StringError>(
229 llvm::formatv(
"address resolution failed ({0})", strerror(EC)),
230 llvm::inconvertibleErrorCode());
235 for (Server = AI; Server !=
nullptr; Server = Server->ai_next) {
238 if ((SockFD = socket(AI->ai_family, AI->ai_socktype, AI->ai_protocol)) < 0)
242 if (connect(SockFD, Server->ai_addr, Server->ai_addrlen) == 0)
251 if (Server ==
nullptr)
252 return llvm::make_error<llvm::StringError>(
"invalid hostname",
253 llvm::inconvertibleErrorCode());
259connectTCPSocket(llvm::StringRef NetworkAddress,
bool UseSharedMemory,
260 unsigned SlabAllocateSize) {
263 return llvm::make_error<llvm::StringError>(
264 "-" + NetworkAddress +
" not supported on non-unix platforms",
265 llvm::inconvertibleErrorCode());
266#elif !LLVM_ENABLE_THREADS
268 return llvm::make_error<llvm::StringError>(
269 "-" + NetworkAddress +
270 " requires threads, but LLVM was built with "
271 "LLVM_ENABLE_THREADS=Off",
272 llvm::inconvertibleErrorCode());
275 auto CreateErr = [NetworkAddress](Twine Details) {
276 return llvm::make_error<llvm::StringError>(
277 formatv(
"Failed to connect TCP socket '{0}': {1}", NetworkAddress,
279 llvm::inconvertibleErrorCode());
282 StringRef
Host, PortStr;
283 std::tie(Host, PortStr) = NetworkAddress.split(
':');
285 return CreateErr(
"Host name for -" + NetworkAddress +
" can not be empty");
287 return CreateErr(
"Port number in -" + NetworkAddress +
" can not be empty");
289 if (PortStr.getAsInteger(10, Port))
290 return CreateErr(
"Port number '" + PortStr +
"' is not a valid integer");
294 return SockFD.takeError();
296 llvm::orc::SimpleRemoteEPC::Setup S = llvm::orc::SimpleRemoteEPC::Setup();
298 S.CreateMemoryManager =
299 [SlabAllocateSize](llvm::orc::SimpleRemoteEPC &EPC) {
303 return llvm::orc::SimpleRemoteEPC::Create<
304 llvm::orc::FDSimpleRemoteEPCTransport>(
305 std::make_unique<llvm::orc::DynamicThreadPoolTaskDispatcher>(
307 std::move(S), *SockFD, *SockFD);
314 llvm::StringRef OrcRuntimePath) {
317 return JTMB.takeError();
320 return JB.takeError();
322 (*JB)->setExecutorProcessControl(std::move(EPC));
323 (*JB)->setPlatformSetUp(
324 llvm::orc::ExecutorNativePlatform(OrcRuntimePath.str()));
326 return std::move(*JB);
330 std::pair<std::unique_ptr<llvm::orc::LLJITBuilder>, uint32_t>>
332 std::unique_ptr<llvm::orc::ExecutorProcessControl> EPC;
333 uint32_t childPid = -1;
341 return ResultOrErr.takeError();
342 childPid = ResultOrErr->second;
343 auto EPCOrErr = std::move(ResultOrErr->first);
344 EPC = std::move(EPCOrErr);
346#if LLVM_ON_UNIX && LLVM_ENABLE_THREADS
351 return EPCOrErr.takeError();
352 EPC = std::move(*EPCOrErr);
354 return llvm::make_error<llvm::StringError>(
355 "Out-of-process JIT over TCP is not supported on this platform",
360 std::unique_ptr<llvm::orc::LLJITBuilder> JB;
365 return JBOrErr.takeError();
366 JB = std::move(*JBOrErr);
369 return std::make_pair(std::move(JB), childPid);
376 return std::move(
IE);
382 return ResOrErr.takeError();
387 return llvm::make_error<llvm::StringError>(
388 "Operation failed. No LLJITBuilder for out-of-process JIT",
395 return JTMB.takeError();
397 JTMB->setCodeModel(
CM);
400 return JB.takeError();
404 llvm::Error Err = llvm::Error::success();
405 std::unique_ptr<IncrementalExecutor> Executor;
407 Executor = std::make_unique<WasmIncrementalExecutor>(Err);
409 Executor = std::make_unique<OrcIncrementalExecutor>(TSC, *
JITBuilder, Err);
413 return std::move(Err);
415 return std::move(Executor);
418llvm::Error IncrementalExecutorBuilder::UpdateOrcRuntimePath(
421 return llvm::Error::success();
429 std::string CompilerRTPath = TC.
getCompilerRT(
C.getArgs(),
"orc_rt");
430 llvm::StringRef CanonicalFilename = llvm::sys::path::filename(CompilerRTPath);
432 if (CanonicalFilename.empty()) {
433 return llvm::make_error<llvm::StringError>(
434 "Could not determine OrcRuntime filename from ToolChain",
435 llvm::inconvertibleErrorCode());
438 OrcRTLibNames.push_back(CanonicalFilename.str());
441 llvm::StringRef LegacySuffix = CanonicalFilename;
442 if (LegacySuffix.consume_front(
"libclang_rt.")) {
443 OrcRTLibNames.push_back((
"lib" + LegacySuffix).str());
448 llvm::sys::path::remove_filename(OrcRTDir);
452 auto findInDir = [&](llvm::StringRef Dir) -> std::optional<std::string> {
453 for (
const auto &LibName : OrcRTLibNames) {
455 llvm::sys::path::append(FullPath, LibName);
456 if (llvm::sys::fs::exists(FullPath))
457 return std::string(FullPath.str());
458 triedPaths.push_back(std::string(FullPath.str()));
464 if (
auto Found = findInDir(OrcRTDir)) {
465 OrcRuntimePath = *
Found;
466 return llvm::Error::success();
471 llvm::StringRef Rel = OrcRTDir.str();
472 if (!Rel.consume_front(llvm::sys::path::parent_path(D.
Dir))) {
473 return llvm::make_error<llvm::StringError>(
474 llvm::formatv(
"OrcRuntime library path ({0}) is not located within the "
475 "Clang resource directory ({1}). Check your installation "
476 "or provide an explicit path via -resource-dir.",
479 llvm::inconvertibleErrorCode());
485 while (llvm::sys::path::has_parent_path(Cursor)) {
486 Cursor = llvm::sys::path::parent_path(Cursor).str();
488 llvm::sys::path::append(Candidate, Rel);
490 if (
auto Found = findInDir(Candidate)) {
491 OrcRuntimePath = *
Found;
492 return llvm::Error::success();
496 if (triedPaths.size() > 32)
502 for (
size_t i = 0; i < triedPaths.size(); ++i) {
505 Joined += triedPaths[i];
508 return llvm::make_error<llvm::StringError>(
509 llvm::formatv(
"OrcRuntime library not found. Checked: {0}",
510 Joined.empty() ?
"<none>" : Joined)
512 std::make_error_code(std::errc::no_such_file_or_directory));
std::optional< llvm::CodeModel::Model > CM
An optional code model to provide to the JITTargetMachineBuilder.
bool IsOutOfProcess
Indicates whether out-of-process JIT execution is enabled.
~IncrementalExecutorBuilder()
std::unique_ptr< IncrementalExecutor > IE
An optional external IncrementalExecutor.
std::function< void()> CustomizeFork
Custom lambda to be executed inside child process/executor.
std::string OOPExecutor
Path to the out-of-process JIT executor.
uint32_t ExecutorPID
PID of the out-of-process JIT executor.
std::string OOPExecutorConnect
bool UseSharedMemory
Indicates whether to use shared memory for communication.
llvm::Expected< std::unique_ptr< IncrementalExecutor > > create(llvm::orc::ThreadSafeContext &TSC, const clang::TargetInfo &TI)
std::string OrcRuntimePath
Path to the ORC runtime library.
std::unique_ptr< llvm::orc::LLJITBuilder > JITBuilder
An optional external orc jit builder.
unsigned SlabAllocateSize
Representing the slab allocation size for memory management in kb.
Exposes information about the current target.
const llvm::Triple & getTriple() const
Returns the target triple of the primary target.
Compilation - A set of tasks to perform for a single driver invocation.
Driver - Encapsulate logic for constructing compilation processes from a set of gcc-driver-like comma...
std::string Dir
The path the driver executable was in, as invoked from the command line.
Defines the clang::TargetInfo interface.
The JSON file list parser is used to communicate input to InstallAPI.
static llvm::Expected< std::unique_ptr< llvm::orc::LLJITBuilder > > createLLJITBuilder(std::unique_ptr< llvm::orc::ExecutorProcessControl > EPC, llvm::StringRef OrcRuntimePath)
static llvm::Expected< std::unique_ptr< llvm::orc::LLJITBuilder > > createDefaultJITBuilder(llvm::orc::JITTargetMachineBuilder JTMB)
static llvm::Expected< std::pair< std::unique_ptr< llvm::orc::LLJITBuilder >, uint32_t > > outOfProcessJITBuilder(const IncrementalExecutorBuilder &IncrExecutorBuilder)
Expected< std::unique_ptr< llvm::jitlink::JITLinkMemoryManager > > createSharedMemoryManager(llvm::orc::SimpleRemoteEPC &SREPC, unsigned SlabAllocateSize)
static llvm::Expected< llvm::orc::JITTargetMachineBuilder > createJITTargetMachineBuilder(const llvm::Triple &TT)
static llvm::Expected< std::pair< std::unique_ptr< llvm::orc::SimpleRemoteEPC >, uint32_t > > launchExecutor(llvm::StringRef ExecutablePath, bool UseSharedMemory, unsigned SlabAllocateSize, std::function< void()> CustomizeFork)
int const char * function