cling/lib/Interpreter/IncrementalJIT.cpp
Kai Luo abeda742aa Backport jitlink ppc64 (#13850)
* Backport JITLink ppc64 backend to LLVM-16

* Backport ELF part

* Backport PCREL relocations

* Use jitlink::Section::blocks::empty instead

* Backport TLS PCREL relocation

* [llvm-project] Synchronize with LLVM monorepo fork

---------

Co-authored-by: Jonas Hahnfeld <jonas.hahnfeld@cern.ch>
2024-03-21 15:59:05 +01:00

721 lines
27 KiB
C++

//--------------------------------------------------------------------*- C++ -*-
// CLING - the C++ LLVM-based InterpreterG :)
// author: Stefan Gränitz <stefan.graenitz@gmail.com>
//
// This file is dual-licensed: you can choose to license it under the University
// of Illinois Open Source License or the GNU Lesser General Public License. See
// LICENSE.TXT for details.
//------------------------------------------------------------------------------
#include "IncrementalJIT.h"
// FIXME: Merge IncrementalExecutor and IncrementalJIT.
#include "IncrementalExecutor.h"
#include "cling/Utils/Output.h"
#include "cling/Utils/Utils.h"
#include <clang/Basic/TargetInfo.h>
#include <clang/Basic/TargetOptions.h>
#include <clang/Frontend/CompilerInstance.h>
#include <llvm/ADT/Triple.h>
#include <llvm/ExecutionEngine/JITLink/EHFrameSupport.h>
#include <llvm/ExecutionEngine/Orc/JITTargetMachineBuilder.h>
#include <llvm/ExecutionEngine/Orc/ObjectLinkingLayer.h>
#include <llvm/ExecutionEngine/Orc/RTDyldObjectLinkingLayer.h>
#include <llvm/ExecutionEngine/SectionMemoryManager.h>
#include <llvm/IR/LLVMContext.h>
#include <llvm/MC/TargetRegistry.h>
#include <llvm/Support/raw_ostream.h>
#include <llvm/Support/Host.h>
#include <llvm/Target/TargetMachine.h>
#include <optional>
#ifdef __linux__
#include <sys/stat.h>
#endif
using namespace llvm;
using namespace llvm::jitlink;
using namespace llvm::orc;
namespace {
class ClingMMapper final : public SectionMemoryManager::MemoryMapper {
public:
sys::MemoryBlock
allocateMappedMemory(SectionMemoryManager::AllocationPurpose Purpose,
size_t NumBytes,
const sys::MemoryBlock* const NearBlock,
unsigned Flags, std::error_code& EC) override {
return sys::Memory::allocateMappedMemory(NumBytes, NearBlock, Flags, EC);
}
std::error_code protectMappedMemory(const sys::MemoryBlock& Block,
unsigned Flags) override {
return sys::Memory::protectMappedMemory(Block, Flags);
}
std::error_code releaseMappedMemory(sys::MemoryBlock& M) override {
// Disabled until CallFunc is informed about unloading, and can
// re-generate the wrapper (if the decl is still available). See
// https://github.com/root-project/root/issues/10898
#if 0
return sys::Memory::releaseMappedMemory(M);
#else
return {};
#endif
}
};
ClingMMapper MMapperInstance;
// A memory manager for Cling that reserves memory for code and data sections
// to keep them contiguous for the emission of one module. This is required
// for working exception handling support since one .eh_frame section will
// refer to many separate .text sections. However, stack unwinding in libgcc
// assumes that two unwinding objects (for example coming from two modules)
// are non-overlapping, which is hard to guarantee with separate allocations
// for the individual code sections.
class ClingMemoryManager : public SectionMemoryManager {
using Super = SectionMemoryManager;
struct AllocInfo {
uint8_t* m_End = nullptr;
uint8_t* m_Current = nullptr;
void setAllocation(uint8_t* Addr, uintptr_t Size) {
m_Current = Addr;
m_End = Addr + Size;
}
uint8_t* getNextAddr(uintptr_t Size, unsigned Alignment) {
if (!Alignment)
Alignment = 16;
assert(!(Alignment & (Alignment - 1)) &&
"Alignment must be a power of two.");
uintptr_t RequiredSize =
Alignment * ((Size + Alignment - 1) / Alignment + 1);
if ((m_Current + RequiredSize) > m_End) {
// This must be the last block.
if ((m_Current + Size) <= m_End) {
RequiredSize = Size;
} else {
cling::errs()
<< "Error in block allocation by ClingMemoryManager.\n"
<< "Not enough memory was reserved for the current module.\n"
<< Size << " (with alignment: " << RequiredSize
<< " ) is needed but we only have " << (m_End - m_Current)
<< ".\n";
return nullptr;
}
}
uintptr_t Addr = (uintptr_t)m_Current;
// Align the address.
Addr = (Addr + Alignment - 1) & ~(uintptr_t)(Alignment - 1);
m_Current = (uint8_t*)(Addr + Size);
return (uint8_t*)Addr;
}
operator bool() { return m_Current != nullptr; }
};
AllocInfo m_Code;
AllocInfo m_ROData;
AllocInfo m_RWData;
public:
ClingMemoryManager() : Super(&MMapperInstance) {}
uint8_t* allocateCodeSection(uintptr_t Size, unsigned Alignment,
unsigned SectionID,
StringRef SectionName) override {
uint8_t* Addr = nullptr;
if (m_Code) {
Addr = m_Code.getNextAddr(Size, Alignment);
}
if (!Addr) {
Addr =
Super::allocateCodeSection(Size, Alignment, SectionID, SectionName);
}
return Addr;
}
uint8_t* allocateDataSection(uintptr_t Size, unsigned Alignment,
unsigned SectionID, StringRef SectionName,
bool IsReadOnly) override {
uint8_t* Addr = nullptr;
if (IsReadOnly) {
if (m_ROData) {
Addr = m_ROData.getNextAddr(Size, Alignment);
}
} else if (m_RWData) {
Addr = m_RWData.getNextAddr(Size, Alignment);
}
if (!Addr) {
Addr = Super::allocateDataSection(Size, Alignment, SectionID,
SectionName, IsReadOnly);
}
return Addr;
}
void reserveAllocationSpace(uintptr_t CodeSize, Align CodeAlign,
uintptr_t RODataSize, Align RODataAlign,
uintptr_t RWDataSize,
Align RWDataAlign) override {
m_Code.setAllocation(
Super::allocateCodeSection(CodeSize, CodeAlign.value(),
/*SectionID=*/0,
/*SectionName=*/"codeReserve"),
CodeSize);
m_ROData.setAllocation(
Super::allocateDataSection(RODataSize, RODataAlign.value(),
/*SectionID=*/0,
/*SectionName=*/"rodataReserve",
/*IsReadOnly=*/true),
RODataSize);
m_RWData.setAllocation(
Super::allocateDataSection(RWDataSize, RWDataAlign.value(),
/*SectionID=*/0,
/*SectionName=*/"rwataReserve",
/*IsReadOnly=*/false),
RWDataSize);
}
bool needsToReserveAllocationSpace() override { return true; }
};
/// A JITLinkMemoryManager for Cling that never frees its allocations.
class ClingJITLinkMemoryManager : public InProcessMemoryManager {
public:
using InProcessMemoryManager::InProcessMemoryManager;
void deallocate(std::vector<FinalizedAlloc> Allocs,
OnDeallocatedFunction OnDeallocated) override {
// Disabled until CallFunc is informed about unloading, and can
// re-generate the wrapper (if the decl is still available). See
// https://github.com/root-project/root/issues/10898
// We still have to release the allocations which resets their addresses
// to FinalizedAlloc::InvalidAddr, or the assertion in ~FinalizedAlloc
// will be unhappy...
for (auto &Alloc : Allocs) {
Alloc.release();
}
// Pretend we successfully deallocated everything...
OnDeallocated(Error::success());
}
};
/// A DynamicLibrarySearchGenerator that uses ResourceTracker to remember
/// which symbols were resolved through dlsym during a transaction's reign.
/// Enables JITDyLib forgetting symbols upon unloading of a shared library.
/// While JITDylib::define() *is* invoked for these symbols, there is no RT
/// provided, and thus resource tracking doesn't work, no symbol removal
/// happens upon unloading the corresponding shared library.
///
/// This might remove more symbols than strictly needed:
/// 1. libA is loaded
/// 2. libB is loaded
/// 3. symbol is resolved from libA
/// 4. libB is unloaded, removing the symbol, too
/// That's fine, it will trigger a subsequent dlsym to re-create the symbol.
class RTDynamicLibrarySearchGenerator : public DefinitionGenerator {
public:
using SymbolPredicate = std::function<bool(const SymbolStringPtr &)>;
using RTGetterFunc = std::function<ResourceTrackerSP()>;
/// Create a RTDynamicLibrarySearchGenerator that searches for symbols in the
/// given sys::DynamicLibrary.
///
/// If the Allow predicate is given then only symbols matching the predicate
/// will be searched for. If the predicate is not given then all symbols will
/// be searched for.
RTDynamicLibrarySearchGenerator(sys::DynamicLibrary Dylib, char GlobalPrefix,
RTGetterFunc RT,
SymbolPredicate Allow = SymbolPredicate());
/// Permanently loads the library at the given path and, on success, returns
/// a DynamicLibrarySearchGenerator that will search it for symbol definitions
/// in the library. On failure returns the reason the library failed to load.
static Expected<std::unique_ptr<RTDynamicLibrarySearchGenerator>>
Load(const char *FileName, char GlobalPrefix, RTGetterFunc RT,
SymbolPredicate Allow = SymbolPredicate());
/// Creates a RTDynamicLibrarySearchGenerator that searches for symbols in
/// the current process.
static Expected<std::unique_ptr<RTDynamicLibrarySearchGenerator>>
GetForCurrentProcess(char GlobalPrefix, RTGetterFunc RT,
SymbolPredicate Allow = SymbolPredicate()) {
return Load(nullptr, GlobalPrefix, std::move(RT), std::move(Allow));
}
Error tryToGenerate(LookupState &LS, LookupKind K, JITDylib &JD,
JITDylibLookupFlags JDLookupFlags,
const SymbolLookupSet &Symbols) override;
private:
sys::DynamicLibrary Dylib;
RTGetterFunc CurrentRT;
SymbolPredicate Allow;
char GlobalPrefix;
};
RTDynamicLibrarySearchGenerator::RTDynamicLibrarySearchGenerator(
sys::DynamicLibrary Dylib, char GlobalPrefix, RTGetterFunc RT,
SymbolPredicate Allow)
: Dylib(std::move(Dylib)), CurrentRT(std::move(RT)),
Allow(std::move(Allow)), GlobalPrefix(GlobalPrefix) {}
Expected<std::unique_ptr<RTDynamicLibrarySearchGenerator>>
RTDynamicLibrarySearchGenerator::Load(const char *FileName, char GlobalPrefix,
RTGetterFunc RT, SymbolPredicate Allow) {
std::string ErrMsg;
auto Lib = sys::DynamicLibrary::getPermanentLibrary(FileName, &ErrMsg);
if (!Lib.isValid())
return make_error<StringError>(std::move(ErrMsg), inconvertibleErrorCode());
return std::make_unique<RTDynamicLibrarySearchGenerator>(
std::move(Lib), GlobalPrefix, RT, std::move(Allow));
}
Error RTDynamicLibrarySearchGenerator::tryToGenerate(
LookupState &LS, LookupKind K, JITDylib &JD,
JITDylibLookupFlags JDLookupFlags, const SymbolLookupSet &Symbols) {
orc::SymbolMap NewSymbols;
for (auto &KV : Symbols) {
auto &Name = KV.first;
if ((*Name).empty())
continue;
if (Allow && !Allow(Name))
continue;
bool StripGlobalPrefix = (GlobalPrefix != '\0' && (*Name).front() == GlobalPrefix);
std::string Tmp((*Name).data() + StripGlobalPrefix,
(*Name).size() - StripGlobalPrefix);
if (void *Addr = Dylib.getAddressOfSymbol(Tmp.c_str())) {
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)), CurrentRT());
}
/// A definition generator that calls a user-provided function that is
/// responsible for providing symbol addresses.
/// This is used by `IncrementalJIT::getGenerator()` to yield a generator that
/// resolves symbols defined in the IncrementalJIT object on which the function
/// is called, which in turn may be used to provide lookup across different
/// IncrementalJIT instances.
class DelegateGenerator : public DefinitionGenerator {
using LookupFunc = std::function<Expected<llvm::orc::ExecutorAddr>(StringRef)>;
LookupFunc lookup;
public:
DelegateGenerator(LookupFunc lookup) : lookup(lookup) {}
Error tryToGenerate(LookupState& LS, LookupKind K, JITDylib& JD,
JITDylibLookupFlags JDLookupFlags,
const SymbolLookupSet& LookupSet) override {
SymbolMap Symbols;
for (auto& KV : LookupSet) {
auto Addr = lookup(*KV.first);
if (auto Err = Addr.takeError())
return Err;
Symbols[KV.first] = JITEvaluatedSymbol(
Addr->getValue(),
JITSymbolFlags::Exported);
}
if (Symbols.empty())
return Error::success();
return JD.define(absoluteSymbols(std::move(Symbols)));
}
};
static bool UseJITLink(const Triple& TT) {
bool jitLink = false;
// Default to JITLink on macOS and RISC-V, as done in (recent) LLVM by
// LLJITBuilderState::prepareForConstruction.
if (TT.getArch() == Triple::riscv64 ||
(TT.isOSBinFormatMachO() &&
(TT.getArch() == Triple::aarch64 || TT.getArch() == Triple::x86_64)) ||
(TT.isOSBinFormatELF() && TT.getArch() == Triple::ppc64le)) {
jitLink = true;
}
// Finally, honor the user's choice by setting an environment variable.
if (const char* clingJitLink = std::getenv("CLING_JITLINK")) {
jitLink = cling::utils::ConvertEnvValueToBool(clingJitLink);
}
return jitLink;
}
static std::unique_ptr<TargetMachine>
CreateTargetMachine(const clang::CompilerInstance& CI, bool JITLink) {
CodeGenOpt::Level OptLevel = CodeGenOpt::Default;
switch (CI.getCodeGenOpts().OptimizationLevel) {
case 0: OptLevel = CodeGenOpt::None; break;
case 1: OptLevel = CodeGenOpt::Less; break;
case 2: OptLevel = CodeGenOpt::Default; break;
case 3: OptLevel = CodeGenOpt::Aggressive; break;
default: OptLevel = CodeGenOpt::Default;
}
const Triple &TT = CI.getTarget().getTriple();
using namespace llvm::orc;
auto JTMB = JITTargetMachineBuilder(TT);
JTMB.addFeatures(CI.getTargetOpts().Features);
JTMB.getOptions().MCOptions.ABIName = CI.getTarget().getABI().str();
JTMB.setCodeGenOptLevel(OptLevel);
#ifdef _WIN32
JTMB.getOptions().EmulatedTLS = false;
#endif // _WIN32
#if defined(__powerpc64__) || defined(__PPC64__)
// We have to use large code model for PowerPC64 because TOC and text sections
// can be more than 2GB apart.
JTMB.setCodeModel(CodeModel::Large);
#endif
if (JITLink) {
// Set up the TargetMachine as otherwise done by
// LLJITBuilderState::prepareForConstruction.
JTMB.setRelocationModel(Reloc::PIC_);
// Set the small code except for macOS on AArch64 - it results in relocation
// targets that are out-of-range.
// TODO: Investigate / report upstream and re-evaluate after a future LLVM
// upgrade.
if (!(TT.isOSBinFormatMachO() && TT.getArch() == Triple::aarch64))
JTMB.setCodeModel(CodeModel::Small);
}
return cantFail(JTMB.createTargetMachine());
}
#if defined(__linux__) && defined(__GLIBC__)
static SymbolMap GetListOfLibcNonsharedSymbols(const LLJIT& Jit) {
// Inject a number of symbols that may be in libc_nonshared.a where they are
// not found automatically. Before DefinitionGenerators in ORCv2, this used
// to be done by RTDyldMemoryManager::getSymbolAddressInProcess See also the
// upstream issue https://github.com/llvm/llvm-project/issues/61289.
static const std::pair<const char*, const void*> NamePtrList[] = {
{"stat", (void*)&stat}, {"fstat", (void*)&fstat},
{"lstat", (void*)&lstat}, {"stat64", (void*)&stat64},
{"fstat64", (void*)&fstat64}, {"lstat64", (void*)&lstat64},
{"fstatat", (void*)&fstatat}, {"fstatat64", (void*)&fstatat64},
{"mknod", (void*)&mknod}, {"mknodat", (void*)&mknodat},
};
SymbolMap LibcNonsharedSymbols;
for (const auto& NamePtr : NamePtrList) {
auto Addr = static_cast<JITTargetAddress>(
reinterpret_cast<uintptr_t>(NamePtr.second));
LibcNonsharedSymbols[Jit.mangleAndIntern(NamePtr.first)] =
JITEvaluatedSymbol(Addr, JITSymbolFlags::Exported);
}
return LibcNonsharedSymbols;
}
#endif
} // unnamed namespace
namespace cling {
///\brief Creates JIT event listener to allow profiling of JITted code with perf
llvm::JITEventListener* createPerfJITEventListener();
IncrementalJIT::IncrementalJIT(
IncrementalExecutor& Executor, const clang::CompilerInstance &CI,
std::unique_ptr<llvm::orc::ExecutorProcessControl> EPC, Error& Err,
void *ExtraLibHandle, bool Verbose)
: SkipHostProcessLookup(false),
m_JITLink(UseJITLink(CI.getTarget().getTriple())),
m_TM(CreateTargetMachine(CI, m_JITLink)),
SingleThreadedContext(std::make_unique<LLVMContext>()) {
ErrorAsOutParameter _(&Err);
LLJITBuilder Builder;
Builder.setDataLayout(m_TM->createDataLayout());
Builder.setExecutorProcessControl(std::move(EPC));
// Create ObjectLinkingLayer with our own MemoryManager.
Builder.setObjectLinkingLayerCreator([&](ExecutionSession& ES,
const Triple& TT)
-> std::unique_ptr<ObjectLayer> {
if (m_JITLink) {
// For JITLink, we only need a custom memory manager to avoid freeing the
// memory segments; the default InProcessMemoryManager (which is mostly
// copied above) already does slab allocation to keep all segments
// together which is needed for exception handling support.
unsigned PageSize = *sys::Process::getPageSize();
auto ObjLinkingLayer = std::make_unique<ObjectLinkingLayer>(
ES, std::make_unique<ClingJITLinkMemoryManager>(PageSize));
ObjLinkingLayer->addPlugin(std::make_unique<EHFrameRegistrationPlugin>(
ES, std::make_unique<InProcessEHFrameRegistrar>()));
return ObjLinkingLayer;
}
auto GetMemMgr = []() { return std::make_unique<ClingMemoryManager>(); };
auto Layer =
std::make_unique<RTDyldObjectLinkingLayer>(ES, std::move(GetMemMgr));
// Register JIT event listeners if enabled
if (cling::utils::ConvertEnvValueToBool(std::getenv("CLING_DEBUG")))
Layer->registerJITEventListener(
*JITEventListener::createGDBRegistrationListener());
#ifdef __linux__
if (cling::utils::ConvertEnvValueToBool(std::getenv("CLING_PROFILE")))
Layer->registerJITEventListener(*cling::createPerfJITEventListener());
#endif
// The following is based on LLJIT::createObjectLinkingLayer.
if (TT.isOSBinFormatCOFF()) {
Layer->setOverrideObjectFlagsWithResponsibilityFlags(true);
Layer->setAutoClaimResponsibilityForObjectSymbols(true);
}
if (TT.isOSBinFormatELF() && (TT.getArch() == Triple::ArchType::ppc64 ||
TT.getArch() == Triple::ArchType::ppc64le))
Layer->setAutoClaimResponsibilityForObjectSymbols(true);
return Layer;
});
Builder.setCompileFunctionCreator([&](llvm::orc::JITTargetMachineBuilder)
-> llvm::Expected<std::unique_ptr<llvm::orc::IRCompileLayer::IRCompiler>> {
return std::make_unique<SimpleCompiler>(*m_TM);
});
if (Expected<std::unique_ptr<LLJIT>> JitInstance = Builder.create()) {
Jit = std::move(*JitInstance);
} else {
Err = JitInstance.takeError();
return;
}
// We use this callback to transfer the ownership of the ThreadSafeModule,
// which owns the Transaction's llvm::Module, to m_CompiledModules.
Jit->getIRCompileLayer().setNotifyCompiled([this](auto &MR,
ThreadSafeModule TSM) {
// FIXME: Don't store them mapped by raw pointers.
const Module *Unsafe = TSM.getModuleUnlocked();
assert(!m_CompiledModules.count(Unsafe) && "Modules are compiled once");
m_CompiledModules[Unsafe] = std::move(TSM);
});
char LinkerPrefix = this->m_TM->createDataLayout().getGlobalPrefix();
// Process symbol resolution
auto HostProcessLookup
= RTDynamicLibrarySearchGenerator::GetForCurrentProcess(LinkerPrefix,
[&]{ return m_CurrentRT; },
[&](const SymbolStringPtr &Sym) {
return !m_ForbidDlSymbols.contains(*Sym); });
if (!HostProcessLookup) {
Err = HostProcessLookup.takeError();
return;
}
Jit->getMainJITDylib().addGenerator(std::move(*HostProcessLookup));
// This must come after process resolution, to consistently resolve global
// symbols (e.g. std::cout) to the same address.
auto LibLookup = std::make_unique<RTDynamicLibrarySearchGenerator>(
llvm::sys::DynamicLibrary(ExtraLibHandle), LinkerPrefix,
[&]{ return m_CurrentRT; },
[&](const SymbolStringPtr &Sym) {
return !m_ForbidDlSymbols.contains(*Sym); });
Jit->getMainJITDylib().addGenerator(std::move(LibLookup));
#if defined(__linux__) && defined(__GLIBC__)
// See comment in ListOfLibcNonsharedSymbols.
cantFail(Jit->getMainJITDylib().define(
absoluteSymbols(GetListOfLibcNonsharedSymbols(*Jit))));
#endif
// This replaces llvm::orc::ExecutionSession::logErrorsToStdErr:
auto&& ErrorReporter = [&Executor, LinkerPrefix, Verbose](Error Err) {
Err = handleErrors(std::move(Err),
[&](std::unique_ptr<SymbolsNotFound> Err) -> Error {
// IncrementalExecutor has its own diagnostics (for
// now) that tries to guess which library needs to be
// loaded.
for (auto&& symbol : Err->getSymbols()) {
std::string symbolStr = (*symbol).str();
if (LinkerPrefix != '\0' &&
symbolStr[0] == LinkerPrefix) {
symbolStr.erase(0, 1);
}
Executor.HandleMissingFunction(symbolStr);
}
// However, the diagnstic here might be superior as
// they show *all* unresolved symbols, so show them in
// case of "verbose" nonetheless.
if (Verbose)
return Error(std::move(Err));
return Error::success();
});
if (!Err)
return;
logAllUnhandledErrors(std::move(Err), errs(), "cling JIT session error: ");
};
Jit->getExecutionSession().setErrorReporter(ErrorReporter);
}
std::unique_ptr<llvm::orc::DefinitionGenerator> IncrementalJIT::getGenerator() {
return std::make_unique<DelegateGenerator>(
[&](StringRef UnmangledName) { return Jit->lookup(UnmangledName); });
}
void IncrementalJIT::addModule(Transaction& T) {
ResourceTrackerSP RT = Jit->getMainJITDylib().createResourceTracker();
m_ResourceTrackers[&T] = RT;
std::unique_ptr<Module> module = T.takeModule();
// Reset the sections of all functions so that they end up in the same text
// section. This is important for TCling on macOS to catch exceptions raised
// by constructors, which requires unwinding information. The addresses in
// the __eh_frame table are relocated against a single __text section when
// loading the MachO binary, which breaks if the call sites of constructors
// end up in a separate init section.
// (see clang::TargetInfo::getStaticInitSectionSpecifier())
for (auto &Fn : module->functions()) {
if (Fn.hasSection()) {
// dbgs() << "Resetting section '" << Fn.getSection() << "' of function "
// << Fn.getName() << "\n";
Fn.setSection("");
}
}
ThreadSafeModule TSM(std::move(module), SingleThreadedContext);
const Module *Unsafe = TSM.getModuleUnlocked();
T.m_CompiledModule = Unsafe;
m_CurrentRT = RT;
if (Error Err = Jit->addIRModule(RT, std::move(TSM))) {
logAllUnhandledErrors(std::move(Err), errs(),
"[IncrementalJIT] addModule() failed: ");
return;
}
}
llvm::Error IncrementalJIT::removeModule(const Transaction& T) {
ResourceTrackerSP RT = std::move(m_ResourceTrackers[&T]);
if (!RT)
return llvm::Error::success();
m_ResourceTrackers.erase(&T);
if (Error Err = RT->remove())
return Err;
auto iMod = m_CompiledModules.find(T.m_CompiledModule);
if (iMod != m_CompiledModules.end())
m_CompiledModules.erase(iMod);
return llvm::Error::success();
}
JITTargetAddress
IncrementalJIT::addOrReplaceDefinition(StringRef Name,
JITTargetAddress KnownAddr) {
void* Symbol = getSymbolAddress(Name, /*IncludeFromHost=*/true);
// Nothing to define, we are redefining the same function. FIXME: Diagnose.
if (Symbol && (JITTargetAddress)Symbol == KnownAddr)
return KnownAddr;
llvm::SmallString<128> LinkerMangledName;
char LinkerPrefix = this->m_TM->createDataLayout().getGlobalPrefix();
bool HasLinkerPrefix = LinkerPrefix != '\0';
if (HasLinkerPrefix && Name.front() == LinkerPrefix) {
LinkerMangledName.assign(1, LinkerPrefix);
LinkerMangledName.append(Name);
} else {
LinkerMangledName.assign(Name);
}
// Let's inject it
bool Inserted;
SymbolMap::iterator It;
std::tie(It, Inserted) = m_InjectedSymbols.try_emplace(
Jit->getExecutionSession().intern(LinkerMangledName),
JITEvaluatedSymbol(KnownAddr, JITSymbolFlags::Exported));
assert(Inserted && "Why wasn't this found in the initial Jit lookup?");
JITDylib& DyLib = Jit->getMainJITDylib();
// We want to replace a symbol with a custom provided one.
if (Symbol && KnownAddr)
// The symbol be in the DyLib or in-process.
llvm::consumeError(DyLib.remove({It->first}));
if (Error Err = DyLib.define(absoluteSymbols({*It}))) {
logAllUnhandledErrors(std::move(Err), errs(),
"[IncrementalJIT] define() failed: ");
return JITTargetAddress{};
}
return KnownAddr;
}
void* IncrementalJIT::getSymbolAddress(StringRef Name, bool IncludeHostSymbols){
std::unique_lock<SharedAtomicFlag> G(SkipHostProcessLookup, std::defer_lock);
if (!IncludeHostSymbols)
G.lock();
std::pair<llvm::StringMapIterator<std::nullopt_t>, bool> insertInfo;
if (!IncludeHostSymbols)
insertInfo = m_ForbidDlSymbols.insert(Name);
Expected<llvm::orc::ExecutorAddr> Symbol = Jit->lookup(Name);
// If m_ForbidDlSymbols already contained Name before we tried to insert it
// then some calling frame has added it and will remove it later because its
// insertInfo.second is true.
if (!IncludeHostSymbols && insertInfo.second)
m_ForbidDlSymbols.erase(insertInfo.first);
if (!Symbol) {
// This interface is allowed to return nullptr on a missing symbol without
// diagnostics.
consumeError(Symbol.takeError());
return nullptr;
}
return jitTargetAddressToPointer<void*>(Symbol->getValue());
}
bool IncrementalJIT::doesSymbolAlreadyExist(StringRef UnmangledName) {
auto Name = Jit->mangle(UnmangledName);
for (auto &&M: m_CompiledModules) {
if (M.first->getNamedValue(Name))
return true;
}
return false;
}
} // namespace cling