forked from OSchip/llvm-project
[ORC] Remove Layer handles from the layer concept.
Handles were returned by addModule and used as keys for removeModule, findSymbolIn, and emitAndFinalize. Their job is now subsumed by VModuleKeys, which simplify resource management by providing a consistent handle across all layers. llvm-svn: 324700
This commit is contained in:
parent
6562b3d954
commit
0976cee8e9
|
@ -47,7 +47,6 @@ private:
|
||||||
IRCompileLayer<decltype(ObjectLayer), SimpleCompiler> CompileLayer;
|
IRCompileLayer<decltype(ObjectLayer), SimpleCompiler> CompileLayer;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
using ModuleHandle = decltype(CompileLayer)::ModuleHandleT;
|
|
||||||
|
|
||||||
KaleidoscopeJIT()
|
KaleidoscopeJIT()
|
||||||
: ES(SSP),
|
: ES(SSP),
|
||||||
|
@ -74,9 +73,11 @@ public:
|
||||||
|
|
||||||
TargetMachine &getTargetMachine() { return *TM; }
|
TargetMachine &getTargetMachine() { return *TM; }
|
||||||
|
|
||||||
ModuleHandle addModule(std::unique_ptr<Module> M) {
|
VModuleKey addModule(std::unique_ptr<Module> M) {
|
||||||
// Add the module to the JIT with a new VModuleKey.
|
// Add the module to the JIT with a new VModuleKey.
|
||||||
return cantFail(CompileLayer.addModule(ES.allocateVModule(), std::move(M)));
|
auto K = ES.allocateVModule();
|
||||||
|
cantFail(CompileLayer.addModule(K, std::move(M)));
|
||||||
|
return K;
|
||||||
}
|
}
|
||||||
|
|
||||||
JITSymbol findSymbol(const std::string Name) {
|
JITSymbol findSymbol(const std::string Name) {
|
||||||
|
@ -90,8 +91,8 @@ public:
|
||||||
return cantFail(findSymbol(Name).getAddress());
|
return cantFail(findSymbol(Name).getAddress());
|
||||||
}
|
}
|
||||||
|
|
||||||
void removeModule(ModuleHandle H) {
|
void removeModule(VModuleKey K) {
|
||||||
cantFail(CompileLayer.removeModule(H));
|
cantFail(CompileLayer.removeModule(K));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -56,7 +56,6 @@ private:
|
||||||
IRTransformLayer<decltype(CompileLayer), OptimizeFunction> OptimizeLayer;
|
IRTransformLayer<decltype(CompileLayer), OptimizeFunction> OptimizeLayer;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
using ModuleHandle = decltype(OptimizeLayer)::ModuleHandleT;
|
|
||||||
|
|
||||||
KaleidoscopeJIT()
|
KaleidoscopeJIT()
|
||||||
: ES(SSP),
|
: ES(SSP),
|
||||||
|
@ -86,10 +85,11 @@ public:
|
||||||
|
|
||||||
TargetMachine &getTargetMachine() { return *TM; }
|
TargetMachine &getTargetMachine() { return *TM; }
|
||||||
|
|
||||||
ModuleHandle addModule(std::unique_ptr<Module> M) {
|
VModuleKey addModule(std::unique_ptr<Module> M) {
|
||||||
// Add the module to the JIT with a new VModuleKey.
|
// Add the module to the JIT with a new VModuleKey.
|
||||||
return cantFail(
|
auto K = ES.allocateVModule();
|
||||||
OptimizeLayer.addModule(ES.allocateVModule(), std::move(M)));
|
cantFail(OptimizeLayer.addModule(K, std::move(M)));
|
||||||
|
return K;
|
||||||
}
|
}
|
||||||
|
|
||||||
JITSymbol findSymbol(const std::string Name) {
|
JITSymbol findSymbol(const std::string Name) {
|
||||||
|
@ -99,8 +99,8 @@ public:
|
||||||
return OptimizeLayer.findSymbol(MangledNameStream.str(), true);
|
return OptimizeLayer.findSymbol(MangledNameStream.str(), true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void removeModule(ModuleHandle H) {
|
void removeModule(VModuleKey K) {
|
||||||
cantFail(OptimizeLayer.removeModule(H));
|
cantFail(OptimizeLayer.removeModule(K));
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
|
@ -63,7 +63,6 @@ private:
|
||||||
CompileOnDemandLayer<decltype(OptimizeLayer)> CODLayer;
|
CompileOnDemandLayer<decltype(OptimizeLayer)> CODLayer;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
using ModuleHandle = decltype(CODLayer)::ModuleHandleT;
|
|
||||||
|
|
||||||
KaleidoscopeJIT()
|
KaleidoscopeJIT()
|
||||||
: ES(SSP), TM(EngineBuilder().selectTarget()), DL(TM->createDataLayout()),
|
: ES(SSP), TM(EngineBuilder().selectTarget()), DL(TM->createDataLayout()),
|
||||||
|
@ -92,7 +91,7 @@ public:
|
||||||
|
|
||||||
TargetMachine &getTargetMachine() { return *TM; }
|
TargetMachine &getTargetMachine() { return *TM; }
|
||||||
|
|
||||||
ModuleHandle addModule(std::unique_ptr<Module> M) {
|
VModuleKey addModule(std::unique_ptr<Module> M) {
|
||||||
// Create a new VModuleKey.
|
// Create a new VModuleKey.
|
||||||
VModuleKey K = ES.allocateVModule();
|
VModuleKey K = ES.allocateVModule();
|
||||||
|
|
||||||
|
@ -111,7 +110,8 @@ public:
|
||||||
[](Error Err) { cantFail(std::move(Err), "lookupFlags failed"); });
|
[](Error Err) { cantFail(std::move(Err), "lookupFlags failed"); });
|
||||||
|
|
||||||
// Add the module to the JIT with the new key.
|
// Add the module to the JIT with the new key.
|
||||||
return cantFail(CODLayer.addModule(K, std::move(M)));
|
cantFail(CODLayer.addModule(K, std::move(M)));
|
||||||
|
return K;
|
||||||
}
|
}
|
||||||
|
|
||||||
JITSymbol findSymbol(const std::string Name) {
|
JITSymbol findSymbol(const std::string Name) {
|
||||||
|
@ -121,8 +121,8 @@ public:
|
||||||
return CODLayer.findSymbol(MangledNameStream.str(), true);
|
return CODLayer.findSymbol(MangledNameStream.str(), true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void removeModule(ModuleHandle H) {
|
void removeModule(VModuleKey K) {
|
||||||
cantFail(CODLayer.removeModule(H));
|
cantFail(CODLayer.removeModule(K));
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
|
@ -89,7 +89,6 @@ private:
|
||||||
std::unique_ptr<IndirectStubsManager> IndirectStubsMgr;
|
std::unique_ptr<IndirectStubsManager> IndirectStubsMgr;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
using ModuleHandle = decltype(OptimizeLayer)::ModuleHandleT;
|
|
||||||
|
|
||||||
KaleidoscopeJIT()
|
KaleidoscopeJIT()
|
||||||
: ES(SSP),
|
: ES(SSP),
|
||||||
|
@ -127,10 +126,11 @@ public:
|
||||||
|
|
||||||
TargetMachine &getTargetMachine() { return *TM; }
|
TargetMachine &getTargetMachine() { return *TM; }
|
||||||
|
|
||||||
ModuleHandle addModule(std::unique_ptr<Module> M) {
|
VModuleKey addModule(std::unique_ptr<Module> M) {
|
||||||
// Add the module to the JIT with a new VModuleKey.
|
// Add the module to the JIT with a new VModuleKey.
|
||||||
return cantFail(
|
auto K = ES.allocateVModule();
|
||||||
OptimizeLayer.addModule(ES.allocateVModule(), std::move(M)));
|
cantFail(OptimizeLayer.addModule(K, std::move(M)));
|
||||||
|
return K;
|
||||||
}
|
}
|
||||||
|
|
||||||
Error addFunctionAST(std::unique_ptr<FunctionAST> FnAST) {
|
Error addFunctionAST(std::unique_ptr<FunctionAST> FnAST) {
|
||||||
|
@ -195,8 +195,8 @@ public:
|
||||||
return OptimizeLayer.findSymbol(mangle(Name), true);
|
return OptimizeLayer.findSymbol(mangle(Name), true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void removeModule(ModuleHandle H) {
|
void removeModule(VModuleKey K) {
|
||||||
cantFail(OptimizeLayer.removeModule(H));
|
cantFail(OptimizeLayer.removeModule(K));
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
|
@ -95,7 +95,6 @@ private:
|
||||||
MyRemote &Remote;
|
MyRemote &Remote;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
using ModuleHandle = decltype(OptimizeLayer)::ModuleHandleT;
|
|
||||||
|
|
||||||
KaleidoscopeJIT(MyRemote &Remote)
|
KaleidoscopeJIT(MyRemote &Remote)
|
||||||
: ES(SSP),
|
: ES(SSP),
|
||||||
|
@ -139,10 +138,11 @@ public:
|
||||||
|
|
||||||
TargetMachine &getTargetMachine() { return *TM; }
|
TargetMachine &getTargetMachine() { return *TM; }
|
||||||
|
|
||||||
ModuleHandle addModule(std::unique_ptr<Module> M) {
|
VModuleKey addModule(std::unique_ptr<Module> M) {
|
||||||
// Add the module with a new VModuleKey.
|
// Add the module with a new VModuleKey.
|
||||||
return cantFail(
|
auto K = ES.allocateVModule();
|
||||||
OptimizeLayer.addModule(ES.allocateVModule(), std::move(M)));
|
cantFail(OptimizeLayer.addModule(K, std::move(M)));
|
||||||
|
return K;
|
||||||
}
|
}
|
||||||
|
|
||||||
Error addFunctionAST(std::unique_ptr<FunctionAST> FnAST) {
|
Error addFunctionAST(std::unique_ptr<FunctionAST> FnAST) {
|
||||||
|
@ -211,8 +211,8 @@ public:
|
||||||
return OptimizeLayer.findSymbol(mangle(Name), true);
|
return OptimizeLayer.findSymbol(mangle(Name), true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void removeModule(ModuleHandle H) {
|
void removeModule(VModuleKey K) {
|
||||||
cantFail(OptimizeLayer.removeModule(H));
|
cantFail(OptimizeLayer.removeModule(K));
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
|
@ -42,7 +42,6 @@ class KaleidoscopeJIT {
|
||||||
public:
|
public:
|
||||||
using ObjLayerT = RTDyldObjectLinkingLayer;
|
using ObjLayerT = RTDyldObjectLinkingLayer;
|
||||||
using CompileLayerT = IRCompileLayer<ObjLayerT, SimpleCompiler>;
|
using CompileLayerT = IRCompileLayer<ObjLayerT, SimpleCompiler>;
|
||||||
using ModuleHandleT = CompileLayerT::ModuleHandleT;
|
|
||||||
|
|
||||||
KaleidoscopeJIT()
|
KaleidoscopeJIT()
|
||||||
: ES(SSP),
|
: ES(SSP),
|
||||||
|
@ -62,16 +61,16 @@ public:
|
||||||
|
|
||||||
TargetMachine &getTargetMachine() { return *TM; }
|
TargetMachine &getTargetMachine() { return *TM; }
|
||||||
|
|
||||||
ModuleHandleT addModule(std::unique_ptr<Module> M) {
|
VModuleKey addModule(std::unique_ptr<Module> M) {
|
||||||
auto H =
|
auto K = ES.allocateVModule();
|
||||||
cantFail(CompileLayer.addModule(ES.allocateVModule(), std::move(M)));
|
cantFail(CompileLayer.addModule(K, std::move(M)));
|
||||||
ModuleHandles.push_back(H);
|
ModuleKeys.push_back(K);
|
||||||
return H;
|
return K;
|
||||||
}
|
}
|
||||||
|
|
||||||
void removeModule(ModuleHandleT H) {
|
void removeModule(VModuleKey K) {
|
||||||
ModuleHandles.erase(find(ModuleHandles, H));
|
ModuleKeys.erase(find(ModuleKeys, K));
|
||||||
cantFail(CompileLayer.removeModule(H));
|
cantFail(CompileLayer.removeModule(K));
|
||||||
}
|
}
|
||||||
|
|
||||||
JITSymbol findSymbol(const std::string Name) {
|
JITSymbol findSymbol(const std::string Name) {
|
||||||
|
@ -105,7 +104,7 @@ private:
|
||||||
// Search modules in reverse order: from last added to first added.
|
// Search modules in reverse order: from last added to first added.
|
||||||
// This is the opposite of the usual search order for dlsym, but makes more
|
// This is the opposite of the usual search order for dlsym, but makes more
|
||||||
// sense in a REPL where we want to bind to the newest available definition.
|
// sense in a REPL where we want to bind to the newest available definition.
|
||||||
for (auto H : make_range(ModuleHandles.rbegin(), ModuleHandles.rend()))
|
for (auto H : make_range(ModuleKeys.rbegin(), ModuleKeys.rend()))
|
||||||
if (auto Sym = CompileLayer.findSymbolIn(H, Name, ExportedSymbolsOnly))
|
if (auto Sym = CompileLayer.findSymbolIn(H, Name, ExportedSymbolsOnly))
|
||||||
return Sym;
|
return Sym;
|
||||||
|
|
||||||
|
@ -133,7 +132,7 @@ private:
|
||||||
const DataLayout DL;
|
const DataLayout DL;
|
||||||
ObjLayerT ObjectLayer;
|
ObjLayerT ObjectLayer;
|
||||||
CompileLayerT CompileLayer;
|
CompileLayerT CompileLayer;
|
||||||
std::vector<ModuleHandleT> ModuleHandles;
|
std::vector<VModuleKey> ModuleKeys;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // end namespace orc
|
} // end namespace orc
|
||||||
|
|
|
@ -31,7 +31,7 @@ extern "C" {
|
||||||
|
|
||||||
typedef struct LLVMOpaqueSharedModule *LLVMSharedModuleRef;
|
typedef struct LLVMOpaqueSharedModule *LLVMSharedModuleRef;
|
||||||
typedef struct LLVMOrcOpaqueJITStack *LLVMOrcJITStackRef;
|
typedef struct LLVMOrcOpaqueJITStack *LLVMOrcJITStackRef;
|
||||||
typedef uint32_t LLVMOrcModuleHandle;
|
typedef uint64_t LLVMOrcModuleHandle;
|
||||||
typedef uint64_t LLVMOrcTargetAddress;
|
typedef uint64_t LLVMOrcTargetAddress;
|
||||||
typedef uint64_t (*LLVMOrcSymbolResolverFn)(const char *Name, void *LookupCtx);
|
typedef uint64_t (*LLVMOrcSymbolResolverFn)(const char *Name, void *LookupCtx);
|
||||||
typedef uint64_t (*LLVMOrcLazyCompileCallbackFn)(LLVMOrcJITStackRef JITStack,
|
typedef uint64_t (*LLVMOrcLazyCompileCallbackFn)(LLVMOrcJITStackRef JITStack,
|
||||||
|
|
|
@ -86,8 +86,6 @@ private:
|
||||||
return LambdaMaterializer<MaterializerFtor>(std::move(M));
|
return LambdaMaterializer<MaterializerFtor>(std::move(M));
|
||||||
}
|
}
|
||||||
|
|
||||||
using BaseLayerModuleHandleT = typename BaseLayerT::ModuleHandleT;
|
|
||||||
|
|
||||||
// Provide type-erasure for the Modules and MemoryManagers.
|
// Provide type-erasure for the Modules and MemoryManagers.
|
||||||
template <typename ResourceT>
|
template <typename ResourceT>
|
||||||
class ResourceOwner {
|
class ResourceOwner {
|
||||||
|
@ -147,6 +145,13 @@ private:
|
||||||
using SourceModulesList = std::vector<SourceModuleEntry>;
|
using SourceModulesList = std::vector<SourceModuleEntry>;
|
||||||
using SourceModuleHandle = typename SourceModulesList::size_type;
|
using SourceModuleHandle = typename SourceModulesList::size_type;
|
||||||
|
|
||||||
|
LogicalDylib() = default;
|
||||||
|
|
||||||
|
LogicalDylib(VModuleKey K, std::shared_ptr<SymbolResolver> BackingResolver,
|
||||||
|
std::unique_ptr<IndirectStubsMgrT> StubsMgr)
|
||||||
|
: K(std::move(K)), BackingResolver(std::move(BackingResolver)),
|
||||||
|
StubsMgr(std::move(StubsMgr)) {}
|
||||||
|
|
||||||
SourceModuleHandle
|
SourceModuleHandle
|
||||||
addSourceModule(std::shared_ptr<Module> M) {
|
addSourceModule(std::shared_ptr<Module> M) {
|
||||||
SourceModuleHandle H = SourceModules.size();
|
SourceModuleHandle H = SourceModules.size();
|
||||||
|
@ -167,8 +172,8 @@ private:
|
||||||
bool ExportedSymbolsOnly) {
|
bool ExportedSymbolsOnly) {
|
||||||
if (auto Sym = StubsMgr->findStub(Name, ExportedSymbolsOnly))
|
if (auto Sym = StubsMgr->findStub(Name, ExportedSymbolsOnly))
|
||||||
return Sym;
|
return Sym;
|
||||||
for (auto BLH : BaseLayerHandles)
|
for (auto BLK : BaseLayerVModuleKeys)
|
||||||
if (auto Sym = BaseLayer.findSymbolIn(BLH, Name, ExportedSymbolsOnly))
|
if (auto Sym = BaseLayer.findSymbolIn(BLK, Name, ExportedSymbolsOnly))
|
||||||
return Sym;
|
return Sym;
|
||||||
else if (auto Err = Sym.takeError())
|
else if (auto Err = Sym.takeError())
|
||||||
return std::move(Err);
|
return std::move(Err);
|
||||||
|
@ -176,8 +181,8 @@ private:
|
||||||
}
|
}
|
||||||
|
|
||||||
Error removeModulesFromBaseLayer(BaseLayerT &BaseLayer) {
|
Error removeModulesFromBaseLayer(BaseLayerT &BaseLayer) {
|
||||||
for (auto &BLH : BaseLayerHandles)
|
for (auto &BLK : BaseLayerVModuleKeys)
|
||||||
if (auto Err = BaseLayer.removeModule(BLH))
|
if (auto Err = BaseLayer.removeModule(BLK))
|
||||||
return Err;
|
return Err;
|
||||||
return Error::success();
|
return Error::success();
|
||||||
}
|
}
|
||||||
|
@ -187,16 +192,11 @@ private:
|
||||||
std::unique_ptr<IndirectStubsMgrT> StubsMgr;
|
std::unique_ptr<IndirectStubsMgrT> StubsMgr;
|
||||||
StaticGlobalRenamer StaticRenamer;
|
StaticGlobalRenamer StaticRenamer;
|
||||||
SourceModulesList SourceModules;
|
SourceModulesList SourceModules;
|
||||||
std::vector<BaseLayerModuleHandleT> BaseLayerHandles;
|
std::vector<VModuleKey> BaseLayerVModuleKeys;
|
||||||
};
|
};
|
||||||
|
|
||||||
using LogicalDylibList = std::list<LogicalDylib>;
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
/// @brief Handle to loaded module.
|
|
||||||
using ModuleHandleT = typename LogicalDylibList::iterator;
|
|
||||||
|
|
||||||
/// @brief Module partitioning functor.
|
/// @brief Module partitioning functor.
|
||||||
using PartitioningFtor = std::function<std::set<Function*>(Function&)>;
|
using PartitioningFtor = std::function<std::set<Function*>(Function&)>;
|
||||||
|
|
||||||
|
@ -228,36 +228,35 @@ public:
|
||||||
~CompileOnDemandLayer() {
|
~CompileOnDemandLayer() {
|
||||||
// FIXME: Report error on log.
|
// FIXME: Report error on log.
|
||||||
while (!LogicalDylibs.empty())
|
while (!LogicalDylibs.empty())
|
||||||
consumeError(removeModule(LogicalDylibs.begin()));
|
consumeError(removeModule(LogicalDylibs.begin()->first));
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Add a module to the compile-on-demand layer.
|
/// @brief Add a module to the compile-on-demand layer.
|
||||||
Expected<ModuleHandleT> addModule(VModuleKey K, std::shared_ptr<Module> M) {
|
Error addModule(VModuleKey K, std::shared_ptr<Module> M) {
|
||||||
|
|
||||||
LogicalDylibs.push_back(LogicalDylib());
|
assert(!LogicalDylibs.count(K) && "VModuleKey K already in use");
|
||||||
auto &LD = LogicalDylibs.back();
|
auto I = LogicalDylibs.insert(
|
||||||
LD.K = std::move(K);
|
LogicalDylibs.end(),
|
||||||
LD.StubsMgr = CreateIndirectStubsManager();
|
std::make_pair(K, LogicalDylib(K, GetSymbolResolver(K),
|
||||||
LD.BackingResolver = GetSymbolResolver(LD.K);
|
CreateIndirectStubsManager())));
|
||||||
|
|
||||||
if (auto Err = addLogicalModule(LD, std::move(M)))
|
return addLogicalModule(I->second, std::move(M));
|
||||||
return std::move(Err);
|
|
||||||
|
|
||||||
return std::prev(LogicalDylibs.end());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Add extra modules to an existing logical module.
|
/// @brief Add extra modules to an existing logical module.
|
||||||
Error addExtraModule(ModuleHandleT H, std::shared_ptr<Module> M) {
|
Error addExtraModule(VModuleKey K, std::shared_ptr<Module> M) {
|
||||||
return addLogicalModule(*H, std::move(M));
|
return addLogicalModule(LogicalDylibs[K], std::move(M));
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Remove the module represented by the given handle.
|
/// @brief Remove the module represented by the given key.
|
||||||
///
|
///
|
||||||
/// This will remove all modules in the layers below that were derived from
|
/// This will remove all modules in the layers below that were derived from
|
||||||
/// the module represented by H.
|
/// the module represented by K.
|
||||||
Error removeModule(ModuleHandleT H) {
|
Error removeModule(VModuleKey K) {
|
||||||
auto Err = H->removeModulesFromBaseLayer(BaseLayer);
|
auto I = LogicalDylibs.find(K);
|
||||||
LogicalDylibs.erase(H);
|
assert(I != LogicalDylibs.end() && "VModuleKey K not valid here");
|
||||||
|
auto Err = I->second.removeModulesFromBaseLayer(BaseLayer);
|
||||||
|
LogicalDylibs.erase(I);
|
||||||
return Err;
|
return Err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -266,11 +265,10 @@ public:
|
||||||
/// @param ExportedSymbolsOnly If true, search only for exported symbols.
|
/// @param ExportedSymbolsOnly If true, search only for exported symbols.
|
||||||
/// @return A handle for the given named symbol, if it exists.
|
/// @return A handle for the given named symbol, if it exists.
|
||||||
JITSymbol findSymbol(StringRef Name, bool ExportedSymbolsOnly) {
|
JITSymbol findSymbol(StringRef Name, bool ExportedSymbolsOnly) {
|
||||||
for (auto LDI = LogicalDylibs.begin(), LDE = LogicalDylibs.end();
|
for (auto &KV : LogicalDylibs) {
|
||||||
LDI != LDE; ++LDI) {
|
if (auto Sym = KV.second.StubsMgr->findStub(Name, ExportedSymbolsOnly))
|
||||||
if (auto Sym = LDI->StubsMgr->findStub(Name, ExportedSymbolsOnly))
|
|
||||||
return Sym;
|
return Sym;
|
||||||
if (auto Sym = findSymbolIn(LDI, Name, ExportedSymbolsOnly))
|
if (auto Sym = findSymbolIn(KV.first, Name, ExportedSymbolsOnly))
|
||||||
return Sym;
|
return Sym;
|
||||||
else if (auto Err = Sym.takeError())
|
else if (auto Err = Sym.takeError())
|
||||||
return std::move(Err);
|
return std::move(Err);
|
||||||
|
@ -280,9 +278,10 @@ public:
|
||||||
|
|
||||||
/// @brief Get the address of a symbol provided by this layer, or some layer
|
/// @brief Get the address of a symbol provided by this layer, or some layer
|
||||||
/// below this one.
|
/// below this one.
|
||||||
JITSymbol findSymbolIn(ModuleHandleT H, const std::string &Name,
|
JITSymbol findSymbolIn(VModuleKey K, const std::string &Name,
|
||||||
bool ExportedSymbolsOnly) {
|
bool ExportedSymbolsOnly) {
|
||||||
return H->findSymbol(BaseLayer, Name, ExportedSymbolsOnly);
|
assert(LogicalDylibs.count(K) && "VModuleKey K is not valid here");
|
||||||
|
return LogicalDylibs[K].findSymbol(BaseLayer, Name, ExportedSymbolsOnly);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Update the stub for the given function to point at FnBodyAddr.
|
/// @brief Update the stub for the given function to point at FnBodyAddr.
|
||||||
|
@ -502,10 +501,10 @@ private:
|
||||||
|
|
||||||
SetSymbolResolver(LD.K, std::move(GVsResolver));
|
SetSymbolResolver(LD.K, std::move(GVsResolver));
|
||||||
|
|
||||||
if (auto GVsHOrErr = BaseLayer.addModule(LD.K, std::move(GVsM)))
|
if (auto Err = BaseLayer.addModule(LD.K, std::move(GVsM)))
|
||||||
LD.BaseLayerHandles.push_back(*GVsHOrErr);
|
return Err;
|
||||||
else
|
|
||||||
return GVsHOrErr.takeError();
|
LD.BaseLayerVModuleKeys.push_back(LD.K);
|
||||||
|
|
||||||
return Error::success();
|
return Error::success();
|
||||||
}
|
}
|
||||||
|
@ -534,11 +533,11 @@ private:
|
||||||
|
|
||||||
JITTargetAddress CalledAddr = 0;
|
JITTargetAddress CalledAddr = 0;
|
||||||
auto Part = Partition(F);
|
auto Part = Partition(F);
|
||||||
if (auto PartHOrErr = emitPartition(LD, LMId, Part)) {
|
if (auto PartKeyOrErr = emitPartition(LD, LMId, Part)) {
|
||||||
auto &PartH = *PartHOrErr;
|
auto &PartKey = *PartKeyOrErr;
|
||||||
for (auto *SubF : Part) {
|
for (auto *SubF : Part) {
|
||||||
std::string FnName = mangle(SubF->getName(), SrcM.getDataLayout());
|
std::string FnName = mangle(SubF->getName(), SrcM.getDataLayout());
|
||||||
if (auto FnBodySym = BaseLayer.findSymbolIn(PartH, FnName, false)) {
|
if (auto FnBodySym = BaseLayer.findSymbolIn(PartKey, FnName, false)) {
|
||||||
if (auto FnBodyAddrOrErr = FnBodySym.getAddress()) {
|
if (auto FnBodyAddrOrErr = FnBodySym.getAddress()) {
|
||||||
JITTargetAddress FnBodyAddr = *FnBodyAddrOrErr;
|
JITTargetAddress FnBodyAddr = *FnBodyAddrOrErr;
|
||||||
|
|
||||||
|
@ -559,15 +558,15 @@ private:
|
||||||
llvm_unreachable("Function not emitted for partition");
|
llvm_unreachable("Function not emitted for partition");
|
||||||
}
|
}
|
||||||
|
|
||||||
LD.BaseLayerHandles.push_back(PartH);
|
LD.BaseLayerVModuleKeys.push_back(PartKey);
|
||||||
} else
|
} else
|
||||||
return PartHOrErr.takeError();
|
return PartKeyOrErr.takeError();
|
||||||
|
|
||||||
return CalledAddr;
|
return CalledAddr;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename PartitionT>
|
template <typename PartitionT>
|
||||||
Expected<BaseLayerModuleHandleT>
|
Expected<VModuleKey>
|
||||||
emitPartition(LogicalDylib &LD,
|
emitPartition(LogicalDylib &LD,
|
||||||
typename LogicalDylib::SourceModuleHandle LMId,
|
typename LogicalDylib::SourceModuleHandle LMId,
|
||||||
const PartitionT &Part) {
|
const PartitionT &Part) {
|
||||||
|
@ -658,7 +657,10 @@ private:
|
||||||
});
|
});
|
||||||
SetSymbolResolver(K, std::move(Resolver));
|
SetSymbolResolver(K, std::move(Resolver));
|
||||||
|
|
||||||
return BaseLayer.addModule(std::move(K), std::move(M));
|
if (auto Err = BaseLayer.addModule(std::move(K), std::move(M)))
|
||||||
|
return std::move(Err);
|
||||||
|
|
||||||
|
return K;
|
||||||
}
|
}
|
||||||
|
|
||||||
ExecutionSession &ES;
|
ExecutionSession &ES;
|
||||||
|
@ -669,7 +671,7 @@ private:
|
||||||
CompileCallbackMgrT &CompileCallbackMgr;
|
CompileCallbackMgrT &CompileCallbackMgr;
|
||||||
IndirectStubsManagerBuilderT CreateIndirectStubsManager;
|
IndirectStubsManagerBuilderT CreateIndirectStubsManager;
|
||||||
|
|
||||||
LogicalDylibList LogicalDylibs;
|
std::map<VModuleKey, LogicalDylib> LogicalDylibs;
|
||||||
bool CloneStubsIntoPartitions;
|
bool CloneStubsIntoPartitions;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -17,13 +17,14 @@
|
||||||
#include "llvm/ADT/StringMap.h"
|
#include "llvm/ADT/StringMap.h"
|
||||||
#include "llvm/ADT/iterator_range.h"
|
#include "llvm/ADT/iterator_range.h"
|
||||||
#include "llvm/ExecutionEngine/JITSymbol.h"
|
#include "llvm/ExecutionEngine/JITSymbol.h"
|
||||||
#include "llvm/ExecutionEngine/RuntimeDyld.h"
|
#include "llvm/ExecutionEngine/Orc/Core.h"
|
||||||
#include "llvm/ExecutionEngine/Orc/OrcError.h"
|
#include "llvm/ExecutionEngine/Orc/OrcError.h"
|
||||||
|
#include "llvm/ExecutionEngine/RuntimeDyld.h"
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
|
||||||
#include <utility>
|
#include <utility>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
namespace llvm {
|
namespace llvm {
|
||||||
|
|
||||||
|
@ -95,9 +96,8 @@ class CtorDtorRunner {
|
||||||
public:
|
public:
|
||||||
/// @brief Construct a CtorDtorRunner for the given range using the given
|
/// @brief Construct a CtorDtorRunner for the given range using the given
|
||||||
/// name mangling function.
|
/// name mangling function.
|
||||||
CtorDtorRunner(std::vector<std::string> CtorDtorNames,
|
CtorDtorRunner(std::vector<std::string> CtorDtorNames, VModuleKey K)
|
||||||
typename JITLayerT::ModuleHandleT H)
|
: CtorDtorNames(std::move(CtorDtorNames)), K(K) {}
|
||||||
: CtorDtorNames(std::move(CtorDtorNames)), H(H) {}
|
|
||||||
|
|
||||||
/// @brief Run the recorded constructors/destructors through the given JIT
|
/// @brief Run the recorded constructors/destructors through the given JIT
|
||||||
/// layer.
|
/// layer.
|
||||||
|
@ -106,7 +106,7 @@ public:
|
||||||
|
|
||||||
for (const auto &CtorDtorName : CtorDtorNames) {
|
for (const auto &CtorDtorName : CtorDtorNames) {
|
||||||
dbgs() << "Searching for ctor/dtor: " << CtorDtorName << "...";
|
dbgs() << "Searching for ctor/dtor: " << CtorDtorName << "...";
|
||||||
if (auto CtorDtorSym = JITLayer.findSymbolIn(H, CtorDtorName, false)) {
|
if (auto CtorDtorSym = JITLayer.findSymbolIn(K, CtorDtorName, false)) {
|
||||||
dbgs() << " found symbol...";
|
dbgs() << " found symbol...";
|
||||||
if (auto AddrOrErr = CtorDtorSym.getAddress()) {
|
if (auto AddrOrErr = CtorDtorSym.getAddress()) {
|
||||||
dbgs() << " at addr " << format("0x%016x", *AddrOrErr) << "\n";
|
dbgs() << " at addr " << format("0x%016x", *AddrOrErr) << "\n";
|
||||||
|
@ -130,7 +130,7 @@ public:
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::vector<std::string> CtorDtorNames;
|
std::vector<std::string> CtorDtorNames;
|
||||||
typename JITLayerT::ModuleHandleT H;
|
orc::VModuleKey K;
|
||||||
};
|
};
|
||||||
|
|
||||||
/// @brief Support class for static dtor execution. For hosted (in-process) JITs
|
/// @brief Support class for static dtor execution. For hosted (in-process) JITs
|
||||||
|
|
|
@ -36,9 +36,6 @@ template <typename BaseLayerT, typename CompileFtor>
|
||||||
class IRCompileLayer {
|
class IRCompileLayer {
|
||||||
public:
|
public:
|
||||||
|
|
||||||
/// @brief Handle to a compiled module.
|
|
||||||
using ModuleHandleT = typename BaseLayerT::ObjHandleT;
|
|
||||||
|
|
||||||
/// @brief Construct an IRCompileLayer with the given BaseLayer, which must
|
/// @brief Construct an IRCompileLayer with the given BaseLayer, which must
|
||||||
/// implement the ObjectLayer concept.
|
/// implement the ObjectLayer concept.
|
||||||
IRCompileLayer(BaseLayerT &BaseLayer, CompileFtor Compile)
|
IRCompileLayer(BaseLayerT &BaseLayer, CompileFtor Compile)
|
||||||
|
@ -49,18 +46,14 @@ public:
|
||||||
|
|
||||||
/// @brief Compile the module, and add the resulting object to the base layer
|
/// @brief Compile the module, and add the resulting object to the base layer
|
||||||
/// along with the given memory manager and symbol resolver.
|
/// along with the given memory manager and symbol resolver.
|
||||||
///
|
Error addModule(VModuleKey K, std::shared_ptr<Module> M) {
|
||||||
/// @return A handle for the added module.
|
|
||||||
Expected<ModuleHandleT> addModule(VModuleKey K, std::shared_ptr<Module> M) {
|
|
||||||
using CompileResult = decltype(Compile(*M));
|
using CompileResult = decltype(Compile(*M));
|
||||||
auto Obj = std::make_shared<CompileResult>(Compile(*M));
|
auto Obj = std::make_shared<CompileResult>(Compile(*M));
|
||||||
return BaseLayer.addObject(std::move(K), std::move(Obj));
|
return BaseLayer.addObject(std::move(K), std::move(Obj));
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Remove the module associated with the handle H.
|
/// @brief Remove the module associated with the VModuleKey K.
|
||||||
Error removeModule(ModuleHandleT H) {
|
Error removeModule(VModuleKey K) { return BaseLayer.removeObject(K); }
|
||||||
return BaseLayer.removeObject(H);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// @brief Search for the given named symbol.
|
/// @brief Search for the given named symbol.
|
||||||
/// @param Name The name of the symbol to search for.
|
/// @param Name The name of the symbol to search for.
|
||||||
|
@ -73,22 +66,20 @@ public:
|
||||||
/// @brief Get the address of the given symbol in compiled module represented
|
/// @brief Get the address of the given symbol in compiled module represented
|
||||||
/// by the handle H. This call is forwarded to the base layer's
|
/// by the handle H. This call is forwarded to the base layer's
|
||||||
/// implementation.
|
/// implementation.
|
||||||
/// @param H The handle for the module to search in.
|
/// @param K The VModuleKey for the module to search in.
|
||||||
/// @param Name The name of the symbol to search for.
|
/// @param Name The name of the symbol to search for.
|
||||||
/// @param ExportedSymbolsOnly If true, search only for exported symbols.
|
/// @param ExportedSymbolsOnly If true, search only for exported symbols.
|
||||||
/// @return A handle for the given named symbol, if it is found in the
|
/// @return A handle for the given named symbol, if it is found in the
|
||||||
/// given module.
|
/// given module.
|
||||||
JITSymbol findSymbolIn(ModuleHandleT H, const std::string &Name,
|
JITSymbol findSymbolIn(VModuleKey K, const std::string &Name,
|
||||||
bool ExportedSymbolsOnly) {
|
bool ExportedSymbolsOnly) {
|
||||||
return BaseLayer.findSymbolIn(H, Name, ExportedSymbolsOnly);
|
return BaseLayer.findSymbolIn(K, Name, ExportedSymbolsOnly);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Immediately emit and finalize the module represented by the given
|
/// @brief Immediately emit and finalize the module represented by the given
|
||||||
/// handle.
|
/// handle.
|
||||||
/// @param H Handle for module to emit/finalize.
|
/// @param H Handle for module to emit/finalize.
|
||||||
Error emitAndFinalize(ModuleHandleT H) {
|
Error emitAndFinalize(VModuleKey K) { return BaseLayer.emitAndFinalize(K); }
|
||||||
return BaseLayer.emitAndFinalize(H);
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
BaseLayerT &BaseLayer;
|
BaseLayerT &BaseLayer;
|
||||||
|
|
|
@ -31,9 +31,6 @@ template <typename BaseLayerT, typename TransformFtor>
|
||||||
class IRTransformLayer {
|
class IRTransformLayer {
|
||||||
public:
|
public:
|
||||||
|
|
||||||
/// @brief Handle to a set of added modules.
|
|
||||||
using ModuleHandleT = typename BaseLayerT::ModuleHandleT;
|
|
||||||
|
|
||||||
/// @brief Construct an IRTransformLayer with the given BaseLayer
|
/// @brief Construct an IRTransformLayer with the given BaseLayer
|
||||||
IRTransformLayer(BaseLayerT &BaseLayer,
|
IRTransformLayer(BaseLayerT &BaseLayer,
|
||||||
TransformFtor Transform = TransformFtor())
|
TransformFtor Transform = TransformFtor())
|
||||||
|
@ -43,12 +40,12 @@ public:
|
||||||
/// the layer below, along with the memory manager and symbol resolver.
|
/// the layer below, along with the memory manager and symbol resolver.
|
||||||
///
|
///
|
||||||
/// @return A handle for the added modules.
|
/// @return A handle for the added modules.
|
||||||
Expected<ModuleHandleT> addModule(VModuleKey K, std::shared_ptr<Module> M) {
|
Error addModule(VModuleKey K, std::shared_ptr<Module> M) {
|
||||||
return BaseLayer.addModule(std::move(K), Transform(std::move(M)));
|
return BaseLayer.addModule(std::move(K), Transform(std::move(M)));
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Remove the module associated with the handle H.
|
/// @brief Remove the module associated with the VModuleKey K.
|
||||||
Error removeModule(ModuleHandleT H) { return BaseLayer.removeModule(H); }
|
Error removeModule(VModuleKey K) { return BaseLayer.removeModule(K); }
|
||||||
|
|
||||||
/// @brief Search for the given named symbol.
|
/// @brief Search for the given named symbol.
|
||||||
/// @param Name The name of the symbol to search for.
|
/// @param Name The name of the symbol to search for.
|
||||||
|
@ -59,24 +56,22 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Get the address of the given symbol in the context of the module
|
/// @brief Get the address of the given symbol in the context of the module
|
||||||
/// represented by the handle H. This call is forwarded to the base
|
/// represented by the VModuleKey K. This call is forwarded to the base
|
||||||
/// layer's implementation.
|
/// layer's implementation.
|
||||||
/// @param H The handle for the module to search in.
|
/// @param H The handle for the module to search in.
|
||||||
/// @param Name The name of the symbol to search for.
|
/// @param Name The name of the symbol to search for.
|
||||||
/// @param ExportedSymbolsOnly If true, search only for exported symbols.
|
/// @param ExportedSymbolsOnly If true, search only for exported symbols.
|
||||||
/// @return A handle for the given named symbol, if it is found in the
|
/// @return A handle for the given named symbol, if it is found in the
|
||||||
/// given module.
|
/// given module.
|
||||||
JITSymbol findSymbolIn(ModuleHandleT H, const std::string &Name,
|
JITSymbol findSymbolIn(VModuleKey K, const std::string &Name,
|
||||||
bool ExportedSymbolsOnly) {
|
bool ExportedSymbolsOnly) {
|
||||||
return BaseLayer.findSymbolIn(H, Name, ExportedSymbolsOnly);
|
return BaseLayer.findSymbolIn(K, Name, ExportedSymbolsOnly);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Immediately emit and finalize the module represented by the given
|
/// @brief Immediately emit and finalize the module represented by the given
|
||||||
/// handle.
|
/// VModuleKey.
|
||||||
/// @param H Handle for module to emit/finalize.
|
/// @param H Handle for module to emit/finalize.
|
||||||
Error emitAndFinalize(ModuleHandleT H) {
|
Error emitAndFinalize(VModuleKey K) { return BaseLayer.emitAndFinalize(K); }
|
||||||
return BaseLayer.emitAndFinalize(H);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// @brief Access the transform functor directly.
|
/// @brief Access the transform functor directly.
|
||||||
TransformFtor& getTransform() { return Transform; }
|
TransformFtor& getTransform() { return Transform; }
|
||||||
|
|
|
@ -40,10 +40,6 @@ namespace orc {
|
||||||
/// is deferred until the first time the client requests the address (via
|
/// is deferred until the first time the client requests the address (via
|
||||||
/// JITSymbol::getAddress) for a symbol contained in this layer.
|
/// JITSymbol::getAddress) for a symbol contained in this layer.
|
||||||
template <typename BaseLayerT> class LazyEmittingLayer {
|
template <typename BaseLayerT> class LazyEmittingLayer {
|
||||||
public:
|
|
||||||
|
|
||||||
using BaseLayerHandleT = typename BaseLayerT::ModuleHandleT;
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
class EmissionDeferredModule {
|
class EmissionDeferredModule {
|
||||||
public:
|
public:
|
||||||
|
@ -65,13 +61,11 @@ private:
|
||||||
return 0;
|
return 0;
|
||||||
else if (this->EmitState == NotEmitted) {
|
else if (this->EmitState == NotEmitted) {
|
||||||
this->EmitState = Emitting;
|
this->EmitState = Emitting;
|
||||||
if (auto HandleOrErr = this->emitToBaseLayer(B))
|
if (auto Err = this->emitToBaseLayer(B))
|
||||||
Handle = std::move(*HandleOrErr);
|
return std::move(Err);
|
||||||
else
|
|
||||||
return HandleOrErr.takeError();
|
|
||||||
this->EmitState = Emitted;
|
this->EmitState = Emitted;
|
||||||
}
|
}
|
||||||
if (auto Sym = B.findSymbolIn(Handle, PName, ExportedSymbolsOnly))
|
if (auto Sym = B.findSymbolIn(K, PName, ExportedSymbolsOnly))
|
||||||
return Sym.getAddress();
|
return Sym.getAddress();
|
||||||
else if (auto Err = Sym.takeError())
|
else if (auto Err = Sym.takeError())
|
||||||
return std::move(Err);
|
return std::move(Err);
|
||||||
|
@ -89,13 +83,13 @@ private:
|
||||||
// RuntimeDyld that did the lookup), so just return a nullptr here.
|
// RuntimeDyld that did the lookup), so just return a nullptr here.
|
||||||
return nullptr;
|
return nullptr;
|
||||||
case Emitted:
|
case Emitted:
|
||||||
return B.findSymbolIn(Handle, Name, ExportedSymbolsOnly);
|
return B.findSymbolIn(K, Name, ExportedSymbolsOnly);
|
||||||
}
|
}
|
||||||
llvm_unreachable("Invalid emit-state.");
|
llvm_unreachable("Invalid emit-state.");
|
||||||
}
|
}
|
||||||
|
|
||||||
Error removeModuleFromBaseLayer(BaseLayerT& BaseLayer) {
|
Error removeModuleFromBaseLayer(BaseLayerT& BaseLayer) {
|
||||||
return EmitState != NotEmitted ? BaseLayer.removeModule(Handle)
|
return EmitState != NotEmitted ? BaseLayer.removeModule(K)
|
||||||
: Error::success();
|
: Error::success();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -104,10 +98,10 @@ private:
|
||||||
"Cannot emitAndFinalize while already emitting");
|
"Cannot emitAndFinalize while already emitting");
|
||||||
if (EmitState == NotEmitted) {
|
if (EmitState == NotEmitted) {
|
||||||
EmitState = Emitting;
|
EmitState = Emitting;
|
||||||
Handle = emitToBaseLayer(BaseLayer);
|
emitToBaseLayer(BaseLayer);
|
||||||
EmitState = Emitted;
|
EmitState = Emitted;
|
||||||
}
|
}
|
||||||
BaseLayer.emitAndFinalize(Handle);
|
BaseLayer.emitAndFinalize(K);
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -135,7 +129,7 @@ private:
|
||||||
return buildMangledSymbols(Name, ExportedSymbolsOnly);
|
return buildMangledSymbols(Name, ExportedSymbolsOnly);
|
||||||
}
|
}
|
||||||
|
|
||||||
Expected<BaseLayerHandleT> emitToBaseLayer(BaseLayerT &BaseLayer) {
|
Error emitToBaseLayer(BaseLayerT &BaseLayer) {
|
||||||
// We don't need the mangled names set any more: Once we've emitted this
|
// We don't need the mangled names set any more: Once we've emitted this
|
||||||
// to the base layer we'll just look for symbols there.
|
// to the base layer we'll just look for symbols there.
|
||||||
MangledSymbols.reset();
|
MangledSymbols.reset();
|
||||||
|
@ -192,40 +186,37 @@ private:
|
||||||
}
|
}
|
||||||
|
|
||||||
enum { NotEmitted, Emitting, Emitted } EmitState = NotEmitted;
|
enum { NotEmitted, Emitting, Emitted } EmitState = NotEmitted;
|
||||||
BaseLayerHandleT Handle;
|
|
||||||
VModuleKey K;
|
VModuleKey K;
|
||||||
std::shared_ptr<Module> M;
|
std::shared_ptr<Module> M;
|
||||||
mutable std::unique_ptr<StringMap<const GlobalValue*>> MangledSymbols;
|
mutable std::unique_ptr<StringMap<const GlobalValue*>> MangledSymbols;
|
||||||
};
|
};
|
||||||
|
|
||||||
using ModuleListT = std::list<std::unique_ptr<EmissionDeferredModule>>;
|
|
||||||
|
|
||||||
BaseLayerT &BaseLayer;
|
BaseLayerT &BaseLayer;
|
||||||
ModuleListT ModuleList;
|
std::map<VModuleKey, std::unique_ptr<EmissionDeferredModule>> ModuleMap;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
/// @brief Handle to a loaded module.
|
|
||||||
using ModuleHandleT = typename ModuleListT::iterator;
|
|
||||||
|
|
||||||
/// @brief Construct a lazy emitting layer.
|
/// @brief Construct a lazy emitting layer.
|
||||||
LazyEmittingLayer(BaseLayerT &BaseLayer) : BaseLayer(BaseLayer) {}
|
LazyEmittingLayer(BaseLayerT &BaseLayer) : BaseLayer(BaseLayer) {}
|
||||||
|
|
||||||
/// @brief Add the given module to the lazy emitting layer.
|
/// @brief Add the given module to the lazy emitting layer.
|
||||||
Expected<ModuleHandleT> addModule(VModuleKey K, std::shared_ptr<Module> M) {
|
Error addModule(VModuleKey K, std::shared_ptr<Module> M) {
|
||||||
return ModuleList.insert(
|
assert(!ModuleMap.count(K) && "VModuleKey K already in use");
|
||||||
ModuleList.end(),
|
ModuleMap[K] =
|
||||||
llvm::make_unique<EmissionDeferredModule>(std::move(K), std::move(M)));
|
llvm::make_unique<EmissionDeferredModule>(std::move(K), std::move(M));
|
||||||
|
return Error::success();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Remove the module represented by the given handle.
|
/// @brief Remove the module represented by the given handle.
|
||||||
///
|
///
|
||||||
/// This method will free the memory associated with the given module, both
|
/// This method will free the memory associated with the given module, both
|
||||||
/// in this layer, and the base layer.
|
/// in this layer, and the base layer.
|
||||||
Error removeModule(ModuleHandleT H) {
|
Error removeModule(VModuleKey K) {
|
||||||
Error Err = (*H)->removeModuleFromBaseLayer(BaseLayer);
|
auto I = ModuleMap.find(K);
|
||||||
ModuleList.erase(H);
|
assert(I != ModuleMap.end() && "VModuleKey K not valid here");
|
||||||
return Err;
|
auto EDM = std::move(I.second);
|
||||||
|
ModuleMap.erase(I);
|
||||||
|
return EDM->removeModuleFromBaseLayer(BaseLayer);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Search for the given named symbol.
|
/// @brief Search for the given named symbol.
|
||||||
|
@ -240,8 +231,8 @@ public:
|
||||||
// If not found then search the deferred modules. If any of these contain a
|
// If not found then search the deferred modules. If any of these contain a
|
||||||
// definition of 'Name' then they will return a JITSymbol that will emit
|
// definition of 'Name' then they will return a JITSymbol that will emit
|
||||||
// the corresponding module when the symbol address is requested.
|
// the corresponding module when the symbol address is requested.
|
||||||
for (auto &DeferredMod : ModuleList)
|
for (auto &KV : ModuleMap)
|
||||||
if (auto Symbol = DeferredMod->find(Name, ExportedSymbolsOnly, BaseLayer))
|
if (auto Symbol = KV.second->find(Name, ExportedSymbolsOnly, BaseLayer))
|
||||||
return Symbol;
|
return Symbol;
|
||||||
|
|
||||||
// If no definition found anywhere return a null symbol.
|
// If no definition found anywhere return a null symbol.
|
||||||
|
@ -249,17 +240,18 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Get the address of the given symbol in the context of the of
|
/// @brief Get the address of the given symbol in the context of the of
|
||||||
/// compiled modules represented by the handle H.
|
/// compiled modules represented by the key K.
|
||||||
JITSymbol findSymbolIn(ModuleHandleT H, const std::string &Name,
|
JITSymbol findSymbolIn(VModuleKey K, const std::string &Name,
|
||||||
bool ExportedSymbolsOnly) {
|
bool ExportedSymbolsOnly) {
|
||||||
return (*H)->find(Name, ExportedSymbolsOnly, BaseLayer);
|
assert(ModuleMap.count(K) && "VModuleKey K not valid here");
|
||||||
|
return ModuleMap[K]->find(Name, ExportedSymbolsOnly, BaseLayer);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Immediately emit and finalize the module represented by the given
|
/// @brief Immediately emit and finalize the module represented by the given
|
||||||
/// handle.
|
/// key.
|
||||||
/// @param H Handle for module to emit/finalize.
|
Error emitAndFinalize(VModuleKey K) {
|
||||||
Error emitAndFinalize(ModuleHandleT H) {
|
assert(ModuleMap.count(K) && "VModuleKey K not valid here");
|
||||||
return (*H)->emitAndFinalize(BaseLayer);
|
return ModuleMap[K]->emitAndFinalize(BaseLayer);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -31,9 +31,6 @@ namespace orc {
|
||||||
template <typename BaseLayerT, typename TransformFtor>
|
template <typename BaseLayerT, typename TransformFtor>
|
||||||
class ObjectTransformLayer {
|
class ObjectTransformLayer {
|
||||||
public:
|
public:
|
||||||
/// @brief Handle to a set of added objects.
|
|
||||||
using ObjHandleT = typename BaseLayerT::ObjHandleT;
|
|
||||||
|
|
||||||
/// @brief Construct an ObjectTransformLayer with the given BaseLayer
|
/// @brief Construct an ObjectTransformLayer with the given BaseLayer
|
||||||
ObjectTransformLayer(BaseLayerT &BaseLayer,
|
ObjectTransformLayer(BaseLayerT &BaseLayer,
|
||||||
TransformFtor Transform = TransformFtor())
|
TransformFtor Transform = TransformFtor())
|
||||||
|
@ -44,13 +41,12 @@ public:
|
||||||
/// memory manager and symbol resolver.
|
/// memory manager and symbol resolver.
|
||||||
///
|
///
|
||||||
/// @return A handle for the added objects.
|
/// @return A handle for the added objects.
|
||||||
template <typename ObjectPtr>
|
template <typename ObjectPtr> Error addObject(VModuleKey K, ObjectPtr Obj) {
|
||||||
Expected<ObjHandleT> addObject(VModuleKey K, ObjectPtr Obj) {
|
|
||||||
return BaseLayer.addObject(std::move(K), Transform(std::move(Obj)));
|
return BaseLayer.addObject(std::move(K), Transform(std::move(Obj)));
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Remove the object set associated with the handle H.
|
/// @brief Remove the object set associated with the VModuleKey K.
|
||||||
Error removeObject(ObjHandleT H) { return BaseLayer.removeObject(H); }
|
Error removeObject(VModuleKey K) { return BaseLayer.removeObject(K); }
|
||||||
|
|
||||||
/// @brief Search for the given named symbol.
|
/// @brief Search for the given named symbol.
|
||||||
/// @param Name The name of the symbol to search for.
|
/// @param Name The name of the symbol to search for.
|
||||||
|
@ -61,29 +57,27 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Get the address of the given symbol in the context of the set of
|
/// @brief Get the address of the given symbol in the context of the set of
|
||||||
/// objects represented by the handle H. This call is forwarded to the
|
/// objects represented by the VModuleKey K. This call is forwarded to
|
||||||
/// base layer's implementation.
|
/// the base layer's implementation.
|
||||||
/// @param H The handle for the object set to search in.
|
/// @param H The handle for the object set to search in.
|
||||||
/// @param Name The name of the symbol to search for.
|
/// @param Name The name of the symbol to search for.
|
||||||
/// @param ExportedSymbolsOnly If true, search only for exported symbols.
|
/// @param ExportedSymbolsOnly If true, search only for exported symbols.
|
||||||
/// @return A handle for the given named symbol, if it is found in the
|
/// @return A handle for the given named symbol, if it is found in the
|
||||||
/// given object set.
|
/// given object set.
|
||||||
JITSymbol findSymbolIn(ObjHandleT H, const std::string &Name,
|
JITSymbol findSymbolIn(VModuleKey K, const std::string &Name,
|
||||||
bool ExportedSymbolsOnly) {
|
bool ExportedSymbolsOnly) {
|
||||||
return BaseLayer.findSymbolIn(H, Name, ExportedSymbolsOnly);
|
return BaseLayer.findSymbolIn(K, Name, ExportedSymbolsOnly);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Immediately emit and finalize the object set represented by the
|
/// @brief Immediately emit and finalize the object set represented by the
|
||||||
/// given handle.
|
/// given VModuleKey K.
|
||||||
/// @param H Handle for object set to emit/finalize.
|
Error emitAndFinalize(VModuleKey K) { return BaseLayer.emitAndFinalize(K); }
|
||||||
Error emitAndFinalize(ObjHandleT H) {
|
|
||||||
return BaseLayer.emitAndFinalize(H);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// @brief Map section addresses for the objects associated with the handle H.
|
/// @brief Map section addresses for the objects associated with the
|
||||||
void mapSectionAddress(ObjHandleT H, const void *LocalAddress,
|
/// VModuleKey K.
|
||||||
|
void mapSectionAddress(VModuleKey K, const void *LocalAddress,
|
||||||
JITTargetAddress TargetAddr) {
|
JITTargetAddress TargetAddr) {
|
||||||
BaseLayer.mapSectionAddress(H, LocalAddress, TargetAddr);
|
BaseLayer.mapSectionAddress(K, LocalAddress, TargetAddr);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Access the transform functor directly.
|
/// @brief Access the transform functor directly.
|
||||||
|
|
|
@ -81,12 +81,6 @@ protected:
|
||||||
StringMap<JITEvaluatedSymbol> SymbolTable;
|
StringMap<JITEvaluatedSymbol> SymbolTable;
|
||||||
bool Finalized = false;
|
bool Finalized = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
using LinkedObjectListT = std::list<std::unique_ptr<LinkedObject>>;
|
|
||||||
|
|
||||||
public:
|
|
||||||
/// @brief Handle to a loaded object.
|
|
||||||
using ObjHandleT = LinkedObjectListT::iterator;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/// @brief Bare bones object linking layer.
|
/// @brief Bare bones object linking layer.
|
||||||
|
@ -101,12 +95,11 @@ public:
|
||||||
using RTDyldObjectLinkingLayerBase::ObjectPtr;
|
using RTDyldObjectLinkingLayerBase::ObjectPtr;
|
||||||
|
|
||||||
/// @brief Functor for receiving object-loaded notifications.
|
/// @brief Functor for receiving object-loaded notifications.
|
||||||
using NotifyLoadedFtor =
|
using NotifyLoadedFtor = std::function<void(
|
||||||
std::function<void(ObjHandleT, const ObjectPtr &Obj,
|
VModuleKey, const ObjectPtr &Obj, const RuntimeDyld::LoadedObjectInfo &)>;
|
||||||
const RuntimeDyld::LoadedObjectInfo &)>;
|
|
||||||
|
|
||||||
/// @brief Functor for receiving finalization notifications.
|
/// @brief Functor for receiving finalization notifications.
|
||||||
using NotifyFinalizedFtor = std::function<void(ObjHandleT)>;
|
using NotifyFinalizedFtor = std::function<void(VModuleKey)>;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
template <typename MemoryManagerPtrT, typename FinalizerFtor>
|
template <typename MemoryManagerPtrT, typename FinalizerFtor>
|
||||||
|
@ -127,10 +120,6 @@ private:
|
||||||
MemMgr->deregisterEHFrames();
|
MemMgr->deregisterEHFrames();
|
||||||
}
|
}
|
||||||
|
|
||||||
void setHandle(ObjHandleT H) {
|
|
||||||
PFC->Handle = H;
|
|
||||||
}
|
|
||||||
|
|
||||||
Error finalize() override {
|
Error finalize() override {
|
||||||
assert(PFC && "mapSectionAddress called on finalized LinkedObject");
|
assert(PFC && "mapSectionAddress called on finalized LinkedObject");
|
||||||
|
|
||||||
|
@ -140,7 +129,7 @@ private:
|
||||||
PFC->RTDyld = &RTDyld;
|
PFC->RTDyld = &RTDyld;
|
||||||
|
|
||||||
this->Finalized = true;
|
this->Finalized = true;
|
||||||
auto Err = PFC->Finalizer(PFC->Handle, RTDyld, std::move(PFC->Obj),
|
auto Err = PFC->Finalizer(RTDyld, std::move(PFC->Obj),
|
||||||
[&]() { this->updateSymbolTable(RTDyld); });
|
[&]() { this->updateSymbolTable(RTDyld); });
|
||||||
|
|
||||||
// Release resources.
|
// Release resources.
|
||||||
|
@ -204,7 +193,6 @@ private:
|
||||||
std::shared_ptr<SymbolResolver> Resolver;
|
std::shared_ptr<SymbolResolver> Resolver;
|
||||||
FinalizerFtor Finalizer;
|
FinalizerFtor Finalizer;
|
||||||
bool ProcessAllSections;
|
bool ProcessAllSections;
|
||||||
ObjHandleT Handle;
|
|
||||||
RuntimeDyld *RTDyld;
|
RuntimeDyld *RTDyld;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -257,20 +245,16 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Add an object to the JIT.
|
/// @brief Add an object to the JIT.
|
||||||
///
|
Error addObject(VModuleKey K, ObjectPtr Obj) {
|
||||||
/// @return A handle that can be used to refer to the loaded object (for
|
auto Finalizer = [&, K](RuntimeDyld &RTDyld, const ObjectPtr &ObjToLoad,
|
||||||
/// symbol searching, finalization, freeing memory, etc.).
|
std::function<void()> LOSHandleLoad) -> Error {
|
||||||
Expected<ObjHandleT> addObject(VModuleKey K, ObjectPtr Obj) {
|
|
||||||
auto Finalizer = [&](ObjHandleT H, RuntimeDyld &RTDyld,
|
|
||||||
const ObjectPtr &ObjToLoad,
|
|
||||||
std::function<void()> LOSHandleLoad) -> Error {
|
|
||||||
std::unique_ptr<RuntimeDyld::LoadedObjectInfo> Info =
|
std::unique_ptr<RuntimeDyld::LoadedObjectInfo> Info =
|
||||||
RTDyld.loadObject(*ObjToLoad->getBinary());
|
RTDyld.loadObject(*ObjToLoad->getBinary());
|
||||||
|
|
||||||
LOSHandleLoad();
|
LOSHandleLoad();
|
||||||
|
|
||||||
if (this->NotifyLoaded)
|
if (this->NotifyLoaded)
|
||||||
this->NotifyLoaded(H, ObjToLoad, *Info);
|
this->NotifyLoaded(K, ObjToLoad, *Info);
|
||||||
|
|
||||||
RTDyld.finalizeWithMemoryManagerLocking();
|
RTDyld.finalizeWithMemoryManagerLocking();
|
||||||
|
|
||||||
|
@ -279,25 +263,21 @@ public:
|
||||||
inconvertibleErrorCode());
|
inconvertibleErrorCode());
|
||||||
|
|
||||||
if (this->NotifyFinalized)
|
if (this->NotifyFinalized)
|
||||||
this->NotifyFinalized(H);
|
this->NotifyFinalized(K);
|
||||||
|
|
||||||
return Error::success();
|
return Error::success();
|
||||||
};
|
};
|
||||||
|
|
||||||
auto LO =
|
assert(!LinkedObjects.count(K) && "VModuleKey already in use");
|
||||||
|
|
||||||
|
LinkedObjects[K] =
|
||||||
createLinkedObject(ES, std::move(Obj), GetMemMgr(K), GetResolver(K),
|
createLinkedObject(ES, std::move(Obj), GetMemMgr(K), GetResolver(K),
|
||||||
std::move(Finalizer), ProcessAllSections);
|
std::move(Finalizer), ProcessAllSections);
|
||||||
// LOS is an owning-ptr. Keep a non-owning one so that we can set the handle
|
|
||||||
// below.
|
|
||||||
auto *LOPtr = LO.get();
|
|
||||||
|
|
||||||
ObjHandleT Handle = LinkedObjList.insert(LinkedObjList.end(), std::move(LO));
|
return Error::success();
|
||||||
LOPtr->setHandle(Handle);
|
|
||||||
|
|
||||||
return Handle;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Remove the object associated with handle H.
|
/// @brief Remove the object associated with VModuleKey K.
|
||||||
///
|
///
|
||||||
/// All memory allocated for the object will be freed, and the sections and
|
/// All memory allocated for the object will be freed, and the sections and
|
||||||
/// symbols it provided will no longer be available. No attempt is made to
|
/// symbols it provided will no longer be available. No attempt is made to
|
||||||
|
@ -305,9 +285,10 @@ public:
|
||||||
/// indirectly) will result in undefined behavior. If dependence tracking is
|
/// indirectly) will result in undefined behavior. If dependence tracking is
|
||||||
/// required to detect or resolve such issues it should be added at a higher
|
/// required to detect or resolve such issues it should be added at a higher
|
||||||
/// layer.
|
/// layer.
|
||||||
Error removeObject(ObjHandleT H) {
|
Error removeObject(VModuleKey K) {
|
||||||
|
assert(LinkedObjects.count(K) && "VModuleKey not associated with object");
|
||||||
// How do we invalidate the symbols in H?
|
// How do we invalidate the symbols in H?
|
||||||
LinkedObjList.erase(H);
|
LinkedObjects.erase(K);
|
||||||
return Error::success();
|
return Error::success();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -316,40 +297,48 @@ public:
|
||||||
/// @param ExportedSymbolsOnly If true, search only for exported symbols.
|
/// @param ExportedSymbolsOnly If true, search only for exported symbols.
|
||||||
/// @return A handle for the given named symbol, if it exists.
|
/// @return A handle for the given named symbol, if it exists.
|
||||||
JITSymbol findSymbol(StringRef Name, bool ExportedSymbolsOnly) {
|
JITSymbol findSymbol(StringRef Name, bool ExportedSymbolsOnly) {
|
||||||
for (auto I = LinkedObjList.begin(), E = LinkedObjList.end(); I != E;
|
for (auto &KV : LinkedObjects)
|
||||||
++I)
|
if (auto Sym = KV.second->getSymbol(Name, ExportedSymbolsOnly))
|
||||||
if (auto Symbol = findSymbolIn(I, Name, ExportedSymbolsOnly))
|
return Sym;
|
||||||
return Symbol;
|
else if (auto Err = Sym.takeError())
|
||||||
|
return std::move(Err);
|
||||||
|
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Search for the given named symbol in the context of the loaded
|
/// @brief Search for the given named symbol in the context of the loaded
|
||||||
/// object represented by the handle H.
|
/// object represented by the VModuleKey K.
|
||||||
/// @param H The handle for the object to search in.
|
/// @param K The VModuleKey for the object to search in.
|
||||||
/// @param Name The name of the symbol to search for.
|
/// @param Name The name of the symbol to search for.
|
||||||
/// @param ExportedSymbolsOnly If true, search only for exported symbols.
|
/// @param ExportedSymbolsOnly If true, search only for exported symbols.
|
||||||
/// @return A handle for the given named symbol, if it is found in the
|
/// @return A handle for the given named symbol, if it is found in the
|
||||||
/// given object.
|
/// given object.
|
||||||
JITSymbol findSymbolIn(ObjHandleT H, StringRef Name,
|
JITSymbol findSymbolIn(VModuleKey K, StringRef Name,
|
||||||
bool ExportedSymbolsOnly) {
|
bool ExportedSymbolsOnly) {
|
||||||
return (*H)->getSymbol(Name, ExportedSymbolsOnly);
|
assert(LinkedObjects.count(K) && "VModuleKey not associated with object");
|
||||||
|
return LinkedObjects[K]->getSymbol(Name, ExportedSymbolsOnly);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Map section addresses for the object associated with the handle H.
|
/// @brief Map section addresses for the object associated with the
|
||||||
void mapSectionAddress(ObjHandleT H, const void *LocalAddress,
|
/// VModuleKey K.
|
||||||
|
void mapSectionAddress(VModuleKey K, const void *LocalAddress,
|
||||||
JITTargetAddress TargetAddr) {
|
JITTargetAddress TargetAddr) {
|
||||||
(*H)->mapSectionAddress(LocalAddress, TargetAddr);
|
assert(LinkedObjects.count(K) && "VModuleKey not associated with object");
|
||||||
|
LinkedObjects[K]->mapSectionAddress(LocalAddress, TargetAddr);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @brief Immediately emit and finalize the object represented by the given
|
/// @brief Immediately emit and finalize the object represented by the given
|
||||||
/// handle.
|
/// VModuleKey.
|
||||||
/// @param H Handle for object to emit/finalize.
|
/// @param K VModuleKey for object to emit/finalize.
|
||||||
Error emitAndFinalize(ObjHandleT H) { return (*H)->finalize(); }
|
Error emitAndFinalize(VModuleKey K) {
|
||||||
|
assert(LinkedObjects.count(K) && "VModuleKey not associated with object");
|
||||||
|
return LinkedObjects[K]->finalize();
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
ExecutionSession &ES;
|
ExecutionSession &ES;
|
||||||
LinkedObjectListT LinkedObjList;
|
|
||||||
|
std::map<VModuleKey, std::unique_ptr<LinkedObject>> LinkedObjects;
|
||||||
MemoryManagerGetter GetMemMgr;
|
MemoryManagerGetter GetMemMgr;
|
||||||
ResolverGetter GetResolver;
|
ResolverGetter GetResolver;
|
||||||
NotifyLoadedFtor NotifyLoaded;
|
NotifyLoadedFtor NotifyLoaded;
|
||||||
|
|
|
@ -33,6 +33,7 @@
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
#include <functional>
|
#include <functional>
|
||||||
|
#include <map>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <set>
|
#include <set>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
@ -49,75 +50,56 @@ DEFINE_SIMPLE_CONVERSION_FUNCTIONS(TargetMachine, LLVMTargetMachineRef)
|
||||||
|
|
||||||
namespace detail {
|
namespace detail {
|
||||||
|
|
||||||
|
// FIXME: Kill this off once the Layer concept becomes an interface.
|
||||||
|
class GenericLayer {
|
||||||
|
public:
|
||||||
|
virtual ~GenericLayer() = default;
|
||||||
|
|
||||||
class GenericHandle {
|
virtual JITSymbol findSymbolIn(orc::VModuleKey K, const std::string &Name,
|
||||||
public:
|
bool ExportedSymbolsOnly) = 0;
|
||||||
GenericHandle(orc::VModuleKey K) : K(K) {}
|
virtual Error removeModule(orc::VModuleKey K) = 0;
|
||||||
|
|
||||||
virtual ~GenericHandle() = default;
|
|
||||||
|
|
||||||
virtual JITSymbol findSymbolIn(const std::string &Name,
|
|
||||||
bool ExportedSymbolsOnly) = 0;
|
|
||||||
virtual Error removeModule(orc::ExecutionSession &ES) = 0;
|
|
||||||
|
|
||||||
orc::VModuleKey K;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename LayerT> class GenericHandleImpl : public GenericHandle {
|
template <typename LayerT> class GenericLayerImpl : public GenericLayer {
|
||||||
public:
|
public:
|
||||||
GenericHandleImpl(LayerT &Layer, typename LayerT::ModuleHandleT Handle,
|
GenericLayerImpl(LayerT &Layer) : Layer(Layer) {}
|
||||||
orc::VModuleKey K)
|
|
||||||
: GenericHandle(std::move(K)), Layer(Layer), Handle(std::move(Handle)) {
|
|
||||||
}
|
|
||||||
|
|
||||||
JITSymbol findSymbolIn(const std::string &Name,
|
JITSymbol findSymbolIn(orc::VModuleKey K, const std::string &Name,
|
||||||
bool ExportedSymbolsOnly) override {
|
bool ExportedSymbolsOnly) override {
|
||||||
return Layer.findSymbolIn(Handle, Name, ExportedSymbolsOnly);
|
return Layer.findSymbolIn(K, Name, ExportedSymbolsOnly);
|
||||||
}
|
}
|
||||||
|
|
||||||
Error removeModule(orc::ExecutionSession &ES) override {
|
Error removeModule(orc::VModuleKey K) override {
|
||||||
auto Err = Layer.removeModule(Handle);
|
return Layer.removeModule(K);
|
||||||
ES.releaseVModule(K);
|
|
||||||
return Err;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
LayerT &Layer;
|
LayerT &Layer;
|
||||||
typename LayerT::ModuleHandleT Handle;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
template <>
|
template <>
|
||||||
class GenericHandleImpl<orc::RTDyldObjectLinkingLayer>
|
class GenericLayerImpl<orc::RTDyldObjectLinkingLayer> : public GenericLayer {
|
||||||
: public GenericHandle {
|
|
||||||
private:
|
private:
|
||||||
using LayerT = orc::RTDyldObjectLinkingLayer;
|
using LayerT = orc::RTDyldObjectLinkingLayer;
|
||||||
public:
|
public:
|
||||||
GenericHandleImpl(LayerT &Layer, typename LayerT::ObjHandleT Handle,
|
GenericLayerImpl(LayerT &Layer) : Layer(Layer) {}
|
||||||
orc::VModuleKey K)
|
|
||||||
: GenericHandle(std::move(K)), Layer(Layer), Handle(std::move(Handle)) {
|
|
||||||
}
|
|
||||||
|
|
||||||
JITSymbol findSymbolIn(const std::string &Name,
|
JITSymbol findSymbolIn(orc::VModuleKey K, const std::string &Name,
|
||||||
bool ExportedSymbolsOnly) override {
|
bool ExportedSymbolsOnly) override {
|
||||||
return Layer.findSymbolIn(Handle, Name, ExportedSymbolsOnly);
|
return Layer.findSymbolIn(K, Name, ExportedSymbolsOnly);
|
||||||
}
|
}
|
||||||
|
|
||||||
Error removeModule(orc::ExecutionSession &ES) override {
|
Error removeModule(orc::VModuleKey K) override {
|
||||||
auto Err = Layer.removeObject(Handle);
|
return Layer.removeObject(K);
|
||||||
ES.releaseVModule(K);
|
|
||||||
return Err;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
LayerT &Layer;
|
LayerT &Layer;
|
||||||
typename LayerT::ObjHandleT Handle;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename LayerT, typename HandleT>
|
template <typename LayerT>
|
||||||
std::unique_ptr<GenericHandleImpl<LayerT>>
|
std::unique_ptr<GenericLayerImpl<LayerT>> createGenericLayer(LayerT &Layer) {
|
||||||
createGenericHandle(LayerT &Layer, HandleT Handle, orc::VModuleKey K) {
|
return llvm::make_unique<GenericLayerImpl<LayerT>>(Layer);
|
||||||
return llvm::make_unique<GenericHandleImpl<LayerT>>(
|
|
||||||
Layer, std::move(Handle), std::move(K));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
} // end namespace detail
|
} // end namespace detail
|
||||||
|
@ -215,7 +197,6 @@ private:
|
||||||
};
|
};
|
||||||
|
|
||||||
public:
|
public:
|
||||||
using ModuleHandleT = unsigned;
|
|
||||||
|
|
||||||
OrcCBindingsStack(TargetMachine &TM,
|
OrcCBindingsStack(TargetMachine &TM,
|
||||||
std::unique_ptr<CompileCallbackMgr> CCMgr,
|
std::unique_ptr<CompileCallbackMgr> CCMgr,
|
||||||
|
@ -304,8 +285,7 @@ public:
|
||||||
}
|
}
|
||||||
template <typename LayerT>
|
template <typename LayerT>
|
||||||
LLVMOrcErrorCode
|
LLVMOrcErrorCode
|
||||||
addIRModule(ModuleHandleT &RetHandle, LayerT &Layer,
|
addIRModule(orc::VModuleKey &RetKey, LayerT &Layer, std::shared_ptr<Module> M,
|
||||||
std::shared_ptr<Module> M,
|
|
||||||
std::unique_ptr<RuntimeDyld::MemoryManager> MemMgr,
|
std::unique_ptr<RuntimeDyld::MemoryManager> MemMgr,
|
||||||
LLVMOrcSymbolResolverFn ExternalResolver,
|
LLVMOrcSymbolResolverFn ExternalResolver,
|
||||||
void *ExternalResolverCtx) {
|
void *ExternalResolverCtx) {
|
||||||
|
@ -323,54 +303,54 @@ public:
|
||||||
DtorNames.push_back(mangle(Dtor.Func->getName()));
|
DtorNames.push_back(mangle(Dtor.Func->getName()));
|
||||||
|
|
||||||
// Add the module to the JIT.
|
// Add the module to the JIT.
|
||||||
ModuleHandleT H;
|
RetKey = ES.allocateVModule();
|
||||||
orc::VModuleKey K = ES.allocateVModule();
|
Resolvers[RetKey] = std::make_shared<CBindingsResolver>(
|
||||||
Resolvers[K] = std::make_shared<CBindingsResolver>(*this, ExternalResolver,
|
*this, ExternalResolver, ExternalResolverCtx);
|
||||||
ExternalResolverCtx);
|
if (auto Err = Layer.addModule(RetKey, std::move(M)))
|
||||||
if (auto LHOrErr = Layer.addModule(K, std::move(M)))
|
return mapError(std::move(Err));
|
||||||
H = createHandle(Layer, *LHOrErr, K);
|
|
||||||
else
|
KeyLayers[RetKey] = detail::createGenericLayer(Layer);
|
||||||
return mapError(LHOrErr.takeError());
|
|
||||||
|
|
||||||
// Run the static constructors, and save the static destructor runner for
|
// Run the static constructors, and save the static destructor runner for
|
||||||
// execution when the JIT is torn down.
|
// execution when the JIT is torn down.
|
||||||
orc::CtorDtorRunner<OrcCBindingsStack> CtorRunner(std::move(CtorNames), H);
|
orc::CtorDtorRunner<OrcCBindingsStack> CtorRunner(std::move(CtorNames),
|
||||||
|
RetKey);
|
||||||
if (auto Err = CtorRunner.runViaLayer(*this))
|
if (auto Err = CtorRunner.runViaLayer(*this))
|
||||||
return mapError(std::move(Err));
|
return mapError(std::move(Err));
|
||||||
|
|
||||||
IRStaticDestructorRunners.emplace_back(std::move(DtorNames), H);
|
IRStaticDestructorRunners.emplace_back(std::move(DtorNames), RetKey);
|
||||||
|
|
||||||
RetHandle = H;
|
|
||||||
return LLVMOrcErrSuccess;
|
return LLVMOrcErrSuccess;
|
||||||
}
|
}
|
||||||
|
|
||||||
LLVMOrcErrorCode addIRModuleEager(ModuleHandleT &RetHandle,
|
LLVMOrcErrorCode addIRModuleEager(orc::VModuleKey &RetKey,
|
||||||
std::shared_ptr<Module> M,
|
std::shared_ptr<Module> M,
|
||||||
LLVMOrcSymbolResolverFn ExternalResolver,
|
LLVMOrcSymbolResolverFn ExternalResolver,
|
||||||
void *ExternalResolverCtx) {
|
void *ExternalResolverCtx) {
|
||||||
return addIRModule(RetHandle, CompileLayer, std::move(M),
|
return addIRModule(RetKey, CompileLayer, std::move(M),
|
||||||
llvm::make_unique<SectionMemoryManager>(),
|
llvm::make_unique<SectionMemoryManager>(),
|
||||||
std::move(ExternalResolver), ExternalResolverCtx);
|
std::move(ExternalResolver), ExternalResolverCtx);
|
||||||
}
|
}
|
||||||
|
|
||||||
LLVMOrcErrorCode addIRModuleLazy(ModuleHandleT &RetHandle,
|
LLVMOrcErrorCode addIRModuleLazy(orc::VModuleKey &RetKey,
|
||||||
std::shared_ptr<Module> M,
|
std::shared_ptr<Module> M,
|
||||||
LLVMOrcSymbolResolverFn ExternalResolver,
|
LLVMOrcSymbolResolverFn ExternalResolver,
|
||||||
void *ExternalResolverCtx) {
|
void *ExternalResolverCtx) {
|
||||||
return addIRModule(RetHandle, CODLayer, std::move(M),
|
return addIRModule(RetKey, CODLayer, std::move(M),
|
||||||
llvm::make_unique<SectionMemoryManager>(),
|
llvm::make_unique<SectionMemoryManager>(),
|
||||||
std::move(ExternalResolver), ExternalResolverCtx);
|
std::move(ExternalResolver), ExternalResolverCtx);
|
||||||
}
|
}
|
||||||
|
|
||||||
LLVMOrcErrorCode removeModule(ModuleHandleT H) {
|
LLVMOrcErrorCode removeModule(orc::VModuleKey K) {
|
||||||
if (auto Err = GenericHandles[H]->removeModule(ES))
|
// FIXME: Should error release the module key?
|
||||||
|
if (auto Err = KeyLayers[K]->removeModule(K))
|
||||||
return mapError(std::move(Err));
|
return mapError(std::move(Err));
|
||||||
GenericHandles[H] = nullptr;
|
ES.releaseVModule(K);
|
||||||
FreeHandleIndexes.push_back(H);
|
KeyLayers.erase(K);
|
||||||
return LLVMOrcErrSuccess;
|
return LLVMOrcErrSuccess;
|
||||||
}
|
}
|
||||||
|
|
||||||
LLVMOrcErrorCode addObject(ModuleHandleT &RetHandle,
|
LLVMOrcErrorCode addObject(orc::VModuleKey &RetKey,
|
||||||
std::unique_ptr<MemoryBuffer> ObjBuffer,
|
std::unique_ptr<MemoryBuffer> ObjBuffer,
|
||||||
LLVMOrcSymbolResolverFn ExternalResolver,
|
LLVMOrcSymbolResolverFn ExternalResolver,
|
||||||
void *ExternalResolverCtx) {
|
void *ExternalResolverCtx) {
|
||||||
|
@ -380,17 +360,14 @@ public:
|
||||||
auto OwningObj =
|
auto OwningObj =
|
||||||
std::make_shared<OwningObject>(std::move(Obj), std::move(ObjBuffer));
|
std::make_shared<OwningObject>(std::move(Obj), std::move(ObjBuffer));
|
||||||
|
|
||||||
orc::VModuleKey K = ES.allocateVModule();
|
RetKey = ES.allocateVModule();
|
||||||
Resolvers[K] = std::make_shared<CBindingsResolver>(
|
Resolvers[RetKey] = std::make_shared<CBindingsResolver>(
|
||||||
*this, ExternalResolver, ExternalResolverCtx);
|
*this, ExternalResolver, ExternalResolverCtx);
|
||||||
|
|
||||||
ModuleHandleT H;
|
if (auto Err = ObjectLayer.addObject(RetKey, std::move(OwningObj)))
|
||||||
if (auto HOrErr = ObjectLayer.addObject(K, std::move(OwningObj)))
|
return mapError(std::move(Err));
|
||||||
H = createHandle(ObjectLayer, *HOrErr, K);
|
|
||||||
else
|
|
||||||
return mapError(HOrErr.takeError());
|
|
||||||
|
|
||||||
RetHandle = H;
|
KeyLayers[RetKey] = detail::createGenericLayer(ObjectLayer);
|
||||||
|
|
||||||
return LLVMOrcErrSuccess;
|
return LLVMOrcErrSuccess;
|
||||||
} else
|
} else
|
||||||
|
@ -404,9 +381,9 @@ public:
|
||||||
return CODLayer.findSymbol(mangle(Name), ExportedSymbolsOnly);
|
return CODLayer.findSymbol(mangle(Name), ExportedSymbolsOnly);
|
||||||
}
|
}
|
||||||
|
|
||||||
JITSymbol findSymbolIn(ModuleHandleT H, const std::string &Name,
|
JITSymbol findSymbolIn(orc::VModuleKey K, const std::string &Name,
|
||||||
bool ExportedSymbolsOnly) {
|
bool ExportedSymbolsOnly) {
|
||||||
return GenericHandles[H]->findSymbolIn(Name, ExportedSymbolsOnly);
|
return KeyLayers[K]->findSymbolIn(K, Name, ExportedSymbolsOnly);
|
||||||
}
|
}
|
||||||
|
|
||||||
LLVMOrcErrorCode findSymbolAddress(JITTargetAddress &RetAddr,
|
LLVMOrcErrorCode findSymbolAddress(JITTargetAddress &RetAddr,
|
||||||
|
@ -432,22 +409,6 @@ public:
|
||||||
const std::string &getErrorMessage() const { return ErrMsg; }
|
const std::string &getErrorMessage() const { return ErrMsg; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
template <typename LayerT, typename HandleT>
|
|
||||||
unsigned createHandle(LayerT &Layer, HandleT Handle, orc::VModuleKey K) {
|
|
||||||
unsigned NewHandle;
|
|
||||||
if (!FreeHandleIndexes.empty()) {
|
|
||||||
NewHandle = FreeHandleIndexes.back();
|
|
||||||
FreeHandleIndexes.pop_back();
|
|
||||||
GenericHandles[NewHandle] =
|
|
||||||
detail::createGenericHandle(Layer, std::move(Handle), std::move(K));
|
|
||||||
return NewHandle;
|
|
||||||
} else {
|
|
||||||
NewHandle = GenericHandles.size();
|
|
||||||
GenericHandles.push_back(
|
|
||||||
detail::createGenericHandle(Layer, std::move(Handle), std::move(K)));
|
|
||||||
}
|
|
||||||
return NewHandle;
|
|
||||||
}
|
|
||||||
|
|
||||||
LLVMOrcErrorCode mapError(Error Err) {
|
LLVMOrcErrorCode mapError(Error Err) {
|
||||||
LLVMOrcErrorCode Result = LLVMOrcErrSuccess;
|
LLVMOrcErrorCode Result = LLVMOrcErrSuccess;
|
||||||
|
@ -479,8 +440,7 @@ private:
|
||||||
CompileLayerT CompileLayer;
|
CompileLayerT CompileLayer;
|
||||||
CODLayerT CODLayer;
|
CODLayerT CODLayer;
|
||||||
|
|
||||||
std::vector<std::unique_ptr<detail::GenericHandle>> GenericHandles;
|
std::map<orc::VModuleKey, std::unique_ptr<detail::GenericLayer>> KeyLayers;
|
||||||
std::vector<unsigned> FreeHandleIndexes;
|
|
||||||
|
|
||||||
orc::LocalCXXRuntimeOverrides CXXRuntimeOverrides;
|
orc::LocalCXXRuntimeOverrides CXXRuntimeOverrides;
|
||||||
std::vector<orc::CtorDtorRunner<OrcCBindingsStack>> IRStaticDestructorRunners;
|
std::vector<orc::CtorDtorRunner<OrcCBindingsStack>> IRStaticDestructorRunners;
|
||||||
|
|
|
@ -393,10 +393,10 @@ private:
|
||||||
|
|
||||||
NotifyObjectLoadedT(OrcMCJITReplacement &M) : M(M) {}
|
NotifyObjectLoadedT(OrcMCJITReplacement &M) : M(M) {}
|
||||||
|
|
||||||
void operator()(RTDyldObjectLinkingLayerBase::ObjHandleT H,
|
void operator()(VModuleKey K,
|
||||||
const RTDyldObjectLinkingLayer::ObjectPtr &Obj,
|
const RTDyldObjectLinkingLayer::ObjectPtr &Obj,
|
||||||
const RuntimeDyld::LoadedObjectInfo &Info) const {
|
const RuntimeDyld::LoadedObjectInfo &Info) const {
|
||||||
M.UnfinalizedSections[H] = std::move(M.SectionsAllocatedSinceLastLoad);
|
M.UnfinalizedSections[K] = std::move(M.SectionsAllocatedSinceLastLoad);
|
||||||
M.SectionsAllocatedSinceLastLoad = SectionAddrSet();
|
M.SectionsAllocatedSinceLastLoad = SectionAddrSet();
|
||||||
M.MemMgr->notifyObjectLoaded(&M, *Obj->getBinary());
|
M.MemMgr->notifyObjectLoaded(&M, *Obj->getBinary());
|
||||||
}
|
}
|
||||||
|
@ -408,9 +408,7 @@ private:
|
||||||
public:
|
public:
|
||||||
NotifyFinalizedT(OrcMCJITReplacement &M) : M(M) {}
|
NotifyFinalizedT(OrcMCJITReplacement &M) : M(M) {}
|
||||||
|
|
||||||
void operator()(RTDyldObjectLinkingLayerBase::ObjHandleT H) {
|
void operator()(VModuleKey K) { M.UnfinalizedSections.erase(K); }
|
||||||
M.UnfinalizedSections.erase(H);
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
OrcMCJITReplacement &M;
|
OrcMCJITReplacement &M;
|
||||||
|
@ -455,15 +453,8 @@ private:
|
||||||
// that have been emitted but not yet finalized so that we can forward the
|
// that have been emitted but not yet finalized so that we can forward the
|
||||||
// mapSectionAddress calls appropriately.
|
// mapSectionAddress calls appropriately.
|
||||||
using SectionAddrSet = std::set<const void *>;
|
using SectionAddrSet = std::set<const void *>;
|
||||||
struct ObjHandleCompare {
|
|
||||||
bool operator()(ObjectLayerT::ObjHandleT H1,
|
|
||||||
ObjectLayerT::ObjHandleT H2) const {
|
|
||||||
return &*H1 < &*H2;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
SectionAddrSet SectionsAllocatedSinceLastLoad;
|
SectionAddrSet SectionsAllocatedSinceLastLoad;
|
||||||
std::map<ObjectLayerT::ObjHandleT, SectionAddrSet, ObjHandleCompare>
|
std::map<VModuleKey, SectionAddrSet> UnfinalizedSections;
|
||||||
UnfinalizedSections;
|
|
||||||
|
|
||||||
std::vector<object::OwningBinary<object::Archive>> Archives;
|
std::vector<object::OwningBinary<object::Archive>> Archives;
|
||||||
};
|
};
|
||||||
|
|
|
@ -55,7 +55,6 @@ public:
|
||||||
using IRDumpLayerT = orc::IRTransformLayer<CompileLayerT, TransformFtor>;
|
using IRDumpLayerT = orc::IRTransformLayer<CompileLayerT, TransformFtor>;
|
||||||
using CODLayerT = orc::CompileOnDemandLayer<IRDumpLayerT, CompileCallbackMgr>;
|
using CODLayerT = orc::CompileOnDemandLayer<IRDumpLayerT, CompileCallbackMgr>;
|
||||||
using IndirectStubsManagerBuilder = CODLayerT::IndirectStubsManagerBuilderT;
|
using IndirectStubsManagerBuilder = CODLayerT::IndirectStubsManagerBuilderT;
|
||||||
using ModuleHandleT = CODLayerT::ModuleHandleT;
|
|
||||||
|
|
||||||
OrcLazyJIT(std::unique_ptr<TargetMachine> TM,
|
OrcLazyJIT(std::unique_ptr<TargetMachine> TM,
|
||||||
std::unique_ptr<CompileCallbackMgr> CCMgr,
|
std::unique_ptr<CompileCallbackMgr> CCMgr,
|
||||||
|
@ -138,7 +137,7 @@ public:
|
||||||
// 1) Search the JIT symbols.
|
// 1) Search the JIT symbols.
|
||||||
// 2) Check for C++ runtime overrides.
|
// 2) Check for C++ runtime overrides.
|
||||||
// 3) Search the host process (LLI)'s symbol table.
|
// 3) Search the host process (LLI)'s symbol table.
|
||||||
if (!ModulesHandle) {
|
if (!ModulesKey) {
|
||||||
auto LegacyLookupInDylib = [this](const std::string &Name) -> JITSymbol {
|
auto LegacyLookupInDylib = [this](const std::string &Name) -> JITSymbol {
|
||||||
if (auto Sym = CODLayer.findSymbol(Name, true))
|
if (auto Sym = CODLayer.findSymbol(Name, true))
|
||||||
return Sym;
|
return Sym;
|
||||||
|
@ -160,9 +159,9 @@ public:
|
||||||
return nullptr;
|
return nullptr;
|
||||||
};
|
};
|
||||||
|
|
||||||
auto K = ES.allocateVModule();
|
ModulesKey = ES.allocateVModule();
|
||||||
assert(!Resolvers.count(K) && "Resolver already present");
|
assert(!Resolvers.count(*ModulesKey) && "Resolver already present");
|
||||||
Resolvers[K] = orc::createSymbolResolver(
|
Resolvers[*ModulesKey] = orc::createSymbolResolver(
|
||||||
[LegacyLookupInDylib](orc::SymbolFlagsMap &SymbolFlags,
|
[LegacyLookupInDylib](orc::SymbolFlagsMap &SymbolFlags,
|
||||||
const orc::SymbolNameSet &Symbols) {
|
const orc::SymbolNameSet &Symbols) {
|
||||||
auto NotFoundViaLegacyLookup = lookupFlagsWithLegacyFn(
|
auto NotFoundViaLegacyLookup = lookupFlagsWithLegacyFn(
|
||||||
|
@ -181,24 +180,20 @@ public:
|
||||||
});
|
});
|
||||||
|
|
||||||
// Add the module to the JIT.
|
// Add the module to the JIT.
|
||||||
if (auto ModulesHandleOrErr =
|
if (auto Err = CODLayer.addModule(*ModulesKey, std::move(M)))
|
||||||
CODLayer.addModule(std::move(K), std::move(M)))
|
return Err;
|
||||||
ModulesHandle = std::move(*ModulesHandleOrErr);
|
|
||||||
else
|
|
||||||
return ModulesHandleOrErr.takeError();
|
|
||||||
|
|
||||||
} else if (auto Err = CODLayer.addExtraModule(*ModulesHandle, std::move(M)))
|
} else if (auto Err = CODLayer.addExtraModule(*ModulesKey, std::move(M)))
|
||||||
return Err;
|
return Err;
|
||||||
|
|
||||||
// Run the static constructors, and save the static destructor runner for
|
// Run the static constructors, and save the static destructor runner for
|
||||||
// execution when the JIT is torn down.
|
// execution when the JIT is torn down.
|
||||||
orc::CtorDtorRunner<CODLayerT> CtorRunner(std::move(CtorNames),
|
orc::CtorDtorRunner<CODLayerT> CtorRunner(std::move(CtorNames),
|
||||||
*ModulesHandle);
|
*ModulesKey);
|
||||||
if (auto Err = CtorRunner.runViaLayer(CODLayer))
|
if (auto Err = CtorRunner.runViaLayer(CODLayer))
|
||||||
return Err;
|
return Err;
|
||||||
|
|
||||||
IRStaticDestructorRunners.emplace_back(std::move(DtorNames),
|
IRStaticDestructorRunners.emplace_back(std::move(DtorNames), *ModulesKey);
|
||||||
*ModulesHandle);
|
|
||||||
|
|
||||||
return Error::success();
|
return Error::success();
|
||||||
}
|
}
|
||||||
|
@ -207,8 +202,8 @@ public:
|
||||||
return CODLayer.findSymbol(mangle(Name), true);
|
return CODLayer.findSymbol(mangle(Name), true);
|
||||||
}
|
}
|
||||||
|
|
||||||
JITSymbol findSymbolIn(ModuleHandleT H, const std::string &Name) {
|
JITSymbol findSymbolIn(orc::VModuleKey K, const std::string &Name) {
|
||||||
return CODLayer.findSymbolIn(H, mangle(Name), true);
|
return CODLayer.findSymbolIn(K, mangle(Name), true);
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -246,7 +241,7 @@ private:
|
||||||
|
|
||||||
orc::LocalCXXRuntimeOverrides CXXRuntimeOverrides;
|
orc::LocalCXXRuntimeOverrides CXXRuntimeOverrides;
|
||||||
std::vector<orc::CtorDtorRunner<CODLayerT>> IRStaticDestructorRunners;
|
std::vector<orc::CtorDtorRunner<CODLayerT>> IRStaticDestructorRunners;
|
||||||
llvm::Optional<CODLayerT::ModuleHandleT> ModulesHandle;
|
llvm::Optional<orc::VModuleKey> ModulesKey;
|
||||||
};
|
};
|
||||||
|
|
||||||
int runOrcLazyJIT(std::vector<std::unique_ptr<Module>> Ms,
|
int runOrcLazyJIT(std::vector<std::unique_ptr<Module>> Ms,
|
||||||
|
|
|
@ -43,17 +43,13 @@ struct AllocatingTransform {
|
||||||
// transform layer called the base layer and forwarded any return value.
|
// transform layer called the base layer and forwarded any return value.
|
||||||
class MockBaseLayer {
|
class MockBaseLayer {
|
||||||
public:
|
public:
|
||||||
typedef int ObjHandleT;
|
|
||||||
|
|
||||||
MockBaseLayer() : MockSymbol(nullptr) { resetExpectations(); }
|
MockBaseLayer() : MockSymbol(nullptr) { resetExpectations(); }
|
||||||
|
|
||||||
template <typename ObjPtrT>
|
template <typename ObjPtrT> llvm::Error addObject(VModuleKey K, ObjPtrT Obj) {
|
||||||
llvm::Expected<ObjHandleT> addObject(VModuleKey K, ObjPtrT Obj) {
|
|
||||||
EXPECT_EQ(MockKey, K) << "Key should pass through";
|
EXPECT_EQ(MockKey, K) << "Key should pass through";
|
||||||
EXPECT_EQ(MockObject + 1, *Obj) << "Transform should be applied";
|
EXPECT_EQ(MockObject + 1, *Obj) << "Transform should be applied";
|
||||||
LastCalled = "addObject";
|
LastCalled = "addObject";
|
||||||
MockObjHandle = 111;
|
return llvm::Error::success();
|
||||||
return MockObjHandle;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename ObjPtrT> void expectAddObject(VModuleKey K, ObjPtrT Obj) {
|
template <typename ObjPtrT> void expectAddObject(VModuleKey K, ObjPtrT Obj) {
|
||||||
|
@ -61,20 +57,18 @@ public:
|
||||||
MockObject = *Obj;
|
MockObject = *Obj;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void verifyAddObject() {
|
||||||
void verifyAddObject(ObjHandleT Returned) {
|
|
||||||
EXPECT_EQ("addObject", LastCalled);
|
EXPECT_EQ("addObject", LastCalled);
|
||||||
EXPECT_EQ(MockObjHandle, Returned) << "Return should pass through";
|
|
||||||
resetExpectations();
|
resetExpectations();
|
||||||
}
|
}
|
||||||
|
|
||||||
llvm::Error removeObject(ObjHandleT H) {
|
llvm::Error removeObject(VModuleKey K) {
|
||||||
EXPECT_EQ(MockObjHandle, H);
|
EXPECT_EQ(MockKey, K);
|
||||||
LastCalled = "removeObject";
|
LastCalled = "removeObject";
|
||||||
return llvm::Error::success();
|
return llvm::Error::success();
|
||||||
}
|
}
|
||||||
|
|
||||||
void expectRemoveObject(ObjHandleT H) { MockObjHandle = H; }
|
void expectRemoveObject(VModuleKey K) { MockKey = K; }
|
||||||
void verifyRemoveObject() {
|
void verifyRemoveObject() {
|
||||||
EXPECT_EQ("removeObject", LastCalled);
|
EXPECT_EQ("removeObject", LastCalled);
|
||||||
resetExpectations();
|
resetExpectations();
|
||||||
|
@ -100,18 +94,18 @@ public:
|
||||||
resetExpectations();
|
resetExpectations();
|
||||||
}
|
}
|
||||||
|
|
||||||
llvm::JITSymbol findSymbolIn(ObjHandleT H, const std::string &Name,
|
llvm::JITSymbol findSymbolIn(VModuleKey K, const std::string &Name,
|
||||||
bool ExportedSymbolsOnly) {
|
bool ExportedSymbolsOnly) {
|
||||||
EXPECT_EQ(MockObjHandle, H) << "Handle should pass through";
|
EXPECT_EQ(MockKey, K) << "VModuleKey should pass through";
|
||||||
EXPECT_EQ(MockName, Name) << "Name should pass through";
|
EXPECT_EQ(MockName, Name) << "Name should pass through";
|
||||||
EXPECT_EQ(MockBool, ExportedSymbolsOnly) << "Flag should pass through";
|
EXPECT_EQ(MockBool, ExportedSymbolsOnly) << "Flag should pass through";
|
||||||
LastCalled = "findSymbolIn";
|
LastCalled = "findSymbolIn";
|
||||||
MockSymbol = llvm::JITSymbol(122, llvm::JITSymbolFlags::None);
|
MockSymbol = llvm::JITSymbol(122, llvm::JITSymbolFlags::None);
|
||||||
return llvm::JITSymbol(122, llvm::JITSymbolFlags::None);
|
return llvm::JITSymbol(122, llvm::JITSymbolFlags::None);
|
||||||
}
|
}
|
||||||
void expectFindSymbolIn(ObjHandleT H, const std::string &Name,
|
void expectFindSymbolIn(VModuleKey K, const std::string &Name,
|
||||||
bool ExportedSymbolsOnly) {
|
bool ExportedSymbolsOnly) {
|
||||||
MockObjHandle = H;
|
MockKey = K;
|
||||||
MockName = Name;
|
MockName = Name;
|
||||||
MockBool = ExportedSymbolsOnly;
|
MockBool = ExportedSymbolsOnly;
|
||||||
}
|
}
|
||||||
|
@ -123,29 +117,29 @@ public:
|
||||||
resetExpectations();
|
resetExpectations();
|
||||||
}
|
}
|
||||||
|
|
||||||
llvm::Error emitAndFinalize(ObjHandleT H) {
|
llvm::Error emitAndFinalize(VModuleKey K) {
|
||||||
EXPECT_EQ(MockObjHandle, H) << "Handle should pass through";
|
EXPECT_EQ(MockKey, K) << "VModuleKey should pass through";
|
||||||
LastCalled = "emitAndFinalize";
|
LastCalled = "emitAndFinalize";
|
||||||
return llvm::Error::success();
|
return llvm::Error::success();
|
||||||
}
|
}
|
||||||
|
|
||||||
void expectEmitAndFinalize(ObjHandleT H) { MockObjHandle = H; }
|
void expectEmitAndFinalize(VModuleKey K) { MockKey = K; }
|
||||||
|
|
||||||
void verifyEmitAndFinalize() {
|
void verifyEmitAndFinalize() {
|
||||||
EXPECT_EQ("emitAndFinalize", LastCalled);
|
EXPECT_EQ("emitAndFinalize", LastCalled);
|
||||||
resetExpectations();
|
resetExpectations();
|
||||||
}
|
}
|
||||||
|
|
||||||
void mapSectionAddress(ObjHandleT H, const void *LocalAddress,
|
void mapSectionAddress(VModuleKey K, const void *LocalAddress,
|
||||||
llvm::JITTargetAddress TargetAddr) {
|
llvm::JITTargetAddress TargetAddr) {
|
||||||
EXPECT_EQ(MockObjHandle, H);
|
EXPECT_EQ(MockKey, K);
|
||||||
EXPECT_EQ(MockLocalAddress, LocalAddress);
|
EXPECT_EQ(MockLocalAddress, LocalAddress);
|
||||||
EXPECT_EQ(MockTargetAddress, TargetAddr);
|
EXPECT_EQ(MockTargetAddress, TargetAddr);
|
||||||
LastCalled = "mapSectionAddress";
|
LastCalled = "mapSectionAddress";
|
||||||
}
|
}
|
||||||
void expectMapSectionAddress(ObjHandleT H, const void *LocalAddress,
|
void expectMapSectionAddress(VModuleKey K, const void *LocalAddress,
|
||||||
llvm::JITTargetAddress TargetAddr) {
|
llvm::JITTargetAddress TargetAddr) {
|
||||||
MockObjHandle = H;
|
MockKey = K;
|
||||||
MockLocalAddress = LocalAddress;
|
MockLocalAddress = LocalAddress;
|
||||||
MockTargetAddress = TargetAddr;
|
MockTargetAddress = TargetAddr;
|
||||||
}
|
}
|
||||||
|
@ -159,7 +153,6 @@ private:
|
||||||
std::string LastCalled;
|
std::string LastCalled;
|
||||||
VModuleKey MockKey;
|
VModuleKey MockKey;
|
||||||
MockObjectFile MockObject;
|
MockObjectFile MockObject;
|
||||||
ObjHandleT MockObjHandle;
|
|
||||||
std::string MockName;
|
std::string MockName;
|
||||||
bool MockBool;
|
bool MockBool;
|
||||||
llvm::JITSymbol MockSymbol;
|
llvm::JITSymbol MockSymbol;
|
||||||
|
@ -172,7 +165,6 @@ private:
|
||||||
LastCalled = "nothing";
|
LastCalled = "nothing";
|
||||||
MockKey = 0;
|
MockKey = 0;
|
||||||
MockObject = 0;
|
MockObject = 0;
|
||||||
MockObjHandle = 0;
|
|
||||||
MockName = "bogus";
|
MockName = "bogus";
|
||||||
MockSymbol = llvm::JITSymbol(nullptr);
|
MockSymbol = llvm::JITSymbol(nullptr);
|
||||||
MockLocalAddress = nullptr;
|
MockLocalAddress = nullptr;
|
||||||
|
@ -206,20 +198,20 @@ TEST(ObjectTransformLayerTest, Main) {
|
||||||
auto K1 = ES.allocateVModule();
|
auto K1 = ES.allocateVModule();
|
||||||
auto Obj1 = std::make_shared<MockObjectFile>(211);
|
auto Obj1 = std::make_shared<MockObjectFile>(211);
|
||||||
M.expectAddObject(K1, Obj1);
|
M.expectAddObject(K1, Obj1);
|
||||||
auto H = cantFail(T1.addObject(K1, std::move(Obj1)));
|
cantFail(T1.addObject(K1, std::move(Obj1)));
|
||||||
M.verifyAddObject(H);
|
M.verifyAddObject();
|
||||||
|
|
||||||
// Test addObjectSet with T2 (mutating)
|
// Test addObjectSet with T2 (mutating)
|
||||||
auto K2 = ES.allocateVModule();
|
auto K2 = ES.allocateVModule();
|
||||||
auto Obj2 = std::make_shared<MockObjectFile>(222);
|
auto Obj2 = std::make_shared<MockObjectFile>(222);
|
||||||
M.expectAddObject(K2, Obj2);
|
M.expectAddObject(K2, Obj2);
|
||||||
H = cantFail(T2.addObject(K2, Obj2));
|
cantFail(T2.addObject(K2, Obj2));
|
||||||
M.verifyAddObject(H);
|
M.verifyAddObject();
|
||||||
EXPECT_EQ(223, *Obj2) << "Expected mutation";
|
EXPECT_EQ(223, *Obj2) << "Expected mutation";
|
||||||
|
|
||||||
// Test removeObjectSet
|
// Test removeObjectSet
|
||||||
M.expectRemoveObject(H);
|
M.expectRemoveObject(K2);
|
||||||
cantFail(T1.removeObject(H));
|
cantFail(T1.removeObject(K2));
|
||||||
M.verifyRemoveObject();
|
M.verifyRemoveObject();
|
||||||
|
|
||||||
// Test findSymbol
|
// Test findSymbol
|
||||||
|
@ -232,20 +224,20 @@ TEST(ObjectTransformLayerTest, Main) {
|
||||||
// Test findSymbolIn
|
// Test findSymbolIn
|
||||||
Name = "bar";
|
Name = "bar";
|
||||||
ExportedOnly = false;
|
ExportedOnly = false;
|
||||||
M.expectFindSymbolIn(H, Name, ExportedOnly);
|
M.expectFindSymbolIn(K1, Name, ExportedOnly);
|
||||||
llvm::JITSymbol Sym2 = T1.findSymbolIn(H, Name, ExportedOnly);
|
llvm::JITSymbol Sym2 = T1.findSymbolIn(K1, Name, ExportedOnly);
|
||||||
M.verifyFindSymbolIn(std::move(Sym2));
|
M.verifyFindSymbolIn(std::move(Sym2));
|
||||||
|
|
||||||
// Test emitAndFinalize
|
// Test emitAndFinalize
|
||||||
M.expectEmitAndFinalize(H);
|
M.expectEmitAndFinalize(K1);
|
||||||
cantFail(T2.emitAndFinalize(H));
|
cantFail(T2.emitAndFinalize(K1));
|
||||||
M.verifyEmitAndFinalize();
|
M.verifyEmitAndFinalize();
|
||||||
|
|
||||||
// Test mapSectionAddress
|
// Test mapSectionAddress
|
||||||
char Buffer[24];
|
char Buffer[24];
|
||||||
llvm::JITTargetAddress MockAddress = 255;
|
llvm::JITTargetAddress MockAddress = 255;
|
||||||
M.expectMapSectionAddress(H, Buffer, MockAddress);
|
M.expectMapSectionAddress(K1, Buffer, MockAddress);
|
||||||
T1.mapSectionAddress(H, Buffer, MockAddress);
|
T1.mapSectionAddress(K1, Buffer, MockAddress);
|
||||||
M.verifyMapSectionAddress();
|
M.verifyMapSectionAddress();
|
||||||
|
|
||||||
// Verify transform getter (non-const)
|
// Verify transform getter (non-const)
|
||||||
|
@ -317,11 +309,11 @@ TEST(ObjectTransformLayerTest, Main) {
|
||||||
|
|
||||||
// Make sure that the calls from ObjectTransformLayer to ObjectLinkingLayer
|
// Make sure that the calls from ObjectTransformLayer to ObjectLinkingLayer
|
||||||
// compile.
|
// compile.
|
||||||
decltype(TransformLayer)::ObjHandleT H2;
|
VModuleKey DummyKey = ES.allocateVModule();
|
||||||
cantFail(TransformLayer.emitAndFinalize(H2));
|
cantFail(TransformLayer.emitAndFinalize(DummyKey));
|
||||||
TransformLayer.findSymbolIn(H2, Name, false);
|
TransformLayer.findSymbolIn(DummyKey, Name, false);
|
||||||
TransformLayer.findSymbol(Name, true);
|
TransformLayer.findSymbol(Name, true);
|
||||||
TransformLayer.mapSectionAddress(H2, nullptr, 0);
|
TransformLayer.mapSectionAddress(DummyKey, nullptr, 0);
|
||||||
cantFail(TransformLayer.removeObject(H2));
|
cantFail(TransformLayer.removeObject(DummyKey));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -100,21 +100,23 @@ TEST(RTDyldObjectLinkingLayerTest, TestSetProcessAllSections) {
|
||||||
|
|
||||||
{
|
{
|
||||||
// Test with ProcessAllSections = false (the default).
|
// Test with ProcessAllSections = false (the default).
|
||||||
auto H = cantFail(ObjLayer.addObject(ES.allocateVModule(), Obj));
|
auto K = ES.allocateVModule();
|
||||||
cantFail(ObjLayer.emitAndFinalize(H));
|
cantFail(ObjLayer.addObject(K, Obj));
|
||||||
|
cantFail(ObjLayer.emitAndFinalize(K));
|
||||||
EXPECT_EQ(DebugSectionSeen, false)
|
EXPECT_EQ(DebugSectionSeen, false)
|
||||||
<< "Unexpected debug info section";
|
<< "Unexpected debug info section";
|
||||||
cantFail(ObjLayer.removeObject(H));
|
cantFail(ObjLayer.removeObject(K));
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
// Test with ProcessAllSections = true.
|
// Test with ProcessAllSections = true.
|
||||||
ObjLayer.setProcessAllSections(true);
|
ObjLayer.setProcessAllSections(true);
|
||||||
auto H = cantFail(ObjLayer.addObject(ES.allocateVModule(), Obj));
|
auto K = ES.allocateVModule();
|
||||||
cantFail(ObjLayer.emitAndFinalize(H));
|
cantFail(ObjLayer.addObject(K, Obj));
|
||||||
|
cantFail(ObjLayer.emitAndFinalize(K));
|
||||||
EXPECT_EQ(DebugSectionSeen, true)
|
EXPECT_EQ(DebugSectionSeen, true)
|
||||||
<< "Expected debug info section not seen";
|
<< "Expected debug info section not seen";
|
||||||
cantFail(ObjLayer.removeObject(H));
|
cantFail(ObjLayer.removeObject(K));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -198,9 +200,9 @@ TEST_F(RTDyldObjectLinkingLayerExecutionTest, NoDuplicateFinalization) {
|
||||||
return lookupWithLegacyFn(Query, Symbols, LegacyLookup);
|
return lookupWithLegacyFn(Query, Symbols, LegacyLookup);
|
||||||
});
|
});
|
||||||
|
|
||||||
auto H = cantFail(ObjLayer.addObject(K2, std::move(Obj2)));
|
cantFail(ObjLayer.addObject(K2, std::move(Obj2)));
|
||||||
cantFail(ObjLayer.emitAndFinalize(H));
|
cantFail(ObjLayer.emitAndFinalize(K2));
|
||||||
cantFail(ObjLayer.removeObject(H));
|
cantFail(ObjLayer.removeObject(K2));
|
||||||
|
|
||||||
// Finalization of module 2 should trigger finalization of module 1.
|
// Finalization of module 2 should trigger finalization of module 1.
|
||||||
// Verify that finalize on SMMW is only called once.
|
// Verify that finalize on SMMW is only called once.
|
||||||
|
@ -264,10 +266,11 @@ TEST_F(RTDyldObjectLinkingLayerExecutionTest, NoPrematureAllocation) {
|
||||||
std::make_shared<object::OwningBinary<object::ObjectFile>>(
|
std::make_shared<object::OwningBinary<object::ObjectFile>>(
|
||||||
Compile(*MB2.getModule()));
|
Compile(*MB2.getModule()));
|
||||||
|
|
||||||
auto H = cantFail(ObjLayer.addObject(ES.allocateVModule(), std::move(Obj1)));
|
auto K = ES.allocateVModule();
|
||||||
|
cantFail(ObjLayer.addObject(K, std::move(Obj1)));
|
||||||
cantFail(ObjLayer.addObject(ES.allocateVModule(), std::move(Obj2)));
|
cantFail(ObjLayer.addObject(ES.allocateVModule(), std::move(Obj2)));
|
||||||
cantFail(ObjLayer.emitAndFinalize(H));
|
cantFail(ObjLayer.emitAndFinalize(K));
|
||||||
cantFail(ObjLayer.removeObject(H));
|
cantFail(ObjLayer.removeObject(K));
|
||||||
|
|
||||||
// Only one call to needsToReserveAllocationSpace should have been made.
|
// Only one call to needsToReserveAllocationSpace should have been made.
|
||||||
EXPECT_EQ(MM->NeedsToReserveAllocationSpaceCount, 1)
|
EXPECT_EQ(MM->NeedsToReserveAllocationSpaceCount, 1)
|
||||||
|
@ -281,8 +284,7 @@ TEST_F(RTDyldObjectLinkingLayerExecutionTest, TestNotifyLoadedSignature) {
|
||||||
RTDyldObjectLinkingLayer ObjLayer(
|
RTDyldObjectLinkingLayer ObjLayer(
|
||||||
ES, [](VModuleKey) { return nullptr; },
|
ES, [](VModuleKey) { return nullptr; },
|
||||||
[](VModuleKey) { return std::make_shared<NullResolver>(); },
|
[](VModuleKey) { return std::make_shared<NullResolver>(); },
|
||||||
[](RTDyldObjectLinkingLayer::ObjHandleT,
|
[](VModuleKey, const RTDyldObjectLinkingLayer::ObjectPtr &obj,
|
||||||
const RTDyldObjectLinkingLayer::ObjectPtr &obj,
|
|
||||||
const RuntimeDyld::LoadedObjectInfo &info) {});
|
const RuntimeDyld::LoadedObjectInfo &info) {});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue