mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-11-25 04:02:41 +01:00
06bc5445fb
and remove the JITSymbolFlags header. llvm-svn: 277766
541 lines
20 KiB
C++
541 lines
20 KiB
C++
//===- CompileOnDemandLayer.h - Compile each function on demand -*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// JIT layer for breaking up modules and inserting callbacks to allow
|
|
// individual functions to be compiled on demand.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLVM_EXECUTIONENGINE_ORC_COMPILEONDEMANDLAYER_H
|
|
#define LLVM_EXECUTIONENGINE_ORC_COMPILEONDEMANDLAYER_H
|
|
|
|
#include "IndirectionUtils.h"
|
|
#include "LambdaResolver.h"
|
|
#include "LogicalDylib.h"
|
|
#include "llvm/ADT/STLExtras.h"
|
|
#include "llvm/Support/Debug.h"
|
|
#include "llvm/Transforms/Utils/Cloning.h"
|
|
#include <list>
|
|
#include <memory>
|
|
#include <set>
|
|
#include <utility>
|
|
|
|
namespace llvm {
|
|
namespace orc {
|
|
|
|
/// @brief Compile-on-demand layer.
|
|
///
|
|
/// When a module is added to this layer a stub is created for each of its
|
|
/// function definitions. The stubs and other global values are immediately
|
|
/// added to the layer below. When a stub is called it triggers the extraction
|
|
/// of the function body from the original module. The extracted body is then
|
|
/// compiled and executed.
|
|
template <typename BaseLayerT,
|
|
typename CompileCallbackMgrT = JITCompileCallbackManager,
|
|
typename IndirectStubsMgrT = IndirectStubsManager>
|
|
class CompileOnDemandLayer {
|
|
private:
|
|
|
|
template <typename MaterializerFtor>
|
|
class LambdaMaterializer final : public ValueMaterializer {
|
|
public:
|
|
LambdaMaterializer(MaterializerFtor M) : M(std::move(M)) {}
|
|
Value *materialize(Value *V) final { return M(V); }
|
|
|
|
private:
|
|
MaterializerFtor M;
|
|
};
|
|
|
|
template <typename MaterializerFtor>
|
|
LambdaMaterializer<MaterializerFtor>
|
|
createLambdaMaterializer(MaterializerFtor M) {
|
|
return LambdaMaterializer<MaterializerFtor>(std::move(M));
|
|
}
|
|
|
|
typedef typename BaseLayerT::ModuleSetHandleT BaseLayerModuleSetHandleT;
|
|
|
|
// Provide type-erasure for the Modules and MemoryManagers.
|
|
template <typename ResourceT>
|
|
class ResourceOwner {
|
|
public:
|
|
ResourceOwner() = default;
|
|
ResourceOwner(const ResourceOwner&) = delete;
|
|
ResourceOwner& operator=(const ResourceOwner&) = delete;
|
|
virtual ~ResourceOwner() { }
|
|
virtual ResourceT& getResource() const = 0;
|
|
};
|
|
|
|
template <typename ResourceT, typename ResourcePtrT>
|
|
class ResourceOwnerImpl : public ResourceOwner<ResourceT> {
|
|
public:
|
|
ResourceOwnerImpl(ResourcePtrT ResourcePtr)
|
|
: ResourcePtr(std::move(ResourcePtr)) {}
|
|
ResourceT& getResource() const override { return *ResourcePtr; }
|
|
private:
|
|
ResourcePtrT ResourcePtr;
|
|
};
|
|
|
|
template <typename ResourceT, typename ResourcePtrT>
|
|
std::unique_ptr<ResourceOwner<ResourceT>>
|
|
wrapOwnership(ResourcePtrT ResourcePtr) {
|
|
typedef ResourceOwnerImpl<ResourceT, ResourcePtrT> RO;
|
|
return llvm::make_unique<RO>(std::move(ResourcePtr));
|
|
}
|
|
|
|
struct LogicalModuleResources {
|
|
std::unique_ptr<ResourceOwner<Module>> SourceModule;
|
|
std::set<const Function*> StubsToClone;
|
|
std::unique_ptr<IndirectStubsMgrT> StubsMgr;
|
|
|
|
LogicalModuleResources() = default;
|
|
|
|
// Explicit move constructor to make MSVC happy.
|
|
LogicalModuleResources(LogicalModuleResources &&Other)
|
|
: SourceModule(std::move(Other.SourceModule)),
|
|
StubsToClone(std::move(Other.StubsToClone)),
|
|
StubsMgr(std::move(Other.StubsMgr)) {}
|
|
|
|
// Explicit move assignment to make MSVC happy.
|
|
LogicalModuleResources& operator=(LogicalModuleResources &&Other) {
|
|
SourceModule = std::move(Other.SourceModule);
|
|
StubsToClone = std::move(Other.StubsToClone);
|
|
StubsMgr = std::move(Other.StubsMgr);
|
|
return *this;
|
|
}
|
|
|
|
JITSymbol findSymbol(StringRef Name, bool ExportedSymbolsOnly) {
|
|
if (Name.endswith("$stub_ptr") && !ExportedSymbolsOnly) {
|
|
assert(!ExportedSymbolsOnly && "Stubs are never exported");
|
|
return StubsMgr->findPointer(Name.drop_back(9));
|
|
}
|
|
return StubsMgr->findStub(Name, ExportedSymbolsOnly);
|
|
}
|
|
|
|
};
|
|
|
|
struct LogicalDylibResources {
|
|
typedef std::function<JITSymbol(const std::string&)> SymbolResolverFtor;
|
|
|
|
typedef std::function<typename BaseLayerT::ModuleSetHandleT(
|
|
BaseLayerT&,
|
|
std::unique_ptr<Module>,
|
|
std::unique_ptr<JITSymbolResolver>)>
|
|
ModuleAdderFtor;
|
|
|
|
LogicalDylibResources() = default;
|
|
|
|
// Explicit move constructor to make MSVC happy.
|
|
LogicalDylibResources(LogicalDylibResources &&Other)
|
|
: ExternalSymbolResolver(std::move(Other.ExternalSymbolResolver)),
|
|
MemMgr(std::move(Other.MemMgr)),
|
|
ModuleAdder(std::move(Other.ModuleAdder)) {}
|
|
|
|
// Explicit move assignment operator to make MSVC happy.
|
|
LogicalDylibResources& operator=(LogicalDylibResources &&Other) {
|
|
ExternalSymbolResolver = std::move(Other.ExternalSymbolResolver);
|
|
MemMgr = std::move(Other.MemMgr);
|
|
ModuleAdder = std::move(Other.ModuleAdder);
|
|
return *this;
|
|
}
|
|
|
|
std::unique_ptr<JITSymbolResolver> ExternalSymbolResolver;
|
|
std::unique_ptr<ResourceOwner<RuntimeDyld::MemoryManager>> MemMgr;
|
|
ModuleAdderFtor ModuleAdder;
|
|
};
|
|
|
|
typedef LogicalDylib<BaseLayerT, LogicalModuleResources,
|
|
LogicalDylibResources> CODLogicalDylib;
|
|
|
|
typedef typename CODLogicalDylib::LogicalModuleHandle LogicalModuleHandle;
|
|
typedef std::list<CODLogicalDylib> LogicalDylibList;
|
|
|
|
public:
|
|
|
|
/// @brief Handle to a set of loaded modules.
|
|
typedef typename LogicalDylibList::iterator ModuleSetHandleT;
|
|
|
|
/// @brief Module partitioning functor.
|
|
typedef std::function<std::set<Function*>(Function&)> PartitioningFtor;
|
|
|
|
/// @brief Builder for IndirectStubsManagers.
|
|
typedef std::function<std::unique_ptr<IndirectStubsMgrT>()>
|
|
IndirectStubsManagerBuilderT;
|
|
|
|
/// @brief Construct a compile-on-demand layer instance.
|
|
CompileOnDemandLayer(BaseLayerT &BaseLayer, PartitioningFtor Partition,
|
|
CompileCallbackMgrT &CallbackMgr,
|
|
IndirectStubsManagerBuilderT CreateIndirectStubsManager,
|
|
bool CloneStubsIntoPartitions = true)
|
|
: BaseLayer(BaseLayer), Partition(std::move(Partition)),
|
|
CompileCallbackMgr(CallbackMgr),
|
|
CreateIndirectStubsManager(std::move(CreateIndirectStubsManager)),
|
|
CloneStubsIntoPartitions(CloneStubsIntoPartitions) {}
|
|
|
|
/// @brief Add a module to the compile-on-demand layer.
|
|
template <typename ModuleSetT, typename MemoryManagerPtrT,
|
|
typename SymbolResolverPtrT>
|
|
ModuleSetHandleT addModuleSet(ModuleSetT Ms,
|
|
MemoryManagerPtrT MemMgr,
|
|
SymbolResolverPtrT Resolver) {
|
|
|
|
LogicalDylibs.push_back(CODLogicalDylib(BaseLayer));
|
|
auto &LDResources = LogicalDylibs.back().getDylibResources();
|
|
|
|
LDResources.ExternalSymbolResolver = std::move(Resolver);
|
|
|
|
auto &MemMgrRef = *MemMgr;
|
|
LDResources.MemMgr =
|
|
wrapOwnership<RuntimeDyld::MemoryManager>(std::move(MemMgr));
|
|
|
|
LDResources.ModuleAdder =
|
|
[&MemMgrRef](BaseLayerT &B, std::unique_ptr<Module> M,
|
|
std::unique_ptr<JITSymbolResolver> R) {
|
|
std::vector<std::unique_ptr<Module>> Ms;
|
|
Ms.push_back(std::move(M));
|
|
return B.addModuleSet(std::move(Ms), &MemMgrRef, std::move(R));
|
|
};
|
|
|
|
// Process each of the modules in this module set.
|
|
for (auto &M : Ms)
|
|
addLogicalModule(LogicalDylibs.back(), std::move(M));
|
|
|
|
return std::prev(LogicalDylibs.end());
|
|
}
|
|
|
|
/// @brief Remove the module represented by the given handle.
|
|
///
|
|
/// This will remove all modules in the layers below that were derived from
|
|
/// the module represented by H.
|
|
void removeModuleSet(ModuleSetHandleT H) {
|
|
LogicalDylibs.erase(H);
|
|
}
|
|
|
|
/// @brief Search for the given named symbol.
|
|
/// @param Name The name of the symbol to search for.
|
|
/// @param ExportedSymbolsOnly If true, search only for exported symbols.
|
|
/// @return A handle for the given named symbol, if it exists.
|
|
JITSymbol findSymbol(StringRef Name, bool ExportedSymbolsOnly) {
|
|
for (auto LDI = LogicalDylibs.begin(), LDE = LogicalDylibs.end();
|
|
LDI != LDE; ++LDI)
|
|
if (auto Symbol = findSymbolIn(LDI, Name, ExportedSymbolsOnly))
|
|
return Symbol;
|
|
return BaseLayer.findSymbol(Name, ExportedSymbolsOnly);
|
|
}
|
|
|
|
/// @brief Get the address of a symbol provided by this layer, or some layer
|
|
/// below this one.
|
|
JITSymbol findSymbolIn(ModuleSetHandleT H, const std::string &Name,
|
|
bool ExportedSymbolsOnly) {
|
|
return H->findSymbol(Name, ExportedSymbolsOnly);
|
|
}
|
|
|
|
/// @brief Update the stub for the given function to point at FnBodyAddr.
|
|
/// This can be used to support re-optimization.
|
|
/// @return true if the function exists and the stub is updated, false
|
|
/// otherwise.
|
|
//
|
|
// FIXME: We should track and free associated resources (unused compile
|
|
// callbacks, uncompiled IR, and no-longer-needed/reachable function
|
|
// implementations).
|
|
// FIXME: Return Error once the JIT APIs are Errorized.
|
|
bool updatePointer(std::string FuncName, JITTargetAddress FnBodyAddr) {
|
|
//Find out which logical dylib contains our symbol
|
|
auto LDI = LogicalDylibs.begin();
|
|
for (auto LDE = LogicalDylibs.end(); LDI != LDE; ++LDI) {
|
|
if (auto LMResources = LDI->getLogicalModuleResourcesForSymbol(FuncName, false)) {
|
|
Module &SrcM = LMResources->SourceModule->getResource();
|
|
std::string CalledFnName = mangle(FuncName, SrcM.getDataLayout());
|
|
if (auto EC = LMResources->StubsMgr->updatePointer(CalledFnName, FnBodyAddr)) {
|
|
return false;
|
|
}
|
|
else
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
private:
|
|
|
|
template <typename ModulePtrT>
|
|
void addLogicalModule(CODLogicalDylib &LD, ModulePtrT SrcMPtr) {
|
|
|
|
// Bump the linkage and rename any anonymous/privote members in SrcM to
|
|
// ensure that everything will resolve properly after we partition SrcM.
|
|
makeAllSymbolsExternallyAccessible(*SrcMPtr);
|
|
|
|
// Create a logical module handle for SrcM within the logical dylib.
|
|
auto LMH = LD.createLogicalModule();
|
|
auto &LMResources = LD.getLogicalModuleResources(LMH);
|
|
|
|
LMResources.SourceModule = wrapOwnership<Module>(std::move(SrcMPtr));
|
|
|
|
Module &SrcM = LMResources.SourceModule->getResource();
|
|
|
|
// Create stub functions.
|
|
const DataLayout &DL = SrcM.getDataLayout();
|
|
{
|
|
typename IndirectStubsMgrT::StubInitsMap StubInits;
|
|
for (auto &F : SrcM) {
|
|
// Skip declarations.
|
|
if (F.isDeclaration())
|
|
continue;
|
|
|
|
// Record all functions defined by this module.
|
|
if (CloneStubsIntoPartitions)
|
|
LMResources.StubsToClone.insert(&F);
|
|
|
|
// Create a callback, associate it with the stub for the function,
|
|
// and set the compile action to compile the partition containing the
|
|
// function.
|
|
auto CCInfo = CompileCallbackMgr.getCompileCallback();
|
|
StubInits[mangle(F.getName(), DL)] =
|
|
std::make_pair(CCInfo.getAddress(),
|
|
JITSymbolFlags::fromGlobalValue(F));
|
|
CCInfo.setCompileAction([this, &LD, LMH, &F]() {
|
|
return this->extractAndCompile(LD, LMH, F);
|
|
});
|
|
}
|
|
|
|
LMResources.StubsMgr = CreateIndirectStubsManager();
|
|
auto EC = LMResources.StubsMgr->createStubs(StubInits);
|
|
(void)EC;
|
|
// FIXME: This should be propagated back to the user. Stub creation may
|
|
// fail for remote JITs.
|
|
assert(!EC && "Error generating stubs");
|
|
}
|
|
|
|
// If this module doesn't contain any globals or aliases we can bail out
|
|
// early and avoid the overhead of creating and managing an empty globals
|
|
// module.
|
|
if (SrcM.global_empty() && SrcM.alias_empty())
|
|
return;
|
|
|
|
// Create the GlobalValues module.
|
|
auto GVsM = llvm::make_unique<Module>((SrcM.getName() + ".globals").str(),
|
|
SrcM.getContext());
|
|
GVsM->setDataLayout(DL);
|
|
|
|
ValueToValueMapTy VMap;
|
|
|
|
// Clone global variable decls.
|
|
for (auto &GV : SrcM.globals())
|
|
if (!GV.isDeclaration() && !VMap.count(&GV))
|
|
cloneGlobalVariableDecl(*GVsM, GV, &VMap);
|
|
|
|
// And the aliases.
|
|
for (auto &A : SrcM.aliases())
|
|
if (!VMap.count(&A))
|
|
cloneGlobalAliasDecl(*GVsM, A, VMap);
|
|
|
|
// Now we need to clone the GV and alias initializers.
|
|
|
|
// Initializers may refer to functions declared (but not defined) in this
|
|
// module. Build a materializer to clone decls on demand.
|
|
auto Materializer = createLambdaMaterializer(
|
|
[this, &GVsM, &LMResources](Value *V) -> Value* {
|
|
if (auto *F = dyn_cast<Function>(V)) {
|
|
// Decls in the original module just get cloned.
|
|
if (F->isDeclaration())
|
|
return cloneFunctionDecl(*GVsM, *F);
|
|
|
|
// Definitions in the original module (which we have emitted stubs
|
|
// for at this point) get turned into a constant alias to the stub
|
|
// instead.
|
|
const DataLayout &DL = GVsM->getDataLayout();
|
|
std::string FName = mangle(F->getName(), DL);
|
|
auto StubSym = LMResources.StubsMgr->findStub(FName, false);
|
|
unsigned PtrBitWidth = DL.getPointerTypeSizeInBits(F->getType());
|
|
ConstantInt *StubAddr =
|
|
ConstantInt::get(GVsM->getContext(),
|
|
APInt(PtrBitWidth, StubSym.getAddress()));
|
|
Constant *Init = ConstantExpr::getCast(Instruction::IntToPtr,
|
|
StubAddr, F->getType());
|
|
return GlobalAlias::create(F->getFunctionType(),
|
|
F->getType()->getAddressSpace(),
|
|
F->getLinkage(), F->getName(),
|
|
Init, GVsM.get());
|
|
}
|
|
// else....
|
|
return nullptr;
|
|
});
|
|
|
|
// Clone the global variable initializers.
|
|
for (auto &GV : SrcM.globals())
|
|
if (!GV.isDeclaration())
|
|
moveGlobalVariableInitializer(GV, VMap, &Materializer);
|
|
|
|
// Clone the global alias initializers.
|
|
for (auto &A : SrcM.aliases()) {
|
|
auto *NewA = cast<GlobalAlias>(VMap[&A]);
|
|
assert(NewA && "Alias not cloned?");
|
|
Value *Init = MapValue(A.getAliasee(), VMap, RF_None, nullptr,
|
|
&Materializer);
|
|
NewA->setAliasee(cast<Constant>(Init));
|
|
}
|
|
|
|
// Build a resolver for the globals module and add it to the base layer.
|
|
auto GVsResolver = createLambdaResolver(
|
|
[&LD, LMH](const std::string &Name) {
|
|
auto &LMResources = LD.getLogicalModuleResources(LMH);
|
|
if (auto Sym = LMResources.StubsMgr->findStub(Name, false))
|
|
return Sym;
|
|
auto &LDResolver = LD.getDylibResources().ExternalSymbolResolver;
|
|
return LDResolver->findSymbolInLogicalDylib(Name);
|
|
},
|
|
[&LD](const std::string &Name) {
|
|
auto &LDResolver = LD.getDylibResources().ExternalSymbolResolver;
|
|
return LDResolver->findSymbol(Name);
|
|
});
|
|
|
|
auto GVsH = LD.getDylibResources().ModuleAdder(BaseLayer, std::move(GVsM),
|
|
std::move(GVsResolver));
|
|
LD.addToLogicalModule(LMH, GVsH);
|
|
}
|
|
|
|
static std::string mangle(StringRef Name, const DataLayout &DL) {
|
|
std::string MangledName;
|
|
{
|
|
raw_string_ostream MangledNameStream(MangledName);
|
|
Mangler::getNameWithPrefix(MangledNameStream, Name, DL);
|
|
}
|
|
return MangledName;
|
|
}
|
|
|
|
JITTargetAddress extractAndCompile(CODLogicalDylib &LD,
|
|
LogicalModuleHandle LMH,
|
|
Function &F) {
|
|
auto &LMResources = LD.getLogicalModuleResources(LMH);
|
|
Module &SrcM = LMResources.SourceModule->getResource();
|
|
|
|
// If F is a declaration we must already have compiled it.
|
|
if (F.isDeclaration())
|
|
return 0;
|
|
|
|
// Grab the name of the function being called here.
|
|
std::string CalledFnName = mangle(F.getName(), SrcM.getDataLayout());
|
|
|
|
auto Part = Partition(F);
|
|
auto PartH = emitPartition(LD, LMH, Part);
|
|
|
|
JITTargetAddress CalledAddr = 0;
|
|
for (auto *SubF : Part) {
|
|
std::string FnName = mangle(SubF->getName(), SrcM.getDataLayout());
|
|
auto FnBodySym = BaseLayer.findSymbolIn(PartH, FnName, false);
|
|
assert(FnBodySym && "Couldn't find function body.");
|
|
|
|
JITTargetAddress FnBodyAddr = FnBodySym.getAddress();
|
|
|
|
// If this is the function we're calling record the address so we can
|
|
// return it from this function.
|
|
if (SubF == &F)
|
|
CalledAddr = FnBodyAddr;
|
|
|
|
// Update the function body pointer for the stub.
|
|
if (auto EC = LMResources.StubsMgr->updatePointer(FnName, FnBodyAddr))
|
|
return 0;
|
|
}
|
|
|
|
return CalledAddr;
|
|
}
|
|
|
|
template <typename PartitionT>
|
|
BaseLayerModuleSetHandleT emitPartition(CODLogicalDylib &LD,
|
|
LogicalModuleHandle LMH,
|
|
const PartitionT &Part) {
|
|
auto &LMResources = LD.getLogicalModuleResources(LMH);
|
|
Module &SrcM = LMResources.SourceModule->getResource();
|
|
|
|
// Create the module.
|
|
std::string NewName = SrcM.getName();
|
|
for (auto *F : Part) {
|
|
NewName += ".";
|
|
NewName += F->getName();
|
|
}
|
|
|
|
auto M = llvm::make_unique<Module>(NewName, SrcM.getContext());
|
|
M->setDataLayout(SrcM.getDataLayout());
|
|
ValueToValueMapTy VMap;
|
|
|
|
auto Materializer = createLambdaMaterializer([this, &LMResources, &M,
|
|
&VMap](Value *V) -> Value * {
|
|
if (auto *GV = dyn_cast<GlobalVariable>(V))
|
|
return cloneGlobalVariableDecl(*M, *GV);
|
|
|
|
if (auto *F = dyn_cast<Function>(V)) {
|
|
// Check whether we want to clone an available_externally definition.
|
|
if (!LMResources.StubsToClone.count(F))
|
|
return cloneFunctionDecl(*M, *F);
|
|
|
|
// Ok - we want an inlinable stub. For that to work we need a decl
|
|
// for the stub pointer.
|
|
auto *StubPtr = createImplPointer(*F->getType(), *M,
|
|
F->getName() + "$stub_ptr", nullptr);
|
|
auto *ClonedF = cloneFunctionDecl(*M, *F);
|
|
makeStub(*ClonedF, *StubPtr);
|
|
ClonedF->setLinkage(GlobalValue::AvailableExternallyLinkage);
|
|
ClonedF->addFnAttr(Attribute::AlwaysInline);
|
|
return ClonedF;
|
|
}
|
|
|
|
if (auto *A = dyn_cast<GlobalAlias>(V)) {
|
|
auto *Ty = A->getValueType();
|
|
if (Ty->isFunctionTy())
|
|
return Function::Create(cast<FunctionType>(Ty),
|
|
GlobalValue::ExternalLinkage, A->getName(),
|
|
M.get());
|
|
|
|
return new GlobalVariable(*M, Ty, false, GlobalValue::ExternalLinkage,
|
|
nullptr, A->getName(), nullptr,
|
|
GlobalValue::NotThreadLocal,
|
|
A->getType()->getAddressSpace());
|
|
}
|
|
|
|
return nullptr;
|
|
});
|
|
|
|
// Create decls in the new module.
|
|
for (auto *F : Part)
|
|
cloneFunctionDecl(*M, *F, &VMap);
|
|
|
|
// Move the function bodies.
|
|
for (auto *F : Part)
|
|
moveFunctionBody(*F, VMap, &Materializer);
|
|
|
|
// Create memory manager and symbol resolver.
|
|
auto Resolver = createLambdaResolver(
|
|
[this, &LD, LMH](const std::string &Name) {
|
|
if (auto Sym = LD.findSymbolInternally(LMH, Name))
|
|
return Sym;
|
|
auto &LDResolver = LD.getDylibResources().ExternalSymbolResolver;
|
|
return LDResolver->findSymbolInLogicalDylib(Name);
|
|
},
|
|
[this, &LD](const std::string &Name) {
|
|
auto &LDResolver = LD.getDylibResources().ExternalSymbolResolver;
|
|
return LDResolver->findSymbol(Name);
|
|
});
|
|
|
|
return LD.getDylibResources().ModuleAdder(BaseLayer, std::move(M),
|
|
std::move(Resolver));
|
|
}
|
|
|
|
BaseLayerT &BaseLayer;
|
|
PartitioningFtor Partition;
|
|
CompileCallbackMgrT &CompileCallbackMgr;
|
|
IndirectStubsManagerBuilderT CreateIndirectStubsManager;
|
|
|
|
LogicalDylibList LogicalDylibs;
|
|
bool CloneStubsIntoPartitions;
|
|
};
|
|
|
|
} // End namespace orc.
|
|
} // End namespace llvm.
|
|
|
|
#endif // LLVM_EXECUTIONENGINE_ORC_COMPILEONDEMANDLAYER_H
|