Allow the atexit functions to be called on their own.

In environments where cling's teardown should be controlled more carefully,
the atexit function handlers should allow to be called separately. This is
useful when a handler depends on alive interpreter and cannot be executed while
the interpreter is half shut down.

This patch should enable TCling's proper shutdown and relax some of the pain
in shutting down ROOT in general.
This commit is contained in:
Vassil Vassilev 2019-12-13 11:25:46 +02:00 committed by SFT
parent ad6a1b47cd
commit bd0b61528c
4 changed files with 24 additions and 7 deletions

View File

@ -804,6 +804,12 @@ namespace cling {
///
void AddAtExitFunc(void (*Func) (void*), void* Arg);
///\brief Run once the list of registered atexit functions. This is useful
/// when an external process wants to control carefully the teardown because
/// the registered atexit functions require alive interpreter service.
///
void runAtExitFuncs();
void GenerateAutoloadingMap(llvm::StringRef inFile, llvm::StringRef outFile,
bool enableMacros = false, bool enableLogs = true);

View File

@ -109,7 +109,7 @@ IncrementalExecutor::IncrementalExecutor(clang::DiagnosticsEngine& diags,
// Keep in source: ~unique_ptr<ClingJIT> needs ClingJIT
IncrementalExecutor::~IncrementalExecutor() {}
void IncrementalExecutor::shuttingDown() {
void IncrementalExecutor::runAtExitFuncs() {
// It is legal to register an atexit handler from within another atexit
// handler and furthor-more the standard says they need to run in reverse
// order, so this function must be recursion safe.
@ -126,7 +126,7 @@ void IncrementalExecutor::shuttingDown() {
AtExit();
// The standard says that they need to run in reverse order, which means
// anything added from 'AtExit()' must now be run!
shuttingDown();
runAtExitFuncs();
}
}

View File

@ -208,13 +208,19 @@ namespace cling {
m_JIT->addModule(module);
}
///\brief Tells the execution context that we are shutting down the system.
///\brief Tells the execution to run all registered atexit functions once.
///
/// This that notification is needed because the execution context needs to
/// perform extra actions like delete all managed by it symbols, which might
/// still require alive system.
/// This rountine should be used with caution only when an external process
/// wants to carefully control the teardown. For example, if the process
/// has registered its own atexit functions which need the interpreter
/// service to be available when they are being executed.
///
void shuttingDown();
void runAtExitFuncs();
///\brief A more meaningful synonym of runAtExitFuncs when used in a more
/// standard teardown.
///
void shuttingDown() { runAtExitFuncs(); }
///\brief Gets the address of an existing global and whether it was JITted.
///

View File

@ -1718,6 +1718,11 @@ namespace cling {
m_Executor->AddAtExitFunc(Func, Arg, getLatestTransaction()->getModule());
}
void Interpreter::runAtExitFuncs() {
assert(!isInSyntaxOnlyMode() && "Must have JIT");
m_Executor->runAtExitFuncs();
}
void Interpreter::GenerateAutoloadingMap(llvm::StringRef inFile,
llvm::StringRef outFile,
bool enableMacros,