1bb72f073SLang Hames //===------- SimpleEPCServer.cpp - EPC over simple abstract channel -------===//
2bb72f073SLang Hames //
3bb72f073SLang Hames // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4bb72f073SLang Hames // See https://llvm.org/LICENSE.txt for license information.
5bb72f073SLang Hames // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6bb72f073SLang Hames //
7bb72f073SLang Hames //===----------------------------------------------------------------------===//
8bb72f073SLang Hames 
9bb72f073SLang Hames #include "llvm/ExecutionEngine/Orc/TargetProcess/SimpleRemoteEPCServer.h"
10bb72f073SLang Hames 
11bb72f073SLang Hames #include "llvm/ExecutionEngine/Orc/Shared/TargetProcessControlTypes.h"
12bb72f073SLang Hames #include "llvm/Support/FormatVariadic.h"
13bb72f073SLang Hames #include "llvm/Support/Host.h"
14bb72f073SLang Hames #include "llvm/Support/Process.h"
15bb72f073SLang Hames 
16*2c8e7849SLang Hames #include "OrcRTBootstrap.h"
17*2c8e7849SLang Hames 
18bb72f073SLang Hames #define DEBUG_TYPE "orc"
19bb72f073SLang Hames 
20bb72f073SLang Hames using namespace llvm::orc::shared;
21bb72f073SLang Hames 
22bb72f073SLang Hames namespace llvm {
23bb72f073SLang Hames namespace orc {
24bb72f073SLang Hames 
25bb72f073SLang Hames SimpleRemoteEPCServer::Dispatcher::~Dispatcher() {}
26bb72f073SLang Hames 
27bb72f073SLang Hames #if LLVM_ENABLE_THREADS
28bb72f073SLang Hames void SimpleRemoteEPCServer::ThreadDispatcher::dispatch(
29bb72f073SLang Hames     unique_function<void()> Work) {
30bb72f073SLang Hames   {
31bb72f073SLang Hames     std::lock_guard<std::mutex> Lock(DispatchMutex);
32bb72f073SLang Hames     if (!Running)
33bb72f073SLang Hames       return;
34bb72f073SLang Hames     ++Outstanding;
35bb72f073SLang Hames   }
36bb72f073SLang Hames 
37bb72f073SLang Hames   std::thread([this, Work = std::move(Work)]() mutable {
38bb72f073SLang Hames     Work();
39bb72f073SLang Hames     std::lock_guard<std::mutex> Lock(DispatchMutex);
40bb72f073SLang Hames     --Outstanding;
41bb72f073SLang Hames     OutstandingCV.notify_all();
42bb72f073SLang Hames   }).detach();
43bb72f073SLang Hames }
44bb72f073SLang Hames 
45bb72f073SLang Hames void SimpleRemoteEPCServer::ThreadDispatcher::shutdown() {
46bb72f073SLang Hames   std::unique_lock<std::mutex> Lock(DispatchMutex);
47bb72f073SLang Hames   Running = false;
48bb72f073SLang Hames   OutstandingCV.wait(Lock, [this]() { return Outstanding == 0; });
49bb72f073SLang Hames }
50bb72f073SLang Hames #endif
51bb72f073SLang Hames 
52bb72f073SLang Hames StringMap<ExecutorAddress> SimpleRemoteEPCServer::defaultBootstrapSymbols() {
53bb72f073SLang Hames   StringMap<ExecutorAddress> DBS;
54*2c8e7849SLang Hames   rt_bootstrap::addTo(DBS);
55bb72f073SLang Hames   DBS["__llvm_orc_load_dylib"] = ExecutorAddress::fromPtr(&loadDylibWrapper);
56bb72f073SLang Hames   DBS["__llvm_orc_lookup_symbols"] =
57bb72f073SLang Hames       ExecutorAddress::fromPtr(&lookupSymbolsWrapper);
58bb72f073SLang Hames   return DBS;
59bb72f073SLang Hames }
60bb72f073SLang Hames 
61bb72f073SLang Hames Expected<SimpleRemoteEPCTransportClient::HandleMessageAction>
62bb72f073SLang Hames SimpleRemoteEPCServer::handleMessage(SimpleRemoteEPCOpcode OpC, uint64_t SeqNo,
63bb72f073SLang Hames                                      ExecutorAddress TagAddr,
64bb72f073SLang Hames                                      SimpleRemoteEPCArgBytesVector ArgBytes) {
65bb72f073SLang Hames   using UT = std::underlying_type_t<SimpleRemoteEPCOpcode>;
66d11a0c5dSLang Hames   if (static_cast<UT>(OpC) > static_cast<UT>(SimpleRemoteEPCOpcode::LastOpC))
67bb72f073SLang Hames     return make_error<StringError>("Unexpected opcode",
68bb72f073SLang Hames                                    inconvertibleErrorCode());
69bb72f073SLang Hames 
70bb72f073SLang Hames   // TODO: Clean detach message?
71bb72f073SLang Hames   switch (OpC) {
72bb72f073SLang Hames   case SimpleRemoteEPCOpcode::Setup:
73bb72f073SLang Hames     return make_error<StringError>("Unexpected Setup opcode",
74bb72f073SLang Hames                                    inconvertibleErrorCode());
75bb72f073SLang Hames   case SimpleRemoteEPCOpcode::Hangup:
76bb72f073SLang Hames     return SimpleRemoteEPCTransportClient::EndSession;
77bb72f073SLang Hames   case SimpleRemoteEPCOpcode::Result:
78bb72f073SLang Hames     if (auto Err = handleResult(SeqNo, TagAddr, std::move(ArgBytes)))
79bb72f073SLang Hames       return std::move(Err);
80bb72f073SLang Hames     break;
81bb72f073SLang Hames   case SimpleRemoteEPCOpcode::CallWrapper:
82bb72f073SLang Hames     handleCallWrapper(SeqNo, TagAddr, std::move(ArgBytes));
83bb72f073SLang Hames     break;
84bb72f073SLang Hames   }
85bb72f073SLang Hames   return ContinueSession;
86bb72f073SLang Hames }
87bb72f073SLang Hames 
88bb72f073SLang Hames Error SimpleRemoteEPCServer::waitForDisconnect() {
89bb72f073SLang Hames   std::unique_lock<std::mutex> Lock(ServerStateMutex);
90bb72f073SLang Hames   ShutdownCV.wait(Lock, [this]() { return RunState == ServerShutDown; });
91bb72f073SLang Hames   return std::move(ShutdownErr);
92bb72f073SLang Hames }
93bb72f073SLang Hames 
94bb72f073SLang Hames void SimpleRemoteEPCServer::handleDisconnect(Error Err) {
95bb72f073SLang Hames   PendingJITDispatchResultsMap TmpPending;
96bb72f073SLang Hames 
97bb72f073SLang Hames   {
98bb72f073SLang Hames     std::lock_guard<std::mutex> Lock(ServerStateMutex);
99bb72f073SLang Hames     std::swap(TmpPending, PendingJITDispatchResults);
100bb72f073SLang Hames     RunState = ServerShuttingDown;
101bb72f073SLang Hames   }
102bb72f073SLang Hames 
103bb72f073SLang Hames   // Send out-of-band errors to any waiting threads.
104bb72f073SLang Hames   for (auto &KV : TmpPending)
105bb72f073SLang Hames     KV.second->set_value(
106bb72f073SLang Hames         shared::WrapperFunctionResult::createOutOfBandError("disconnecting"));
107bb72f073SLang Hames 
108bb72f073SLang Hames   // TODO: Free attached resources.
109bb72f073SLang Hames   // 1. Close libraries in DylibHandles.
110bb72f073SLang Hames 
111bb72f073SLang Hames   // Wait for dispatcher to clear.
112bb72f073SLang Hames   D->shutdown();
113bb72f073SLang Hames 
114bb72f073SLang Hames   std::lock_guard<std::mutex> Lock(ServerStateMutex);
115bb72f073SLang Hames   ShutdownErr = joinErrors(std::move(ShutdownErr), std::move(Err));
116bb72f073SLang Hames   RunState = ServerShutDown;
117bb72f073SLang Hames   ShutdownCV.notify_all();
118bb72f073SLang Hames }
119bb72f073SLang Hames 
120bb72f073SLang Hames Error SimpleRemoteEPCServer::sendSetupMessage(
121bb72f073SLang Hames     StringMap<ExecutorAddress> BootstrapSymbols) {
122bb72f073SLang Hames 
123bb72f073SLang Hames   using namespace SimpleRemoteEPCDefaultBootstrapSymbolNames;
124bb72f073SLang Hames 
125bb72f073SLang Hames   std::vector<char> SetupPacket;
126bb72f073SLang Hames   SimpleRemoteEPCExecutorInfo EI;
127bb72f073SLang Hames   EI.TargetTriple = sys::getProcessTriple();
128bb72f073SLang Hames   if (auto PageSize = sys::Process::getPageSize())
129bb72f073SLang Hames     EI.PageSize = *PageSize;
130bb72f073SLang Hames   else
131bb72f073SLang Hames     return PageSize.takeError();
132bb72f073SLang Hames   EI.BootstrapSymbols = std::move(BootstrapSymbols);
133bb72f073SLang Hames 
134bb72f073SLang Hames   assert(!EI.BootstrapSymbols.count(ExecutorSessionObjectName) &&
135bb72f073SLang Hames          "Dispatch context name should not be set");
136bb72f073SLang Hames   assert(!EI.BootstrapSymbols.count(DispatchFnName) &&
137bb72f073SLang Hames          "Dispatch function name should not be set");
138bb72f073SLang Hames   EI.BootstrapSymbols[ExecutorSessionObjectName] =
139bb72f073SLang Hames       ExecutorAddress::fromPtr(this);
140bb72f073SLang Hames   EI.BootstrapSymbols[DispatchFnName] =
141bb72f073SLang Hames       ExecutorAddress::fromPtr(jitDispatchEntry);
142bb72f073SLang Hames 
143bb72f073SLang Hames   using SPSSerialize =
144bb72f073SLang Hames       shared::SPSArgList<shared::SPSSimpleRemoteEPCExecutorInfo>;
145bb72f073SLang Hames   auto SetupPacketBytes =
146bb72f073SLang Hames       shared::WrapperFunctionResult::allocate(SPSSerialize::size(EI));
147bb72f073SLang Hames   shared::SPSOutputBuffer OB(SetupPacketBytes.data(), SetupPacketBytes.size());
148bb72f073SLang Hames   if (!SPSSerialize::serialize(OB, EI))
149bb72f073SLang Hames     return make_error<StringError>("Could not send setup packet",
150bb72f073SLang Hames                                    inconvertibleErrorCode());
151bb72f073SLang Hames 
152bb72f073SLang Hames   return T->sendMessage(SimpleRemoteEPCOpcode::Setup, 0, ExecutorAddress(),
153bb72f073SLang Hames                         {SetupPacketBytes.data(), SetupPacketBytes.size()});
154bb72f073SLang Hames }
155bb72f073SLang Hames 
156bb72f073SLang Hames Error SimpleRemoteEPCServer::handleResult(
157bb72f073SLang Hames     uint64_t SeqNo, ExecutorAddress TagAddr,
158bb72f073SLang Hames     SimpleRemoteEPCArgBytesVector ArgBytes) {
159bb72f073SLang Hames   std::promise<shared::WrapperFunctionResult> *P = nullptr;
160bb72f073SLang Hames   {
161bb72f073SLang Hames     std::lock_guard<std::mutex> Lock(ServerStateMutex);
162bb72f073SLang Hames     auto I = PendingJITDispatchResults.find(SeqNo);
163bb72f073SLang Hames     if (I == PendingJITDispatchResults.end())
164bb72f073SLang Hames       return make_error<StringError>("No call for sequence number " +
165bb72f073SLang Hames                                          Twine(SeqNo),
166bb72f073SLang Hames                                      inconvertibleErrorCode());
167bb72f073SLang Hames     P = I->second;
168bb72f073SLang Hames     PendingJITDispatchResults.erase(I);
169bb72f073SLang Hames     releaseSeqNo(SeqNo);
170bb72f073SLang Hames   }
171bb72f073SLang Hames   auto R = shared::WrapperFunctionResult::allocate(ArgBytes.size());
172bb72f073SLang Hames   memcpy(R.data(), ArgBytes.data(), ArgBytes.size());
173bb72f073SLang Hames   P->set_value(std::move(R));
174bb72f073SLang Hames   return Error::success();
175bb72f073SLang Hames }
176bb72f073SLang Hames 
177bb72f073SLang Hames void SimpleRemoteEPCServer::handleCallWrapper(
178bb72f073SLang Hames     uint64_t RemoteSeqNo, ExecutorAddress TagAddr,
179bb72f073SLang Hames     SimpleRemoteEPCArgBytesVector ArgBytes) {
180bb72f073SLang Hames   D->dispatch([this, RemoteSeqNo, TagAddr, ArgBytes = std::move(ArgBytes)]() {
181bb72f073SLang Hames     using WrapperFnTy =
182bb72f073SLang Hames         shared::detail::CWrapperFunctionResult (*)(const char *, size_t);
183bb72f073SLang Hames     auto *Fn = TagAddr.toPtr<WrapperFnTy>();
184bb72f073SLang Hames     shared::WrapperFunctionResult ResultBytes(
185bb72f073SLang Hames         Fn(ArgBytes.data(), ArgBytes.size()));
186bb72f073SLang Hames     if (auto Err = T->sendMessage(SimpleRemoteEPCOpcode::Result, RemoteSeqNo,
187bb72f073SLang Hames                                   ExecutorAddress(),
188bb72f073SLang Hames                                   {ResultBytes.data(), ResultBytes.size()}))
189bb72f073SLang Hames       ReportError(std::move(Err));
190bb72f073SLang Hames   });
191bb72f073SLang Hames }
192bb72f073SLang Hames 
193bb72f073SLang Hames shared::detail::CWrapperFunctionResult
194bb72f073SLang Hames SimpleRemoteEPCServer::loadDylibWrapper(const char *ArgData, size_t ArgSize) {
195bb72f073SLang Hames   return shared::WrapperFunction<shared::SPSLoadDylibSignature>::handle(
196bb72f073SLang Hames              ArgData, ArgSize,
197bb72f073SLang Hames              [](ExecutorAddress ExecutorSessionObj, std::string Path,
198bb72f073SLang Hames                 uint64_t Flags) -> Expected<uint64_t> {
199bb72f073SLang Hames                return ExecutorSessionObj.toPtr<SimpleRemoteEPCServer *>()
200bb72f073SLang Hames                    ->loadDylib(Path, Flags);
201bb72f073SLang Hames              })
202bb72f073SLang Hames       .release();
203bb72f073SLang Hames }
204bb72f073SLang Hames 
205bb72f073SLang Hames shared::detail::CWrapperFunctionResult
206bb72f073SLang Hames SimpleRemoteEPCServer::lookupSymbolsWrapper(const char *ArgData,
207bb72f073SLang Hames                                             size_t ArgSize) {
208bb72f073SLang Hames   return shared::WrapperFunction<shared::SPSLookupSymbolsSignature>::handle(
209bb72f073SLang Hames              ArgData, ArgSize,
210bb72f073SLang Hames              [](ExecutorAddress ExecutorSessionObj,
211bb72f073SLang Hames                 std::vector<RemoteSymbolLookup> Lookup) {
212bb72f073SLang Hames                return ExecutorSessionObj.toPtr<SimpleRemoteEPCServer *>()
213bb72f073SLang Hames                    ->lookupSymbols(Lookup);
214bb72f073SLang Hames              })
215bb72f073SLang Hames       .release();
216bb72f073SLang Hames }
217bb72f073SLang Hames 
218bb72f073SLang Hames Expected<tpctypes::DylibHandle>
219bb72f073SLang Hames SimpleRemoteEPCServer::loadDylib(const std::string &Path, uint64_t Mode) {
220bb72f073SLang Hames   std::string ErrMsg;
221bb72f073SLang Hames   const char *P = Path.empty() ? nullptr : Path.c_str();
222bb72f073SLang Hames   auto DL = sys::DynamicLibrary::getPermanentLibrary(P, &ErrMsg);
223bb72f073SLang Hames   if (!DL.isValid())
224bb72f073SLang Hames     return make_error<StringError>(std::move(ErrMsg), inconvertibleErrorCode());
225bb72f073SLang Hames   std::lock_guard<std::mutex> Lock(ServerStateMutex);
226bb72f073SLang Hames   uint64_t Id = Dylibs.size();
227bb72f073SLang Hames   Dylibs.push_back(std::move(DL));
228bb72f073SLang Hames   return Id;
229bb72f073SLang Hames }
230bb72f073SLang Hames 
231bb72f073SLang Hames Expected<std::vector<std::vector<ExecutorAddress>>>
232bb72f073SLang Hames SimpleRemoteEPCServer::lookupSymbols(const std::vector<RemoteSymbolLookup> &L) {
233bb72f073SLang Hames   std::vector<std::vector<ExecutorAddress>> Result;
234bb72f073SLang Hames 
235bb72f073SLang Hames   for (const auto &E : L) {
236bb72f073SLang Hames     if (E.H >= Dylibs.size())
237bb72f073SLang Hames       return make_error<StringError>("Unrecognized handle",
238bb72f073SLang Hames                                      inconvertibleErrorCode());
239bb72f073SLang Hames     auto &DL = Dylibs[E.H];
240bb72f073SLang Hames     Result.push_back({});
241bb72f073SLang Hames 
242bb72f073SLang Hames     for (const auto &Sym : E.Symbols) {
243bb72f073SLang Hames 
244bb72f073SLang Hames       const char *DemangledSymName = Sym.Name.c_str();
245bb72f073SLang Hames #ifdef __APPLE__
246bb72f073SLang Hames       if (*DemangledSymName == '_')
247bb72f073SLang Hames         ++DemangledSymName;
248bb72f073SLang Hames #endif
249bb72f073SLang Hames 
250bb72f073SLang Hames       void *Addr = DL.getAddressOfSymbol(DemangledSymName);
251bb72f073SLang Hames       if (!Addr && Sym.Required)
252bb72f073SLang Hames         return make_error<StringError>(Twine("Missing definition for ") +
253bb72f073SLang Hames                                            DemangledSymName,
254bb72f073SLang Hames                                        inconvertibleErrorCode());
255bb72f073SLang Hames 
256bb72f073SLang Hames       Result.back().push_back(ExecutorAddress::fromPtr(Addr));
257bb72f073SLang Hames     }
258bb72f073SLang Hames   }
259bb72f073SLang Hames 
260bb72f073SLang Hames   return std::move(Result);
261bb72f073SLang Hames }
262bb72f073SLang Hames 
263bb72f073SLang Hames shared::WrapperFunctionResult
264bb72f073SLang Hames SimpleRemoteEPCServer::doJITDispatch(const void *FnTag, const char *ArgData,
265bb72f073SLang Hames                                      size_t ArgSize) {
266bb72f073SLang Hames   uint64_t SeqNo;
267bb72f073SLang Hames   std::promise<shared::WrapperFunctionResult> ResultP;
268bb72f073SLang Hames   auto ResultF = ResultP.get_future();
269bb72f073SLang Hames   {
270bb72f073SLang Hames     std::lock_guard<std::mutex> Lock(ServerStateMutex);
271bb72f073SLang Hames     if (RunState != ServerRunning)
272bb72f073SLang Hames       return shared::WrapperFunctionResult::createOutOfBandError(
273bb72f073SLang Hames           "jit_dispatch not available (EPC server shut down)");
274bb72f073SLang Hames 
275bb72f073SLang Hames     SeqNo = getNextSeqNo();
276bb72f073SLang Hames     assert(!PendingJITDispatchResults.count(SeqNo) && "SeqNo already in use");
277bb72f073SLang Hames     PendingJITDispatchResults[SeqNo] = &ResultP;
278bb72f073SLang Hames   }
279bb72f073SLang Hames 
280bb72f073SLang Hames   if (auto Err =
281bb72f073SLang Hames           T->sendMessage(SimpleRemoteEPCOpcode::CallWrapper, SeqNo,
282bb72f073SLang Hames                          ExecutorAddress::fromPtr(FnTag), {ArgData, ArgSize}))
283bb72f073SLang Hames     ReportError(std::move(Err));
284bb72f073SLang Hames 
285bb72f073SLang Hames   return ResultF.get();
286bb72f073SLang Hames }
287bb72f073SLang Hames 
288bb72f073SLang Hames shared::detail::CWrapperFunctionResult
289bb72f073SLang Hames SimpleRemoteEPCServer::jitDispatchEntry(void *DispatchCtx, const void *FnTag,
290bb72f073SLang Hames                                         const char *ArgData, size_t ArgSize) {
291bb72f073SLang Hames   return reinterpret_cast<SimpleRemoteEPCServer *>(DispatchCtx)
292bb72f073SLang Hames       ->doJITDispatch(FnTag, ArgData, ArgSize)
293bb72f073SLang Hames       .release();
294bb72f073SLang Hames }
295bb72f073SLang Hames 
296bb72f073SLang Hames } // end namespace orc
297bb72f073SLang Hames } // end namespace llvm
298