Skip to content
Snippets Groups Projects
Commit 23989e77 authored by Axel Naumann's avatar Axel Naumann
Browse files

Keep trying to remap __cxa_atexit until succeeded.

__cxa_atexit is generated by clang::CodeGen upon seeing the first static
destructor. Instead of trying to provoke that we just continue to search for it.
parent c5c3070b
No related branches found
No related tags found
No related merge requests found
...@@ -76,39 +76,16 @@ void IncrementalExecutor::shuttingDown() { ...@@ -76,39 +76,16 @@ void IncrementalExecutor::shuttingDown() {
} }
void IncrementalExecutor::remapCXAAtExit() { void IncrementalExecutor::remapCXAAtExit() {
assert(!m_CxaAtExitRemapped && "__cxa_at_exit already remapped."); if (m_CxaAtExitRemapped)
llvm::Function* clingAtExit return;
= m_engine->FindFunctionNamed("cling_cxa_atexit");
assert(clingAtExit && "cling_cxa_atexit must exist.");
llvm::Function* atExit = m_engine->FindFunctionNamed("__cxa_atexit"); llvm::Function* atExit = m_engine->FindFunctionNamed("__cxa_atexit");
if (!atExit) { if (!atExit)
// Inject __cxa_atexit into module return;
llvm::Type* retTy = 0;
llvm::Type* voidPtrTy = 0;
if (sizeof(int) == 4) {
retTy = llvm::Type::getInt32Ty(llvm::getGlobalContext());
voidPtrTy = llvm::Type::getInt32PtrTy(llvm::getGlobalContext());
} else if (sizeof(int) == 8) {
retTy = llvm::Type::getInt64Ty(llvm::getGlobalContext());
voidPtrTy = llvm::Type::getInt64PtrTy(llvm::getGlobalContext());
} else {
assert(retTy && "Unsupported sizeof(int)!");
retTy = llvm::Type::getInt64Ty(llvm::getGlobalContext());
voidPtrTy = llvm::Type::getInt64PtrTy(llvm::getGlobalContext());
}
llvm::SmallVector<llvm::Type*, 3> argTy; llvm::Function* clingAtExit
argTy.push_back(voidPtrTy); = m_engine->FindFunctionNamed("cling_cxa_atexit");
argTy.push_back(voidPtrTy); assert(clingAtExit && "cling_cxa_atexit must exist.");
argTy.push_back(voidPtrTy);
llvm::FunctionType* cxaatexitTy
= llvm::FunctionType::get(retTy, argTy, false /*varArg*/);
llvm::Function* atexitFunc
= llvm::Function::Create(cxaatexitTy, llvm::GlobalValue::InternalLinkage,
"__cxa_atexit", 0 /*module*/);
m_engine->addGlobalMapping(atexitFunc, clingAtExit);
}
void* clingAtExitAddr = m_engine->getPointerToFunction(clingAtExit); void* clingAtExitAddr = m_engine->getPointerToFunction(clingAtExit);
assert(clingAtExitAddr && "cannot find cling_cxa_atexit"); assert(clingAtExitAddr && "cannot find cling_cxa_atexit");
...@@ -194,6 +171,8 @@ IncrementalExecutor::executeFunction(llvm::StringRef funcname, ...@@ -194,6 +171,8 @@ IncrementalExecutor::executeFunction(llvm::StringRef funcname,
// We don't care whether something was unresolved before. // We don't care whether something was unresolved before.
m_unresolvedSymbols.clear(); m_unresolvedSymbols.clear();
remapCXAAtExit();
llvm::Function* f = m_engine->FindFunctionNamed(funcname.str().c_str()); llvm::Function* f = m_engine->FindFunctionNamed(funcname.str().c_str());
if (!f) { if (!f) {
llvm::errs() << "IncrementalExecutor::executeFunction: " llvm::errs() << "IncrementalExecutor::executeFunction: "
...@@ -276,6 +255,7 @@ IncrementalExecutor::runStaticInitializersOnce(llvm::Module* m) { ...@@ -276,6 +255,7 @@ IncrementalExecutor::runStaticInitializersOnce(llvm::Module* m) {
// Execute the ctor/dtor function! // Execute the ctor/dtor function!
if (llvm::Function *F = llvm::dyn_cast<llvm::Function>(FP)) { if (llvm::Function *F = llvm::dyn_cast<llvm::Function>(FP)) {
remapCXAAtExit();
m_engine->getPointerToFunction(F); m_engine->getPointerToFunction(F);
// check if there is any unresolved symbol in the list // check if there is any unresolved symbol in the list
if (!m_unresolvedSymbols.empty()) { if (!m_unresolvedSymbols.empty()) {
...@@ -340,7 +320,7 @@ IncrementalExecutor::addSymbol(const char* symbolName, void* symbolAddress) { ...@@ -340,7 +320,7 @@ IncrementalExecutor::addSymbol(const char* symbolName, void* symbolAddress) {
void* IncrementalExecutor::getAddressOfGlobal(llvm::Module* m, void* IncrementalExecutor::getAddressOfGlobal(llvm::Module* m,
llvm::StringRef symbolName, llvm::StringRef symbolName,
bool* fromJIT /*=0*/) const { bool* fromJIT /*=0*/) {
// Return a symbol's address, and whether it was jitted. // Return a symbol's address, and whether it was jitted.
void* address void* address
= llvm::sys::DynamicLibrary::SearchForAddressOfSymbol(symbolName); = llvm::sys::DynamicLibrary::SearchForAddressOfSymbol(symbolName);
...@@ -352,13 +332,15 @@ void* IncrementalExecutor::getAddressOfGlobal(llvm::Module* m, ...@@ -352,13 +332,15 @@ void* IncrementalExecutor::getAddressOfGlobal(llvm::Module* m,
if (!gvar) if (!gvar)
return 0; return 0;
remapCXAAtExit();
address = m_engine->getPointerToGlobal(gvar); address = m_engine->getPointerToGlobal(gvar);
} }
return address; return address;
} }
void* void*
IncrementalExecutor::getPointerToGlobalFromJIT(const llvm::GlobalValue& GV)const{ IncrementalExecutor::getPointerToGlobalFromJIT(const llvm::GlobalValue& GV) {
remapCXAAtExit();
if (void* addr = m_engine->getPointerToGlobalIfAvailable(&GV)) if (void* addr = m_engine->getPointerToGlobalIfAvailable(&GV))
return addr; return addr;
......
...@@ -141,11 +141,6 @@ namespace cling { ...@@ -141,11 +141,6 @@ namespace cling {
/// ///
void shuttingDown(); void shuttingDown();
///\brief Remaps the __cxa_at_exit with a interpreter-controlled one, so
/// that the interpreter can call the object destructors at the right time.
///
void remapCXAAtExit();
///\brief Gets the address of an existing global and whether it was JITted. ///\brief Gets the address of an existing global and whether it was JITted.
/// ///
/// JIT symbols might not be immediately convertible to e.g. a function /// JIT symbols might not be immediately convertible to e.g. a function
...@@ -156,14 +151,14 @@ namespace cling { ...@@ -156,14 +151,14 @@ namespace cling {
///\param[out] fromJIT - whether the symbol was JITted. ///\param[out] fromJIT - whether the symbol was JITted.
/// ///
void* getAddressOfGlobal(llvm::Module* m, llvm::StringRef mangledName, void* getAddressOfGlobal(llvm::Module* m, llvm::StringRef mangledName,
bool* fromJIT = 0) const; bool* fromJIT = 0);
///\brief Return the address of a global from the ExecutionEngine (as ///\brief Return the address of a global from the ExecutionEngine (as
/// opposed to dynamic libraries). Forces the emission of the symbol if /// opposed to dynamic libraries). Forces the emission of the symbol if
/// it has not happened yet. /// it has not happened yet.
/// ///
///param[in] GV - global value for which the address will be returned. ///param[in] GV - global value for which the address will be returned.
void* getPointerToGlobalFromJIT(const llvm::GlobalValue& GV) const; void* getPointerToGlobalFromJIT(const llvm::GlobalValue& GV);
llvm::ExecutionEngine* getExecutionEngine() const { llvm::ExecutionEngine* getExecutionEngine() const {
if (!m_engine) if (!m_engine)
...@@ -177,6 +172,11 @@ namespace cling { ...@@ -177,6 +172,11 @@ namespace cling {
const cling::Transaction* clingT); const cling::Transaction* clingT);
private: private:
///\brief Remaps the __cxa_at_exit with a interpreter-controlled one, so
/// that the interpreter can call the object destructors at the right time.
///
void remapCXAAtExit();
static void* HandleMissingFunction(const std::string&); static void* HandleMissingFunction(const std::string&);
static void* NotifyLazyFunctionCreators(const std::string&); static void* NotifyLazyFunctionCreators(const std::string&);
......
...@@ -103,7 +103,7 @@ namespace cling { ...@@ -103,7 +103,7 @@ namespace cling {
const char* llvmdir); const char* llvmdir);
~IncrementalParser(); ~IncrementalParser();
void Initialize(); void Initialize();
clang::CompilerInstance* getCI() const { return m_CI.get(); } clang::CompilerInstance* getCI() const { return m_CI.get(); }
clang::Parser* getParser() const { return m_Parser.get(); } clang::Parser* getParser() const { return m_Parser.get(); }
clang::CodeGenerator* getCodeGenerator() const { return m_CodeGen.get(); } clang::CodeGenerator* getCodeGenerator() const { return m_CodeGen.get(); }
......
...@@ -233,7 +233,6 @@ namespace cling { ...@@ -233,7 +233,6 @@ namespace cling {
"cling::Interpreter *gCling=(cling::Interpreter*)" "cling::Interpreter *gCling=(cling::Interpreter*)"
<< (uintptr_t)this << ";} }"; << (uintptr_t)this << ";} }";
declare(initializer.str()); declare(initializer.str());
m_Executor->remapCXAAtExit();
} }
declare("#include \"cling/Interpreter/ValuePrinter.h\""); declare("#include \"cling/Interpreter/ValuePrinter.h\"");
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment