Add another symbol generator to resolve the generated lazy symbol.
The in-process symbol generator resolves symbols available in the process and all dlopened libraries. However, upon unresolved symbol, we trigger our lazy loading function creator callbacks which load the relevant library. Then we need another in-process symbol generator to catch the newly created symbol so that the JIT is happy. This solution is suboptimal and in the comments is provided a cleaner implementation, however, this implementation loads libraries for weak symbols that the JIT could generate. This needs to be further investigated.
This commit is contained in:
parent
090176dcfd
commit
c8e1a89726
@ -24,18 +24,58 @@ using namespace llvm::orc;
|
|||||||
|
|
||||||
namespace cling {
|
namespace cling {
|
||||||
|
|
||||||
class NotifyLazyFunctionCreatorsGenerator : public DefinitionGenerator {
|
/// This class is a combination of the logic in DynamicLibrarySearchGenerator,
|
||||||
|
/// falling back to our symbol resolution logic.
|
||||||
|
class HostLookupLazyFallbackGenerator : public DefinitionGenerator {
|
||||||
const IncrementalExecutor & m_IncrExecutor;
|
const IncrementalExecutor & m_IncrExecutor;
|
||||||
|
char m_GlobalPrefix;
|
||||||
public:
|
public:
|
||||||
NotifyLazyFunctionCreatorsGenerator(const IncrementalExecutor &Exe)
|
HostLookupLazyFallbackGenerator(const IncrementalExecutor &Exe,
|
||||||
: m_IncrExecutor(Exe) { }
|
char GlobalPrefix)
|
||||||
|
: m_IncrExecutor(Exe), m_GlobalPrefix(GlobalPrefix) { }
|
||||||
|
|
||||||
Error tryToGenerate(LookupState& LS, LookupKind K, JITDylib& JD,
|
Error tryToGenerate(LookupState& LS, LookupKind K, JITDylib& JD,
|
||||||
JITDylibLookupFlags JDLookupFlags,
|
JITDylibLookupFlags JDLookupFlags,
|
||||||
const SymbolLookupSet& LookupSet) override {
|
const SymbolLookupSet& Symbols) override {
|
||||||
|
|
||||||
|
// FIXME: Uncomment when we figure out how to not load weak symbols from
|
||||||
|
// m_IncrExecutor.NotifyLazyFunctionCreators
|
||||||
|
|
||||||
|
// orc::SymbolMap NewSymbols;
|
||||||
|
|
||||||
|
// bool HasGlobalPrefix = (m_GlobalPrefix != '\0');
|
||||||
|
|
||||||
|
// for (auto &KV : Symbols) {
|
||||||
|
// auto &Name = KV.first;
|
||||||
|
|
||||||
|
// if ((*Name).empty())
|
||||||
|
// continue;
|
||||||
|
|
||||||
|
// if (HasGlobalPrefix && (*Name).front() != m_GlobalPrefix)
|
||||||
|
// continue;
|
||||||
|
|
||||||
|
// std::string Tmp((*Name).data() + HasGlobalPrefix,
|
||||||
|
// (*Name).size() - HasGlobalPrefix);
|
||||||
|
// void *Addr = sys::DynamicLibrary::SearchForAddressOfSymbol(Tmp.c_str());
|
||||||
|
// // FIXME: Here we will load random libraries due to weak symbols which is
|
||||||
|
// // suboptimal. We should let the JIT create them.
|
||||||
|
// if (!Addr)
|
||||||
|
// Addr = m_IncrExecutor.NotifyLazyFunctionCreators(Tmp.c_str());
|
||||||
|
// if (Addr) {
|
||||||
|
// NewSymbols[Name] = JITEvaluatedSymbol(
|
||||||
|
// static_cast<JITTargetAddress>(reinterpret_cast<uintptr_t>(Addr)),
|
||||||
|
// JITSymbolFlags::Exported);
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
// if (NewSymbols.empty())
|
||||||
|
// return Error::success();
|
||||||
|
|
||||||
|
// return JD.define(absoluteSymbols(std::move(NewSymbols)));
|
||||||
SymbolNameSet Missing;
|
SymbolNameSet Missing;
|
||||||
for (llvm::orc::SymbolStringPtr Name : LookupSet.getSymbolNames())
|
for (llvm::orc::SymbolStringPtr Name : Symbols.getSymbolNames())
|
||||||
if (!m_IncrExecutor.NotifyLazyFunctionCreators((*Name).str()))
|
if (!sys::DynamicLibrary::SearchForAddressOfSymbol((*Name).str()) &&
|
||||||
|
!m_IncrExecutor.NotifyLazyFunctionCreators((*Name).str()))
|
||||||
Missing.insert(Name);
|
Missing.insert(Name);
|
||||||
|
|
||||||
if (!Missing.empty())
|
if (!Missing.empty())
|
||||||
@ -88,17 +128,33 @@ IncrementalJIT::IncrementalJIT(
|
|||||||
});
|
});
|
||||||
|
|
||||||
// FIXME: Make host process symbol lookup optional on a per-query basis
|
// FIXME: Make host process symbol lookup optional on a per-query basis
|
||||||
|
|
||||||
char LinkerPrefix = this->TM->createDataLayout().getGlobalPrefix();
|
char LinkerPrefix = this->TM->createDataLayout().getGlobalPrefix();
|
||||||
|
|
||||||
|
// Process symbol resolution
|
||||||
Expected<std::unique_ptr<DynamicLibrarySearchGenerator>> HostProcessLookup =
|
Expected<std::unique_ptr<DynamicLibrarySearchGenerator>> HostProcessLookup =
|
||||||
|
DynamicLibrarySearchGenerator::GetForCurrentProcess(LinkerPrefix);
|
||||||
|
if (!HostProcessLookup) {
|
||||||
|
Err = HostProcessLookup.takeError();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
Jit->getMainJITDylib().addGenerator(std::move(*HostProcessLookup));
|
||||||
|
|
||||||
|
// Lazy symbol generation callback
|
||||||
|
auto Notifier =
|
||||||
|
std::make_unique<HostLookupLazyFallbackGenerator>(Executor, LinkerPrefix);
|
||||||
|
Jit->getMainJITDylib().addGenerator(std::move(Notifier));
|
||||||
|
|
||||||
|
// Process symbol resolution after the callback.
|
||||||
|
// FIXME: if we resolve the FIXME in HostLookupLazyFallbackGenerator, we will
|
||||||
|
// need just one generator.
|
||||||
|
HostProcessLookup =
|
||||||
DynamicLibrarySearchGenerator::GetForCurrentProcess(LinkerPrefix);
|
DynamicLibrarySearchGenerator::GetForCurrentProcess(LinkerPrefix);
|
||||||
if (!HostProcessLookup) {
|
if (!HostProcessLookup) {
|
||||||
Err = HostProcessLookup.takeError();
|
Err = HostProcessLookup.takeError();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
Jit->getMainJITDylib().addGenerator(std::move(*HostProcessLookup));
|
Jit->getMainJITDylib().addGenerator(std::move(*HostProcessLookup));
|
||||||
auto Notifier = std::make_unique<NotifyLazyFunctionCreatorsGenerator>(Executor);
|
|
||||||
Jit->getMainJITDylib().addGenerator(std::move(Notifier));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void IncrementalJIT::addModule(Transaction& T) {
|
void IncrementalJIT::addModule(Transaction& T) {
|
||||||
|
Loading…
Reference in New Issue
Block a user