2018-06-27 05:35:48 +08:00
|
|
|
//===--------- LLJIT.cpp - An ORC-based JIT for compiling LLVM IR ---------===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2018-06-27 05:35:48 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "llvm/ExecutionEngine/Orc/LLJIT.h"
|
2019-12-16 11:51:35 +08:00
|
|
|
#include "llvm/ExecutionEngine/JITLink/JITLinkMemoryManager.h"
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
#include "llvm/ExecutionEngine/Orc/MachOPlatform.h"
|
2019-12-16 11:51:35 +08:00
|
|
|
#include "llvm/ExecutionEngine/Orc/ObjectLinkingLayer.h"
|
2018-06-27 05:35:48 +08:00
|
|
|
#include "llvm/ExecutionEngine/Orc/OrcError.h"
|
2019-04-30 06:37:27 +08:00
|
|
|
#include "llvm/ExecutionEngine/Orc/RTDyldObjectLinkingLayer.h"
|
2018-06-27 05:35:48 +08:00
|
|
|
#include "llvm/ExecutionEngine/SectionMemoryManager.h"
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
#include "llvm/IR/GlobalVariable.h"
|
|
|
|
#include "llvm/IR/IRBuilder.h"
|
2018-06-27 05:35:48 +08:00
|
|
|
#include "llvm/IR/Mangler.h"
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
#include "llvm/IR/Module.h"
|
|
|
|
#include "llvm/Support/DynamicLibrary.h"
|
|
|
|
|
|
|
|
#include <map>
|
|
|
|
|
2020-03-05 04:36:50 +08:00
|
|
|
#define DEBUG_TYPE "orc"
|
|
|
|
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
using namespace llvm;
|
|
|
|
using namespace llvm::orc;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
/// Add a reference to the __dso_handle global to the given module.
|
|
|
|
/// Returns a reference to the __dso_handle IR decl.
|
|
|
|
GlobalVariable *addDSOHandleDecl(Module &M) {
|
|
|
|
auto DSOHandleTy = StructType::create(M.getContext(), "lljit.dso_handle");
|
|
|
|
return new GlobalVariable(M, DSOHandleTy, true, GlobalValue::ExternalLinkage,
|
|
|
|
nullptr, "__dso_handle");
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Adds helper function decls and wrapper functions that call the helper with
|
|
|
|
/// some additional prefix arguments.
|
|
|
|
///
|
|
|
|
/// E.g. For wrapper "foo" with type i8(i8, i64), helper "bar", and prefix
|
|
|
|
/// args i32 4 and i16 12345, this function will add:
|
|
|
|
///
|
|
|
|
/// declare i8 @bar(i32, i16, i8, i64)
|
|
|
|
///
|
|
|
|
/// define i8 @foo(i8, i64) {
|
|
|
|
/// entry:
|
|
|
|
/// %2 = call i8 @bar(i32 4, i16 12345, i8 %0, i64 %1)
|
|
|
|
/// ret i8 %2
|
|
|
|
/// }
|
|
|
|
///
|
|
|
|
Function *addHelperAndWrapper(Module &M, StringRef WrapperName,
|
|
|
|
FunctionType *WrapperFnType,
|
|
|
|
GlobalValue::VisibilityTypes WrapperVisibility,
|
|
|
|
StringRef HelperName,
|
|
|
|
ArrayRef<Value *> HelperPrefixArgs) {
|
|
|
|
std::vector<Type *> HelperArgTypes;
|
|
|
|
for (auto *Arg : HelperPrefixArgs)
|
|
|
|
HelperArgTypes.push_back(Arg->getType());
|
|
|
|
for (auto *T : WrapperFnType->params())
|
|
|
|
HelperArgTypes.push_back(T);
|
|
|
|
auto *HelperFnType =
|
|
|
|
FunctionType::get(WrapperFnType->getReturnType(), HelperArgTypes, false);
|
|
|
|
auto *HelperFn = Function::Create(HelperFnType, GlobalValue::ExternalLinkage,
|
|
|
|
HelperName, M);
|
|
|
|
|
|
|
|
auto *WrapperFn = Function::Create(
|
|
|
|
WrapperFnType, GlobalValue::ExternalLinkage, WrapperName, M);
|
|
|
|
WrapperFn->setVisibility(WrapperVisibility);
|
|
|
|
|
|
|
|
auto *EntryBlock = BasicBlock::Create(M.getContext(), "entry", WrapperFn);
|
|
|
|
IRBuilder<> IB(EntryBlock);
|
|
|
|
|
|
|
|
std::vector<Value *> HelperArgs;
|
|
|
|
for (auto *Arg : HelperPrefixArgs)
|
|
|
|
HelperArgs.push_back(Arg);
|
|
|
|
for (auto &Arg : WrapperFn->args())
|
|
|
|
HelperArgs.push_back(&Arg);
|
|
|
|
auto *HelperResult = IB.CreateCall(HelperFn, HelperArgs);
|
|
|
|
if (HelperFn->getReturnType()->isVoidTy())
|
|
|
|
IB.CreateRetVoid();
|
|
|
|
else
|
|
|
|
IB.CreateRet(HelperResult);
|
|
|
|
|
|
|
|
return WrapperFn;
|
|
|
|
}
|
|
|
|
|
|
|
|
class GenericLLVMIRPlatformSupport;
|
|
|
|
|
|
|
|
/// orc::Platform component of Generic LLVM IR Platform support.
|
|
|
|
/// Just forwards calls to the GenericLLVMIRPlatformSupport class below.
|
|
|
|
class GenericLLVMIRPlatform : public Platform {
|
|
|
|
public:
|
|
|
|
GenericLLVMIRPlatform(GenericLLVMIRPlatformSupport &S) : S(S) {}
|
|
|
|
Error setupJITDylib(JITDylib &JD) override;
|
|
|
|
Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU) override;
|
|
|
|
Error notifyRemoving(JITDylib &JD, VModuleKey K) override {
|
|
|
|
// Noop -- Nothing to do (yet).
|
|
|
|
return Error::success();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
GenericLLVMIRPlatformSupport &S;
|
|
|
|
};
|
|
|
|
|
|
|
|
/// This transform parses llvm.global_ctors to produce a single initialization
|
|
|
|
/// function for the module, records the function, then deletes
|
|
|
|
/// llvm.global_ctors.
|
|
|
|
class GlobalCtorDtorScraper {
|
|
|
|
public:
|
2020-02-23 01:49:55 +08:00
|
|
|
|
|
|
|
GlobalCtorDtorScraper(GenericLLVMIRPlatformSupport &PS,
|
|
|
|
StringRef InitFunctionPrefix)
|
|
|
|
: PS(PS), InitFunctionPrefix(InitFunctionPrefix) {}
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
Expected<ThreadSafeModule> operator()(ThreadSafeModule TSM,
|
|
|
|
MaterializationResponsibility &R);
|
|
|
|
|
|
|
|
private:
|
|
|
|
GenericLLVMIRPlatformSupport &PS;
|
2020-02-23 01:49:55 +08:00
|
|
|
StringRef InitFunctionPrefix;
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/// Generic IR Platform Support
|
|
|
|
///
|
|
|
|
/// Scrapes llvm.global_ctors and llvm.global_dtors and replaces them with
|
|
|
|
/// specially named 'init' and 'deinit'. Injects definitions / interposes for
|
|
|
|
/// some runtime API, including __cxa_atexit, dlopen, and dlclose.
|
|
|
|
class GenericLLVMIRPlatformSupport : public LLJIT::PlatformSupport {
|
|
|
|
public:
|
|
|
|
// GenericLLVMIRPlatform &P) : P(P) {
|
|
|
|
GenericLLVMIRPlatformSupport(LLJIT &J) : J(J) {
|
|
|
|
|
2020-02-23 01:49:55 +08:00
|
|
|
MangleAndInterner Mangle(getExecutionSession(), J.getDataLayout());
|
|
|
|
InitFunctionPrefix = Mangle("__orc_init_func.");
|
|
|
|
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
getExecutionSession().setPlatform(
|
|
|
|
std::make_unique<GenericLLVMIRPlatform>(*this));
|
|
|
|
|
2020-02-23 01:49:55 +08:00
|
|
|
setInitTransform(J, GlobalCtorDtorScraper(*this, *InitFunctionPrefix));
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
|
|
|
|
SymbolMap StdInterposes;
|
|
|
|
|
|
|
|
StdInterposes[Mangle("__lljit.platform_support_instance")] =
|
|
|
|
JITEvaluatedSymbol(pointerToJITTargetAddress(this), JITSymbolFlags());
|
|
|
|
StdInterposes[Mangle("__lljit.cxa_atexit_helper")] = JITEvaluatedSymbol(
|
|
|
|
pointerToJITTargetAddress(registerAtExitHelper), JITSymbolFlags());
|
|
|
|
StdInterposes[Mangle("__lljit.run_atexits_helper")] = JITEvaluatedSymbol(
|
|
|
|
pointerToJITTargetAddress(runAtExitsHelper), JITSymbolFlags());
|
|
|
|
|
|
|
|
cantFail(
|
|
|
|
J.getMainJITDylib().define(absoluteSymbols(std::move(StdInterposes))));
|
|
|
|
cantFail(setupJITDylib(J.getMainJITDylib()));
|
|
|
|
cantFail(J.addIRModule(J.getMainJITDylib(), createPlatformRuntimeModule()));
|
|
|
|
}
|
|
|
|
|
|
|
|
ExecutionSession &getExecutionSession() { return J.getExecutionSession(); }
|
|
|
|
|
|
|
|
/// Adds a module that defines the __dso_handle global.
|
|
|
|
Error setupJITDylib(JITDylib &JD) {
|
|
|
|
auto Ctx = std::make_unique<LLVMContext>();
|
|
|
|
auto M = std::make_unique<Module>("__standard_lib", *Ctx);
|
|
|
|
M->setDataLayout(J.getDataLayout());
|
|
|
|
|
|
|
|
auto *Int64Ty = Type::getInt64Ty(*Ctx);
|
|
|
|
auto *DSOHandle = new GlobalVariable(
|
|
|
|
*M, Int64Ty, true, GlobalValue::ExternalLinkage,
|
|
|
|
ConstantInt::get(Int64Ty, reinterpret_cast<uintptr_t>(&JD)),
|
|
|
|
"__dso_handle");
|
|
|
|
DSOHandle->setVisibility(GlobalValue::HiddenVisibility);
|
|
|
|
DSOHandle->setInitializer(
|
|
|
|
ConstantInt::get(Int64Ty, pointerToJITTargetAddress(&JD)));
|
|
|
|
return J.addIRModule(JD, ThreadSafeModule(std::move(M), std::move(Ctx)));
|
|
|
|
}
|
|
|
|
|
|
|
|
Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU) {
|
|
|
|
std::lock_guard<std::mutex> Lock(PlatformSupportMutex);
|
|
|
|
if (auto &InitSym = MU.getInitializerSymbol())
|
|
|
|
InitSymbols[&JD].add(InitSym);
|
2020-02-23 01:49:55 +08:00
|
|
|
else {
|
|
|
|
// If there's no identified init symbol attached, but there is a symbol
|
|
|
|
// with the GenericIRPlatform::InitFunctionPrefix, then treat that as
|
|
|
|
// an init function. Add the symbol to both the InitSymbols map (which
|
|
|
|
// will trigger a lookup to materialize the module) and the InitFunctions
|
|
|
|
// map (which holds the names of the symbols to execute).
|
|
|
|
for (auto &KV : MU.getSymbols())
|
|
|
|
if ((*KV.first).startswith(*InitFunctionPrefix)) {
|
|
|
|
InitSymbols[&JD].add(KV.first);
|
|
|
|
InitFunctions[&JD].add(KV.first);
|
|
|
|
}
|
|
|
|
}
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
return Error::success();
|
|
|
|
}
|
|
|
|
|
|
|
|
Error initialize(JITDylib &JD) override {
|
2020-03-05 04:36:50 +08:00
|
|
|
LLVM_DEBUG({
|
|
|
|
dbgs() << "GenericLLVMIRPlatformSupport getting initializers to run\n";
|
|
|
|
});
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
if (auto Initializers = getInitializers(JD)) {
|
2020-03-05 04:36:50 +08:00
|
|
|
LLVM_DEBUG(
|
|
|
|
{ dbgs() << "GenericLLVMIRPlatformSupport running initializers\n"; });
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
for (auto InitFnAddr : *Initializers) {
|
2020-03-05 04:36:50 +08:00
|
|
|
LLVM_DEBUG({
|
|
|
|
dbgs() << " Running init " << formatv("{0:x16}", InitFnAddr)
|
|
|
|
<< "...\n";
|
|
|
|
});
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
auto *InitFn = jitTargetAddressToFunction<void (*)()>(InitFnAddr);
|
|
|
|
InitFn();
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
return Initializers.takeError();
|
|
|
|
return Error::success();
|
|
|
|
}
|
|
|
|
|
|
|
|
Error deinitialize(JITDylib &JD) override {
|
2020-03-05 04:36:50 +08:00
|
|
|
LLVM_DEBUG({
|
|
|
|
dbgs() << "GenericLLVMIRPlatformSupport getting deinitializers to run\n";
|
|
|
|
});
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
if (auto Deinitializers = getDeinitializers(JD)) {
|
2020-03-05 04:36:50 +08:00
|
|
|
LLVM_DEBUG({
|
|
|
|
dbgs() << "GenericLLVMIRPlatformSupport running deinitializers\n";
|
|
|
|
});
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
for (auto DeinitFnAddr : *Deinitializers) {
|
2020-03-05 04:36:50 +08:00
|
|
|
LLVM_DEBUG({
|
|
|
|
dbgs() << " Running init " << formatv("{0:x16}", DeinitFnAddr)
|
|
|
|
<< "...\n";
|
|
|
|
});
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
auto *DeinitFn = jitTargetAddressToFunction<void (*)()>(DeinitFnAddr);
|
|
|
|
DeinitFn();
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
return Deinitializers.takeError();
|
|
|
|
|
|
|
|
return Error::success();
|
|
|
|
}
|
|
|
|
|
|
|
|
void registerInitFunc(JITDylib &JD, SymbolStringPtr InitName) {
|
|
|
|
std::lock_guard<std::mutex> Lock(PlatformSupportMutex);
|
|
|
|
InitFunctions[&JD].add(InitName);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Expected<std::vector<JITTargetAddress>> getInitializers(JITDylib &JD) {
|
|
|
|
if (auto Err = issueInitLookups(JD))
|
|
|
|
return std::move(Err);
|
|
|
|
|
|
|
|
DenseMap<JITDylib *, SymbolLookupSet> LookupSymbols;
|
|
|
|
std::vector<JITDylib *> DFSLinkOrder;
|
|
|
|
|
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> Lock(PlatformSupportMutex);
|
|
|
|
DFSLinkOrder = getDFSLinkOrder(JD);
|
|
|
|
|
|
|
|
for (auto *NextJD : DFSLinkOrder) {
|
|
|
|
auto IFItr = InitFunctions.find(NextJD);
|
|
|
|
if (IFItr != InitFunctions.end()) {
|
|
|
|
LookupSymbols[NextJD] = std::move(IFItr->second);
|
|
|
|
InitFunctions.erase(IFItr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-05 04:36:50 +08:00
|
|
|
LLVM_DEBUG({
|
|
|
|
dbgs() << "JITDylib init order is [ ";
|
|
|
|
for (auto *JD : llvm::reverse(DFSLinkOrder))
|
|
|
|
dbgs() << "\"" << JD->getName() << "\" ";
|
|
|
|
dbgs() << "]\n";
|
|
|
|
dbgs() << "Looking up init functions:\n";
|
|
|
|
for (auto &KV : LookupSymbols)
|
|
|
|
dbgs() << " \"" << KV.first->getName() << "\": " << KV.second << "\n";
|
|
|
|
});
|
|
|
|
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
auto &ES = getExecutionSession();
|
|
|
|
auto LookupResult = Platform::lookupInitSymbols(ES, LookupSymbols);
|
|
|
|
|
|
|
|
if (!LookupResult)
|
|
|
|
return LookupResult.takeError();
|
|
|
|
|
|
|
|
std::vector<JITTargetAddress> Initializers;
|
|
|
|
while (!DFSLinkOrder.empty()) {
|
|
|
|
auto &NextJD = *DFSLinkOrder.back();
|
|
|
|
DFSLinkOrder.pop_back();
|
|
|
|
auto InitsItr = LookupResult->find(&NextJD);
|
|
|
|
if (InitsItr == LookupResult->end())
|
|
|
|
continue;
|
|
|
|
for (auto &KV : InitsItr->second)
|
|
|
|
Initializers.push_back(KV.second.getAddress());
|
|
|
|
}
|
|
|
|
|
|
|
|
return Initializers;
|
|
|
|
}
|
|
|
|
|
|
|
|
Expected<std::vector<JITTargetAddress>> getDeinitializers(JITDylib &JD) {
|
|
|
|
auto &ES = getExecutionSession();
|
|
|
|
|
|
|
|
MangleAndInterner Mangle(getExecutionSession(), J.getDataLayout());
|
|
|
|
auto LLJITRunAtExits = Mangle("__lljit_run_atexits");
|
|
|
|
|
|
|
|
DenseMap<JITDylib *, SymbolLookupSet> LookupSymbols;
|
|
|
|
std::vector<JITDylib *> DFSLinkOrder;
|
|
|
|
|
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> Lock(PlatformSupportMutex);
|
|
|
|
DFSLinkOrder = getDFSLinkOrder(JD);
|
|
|
|
|
|
|
|
for (auto *NextJD : DFSLinkOrder) {
|
|
|
|
auto &JDLookupSymbols = LookupSymbols[NextJD];
|
|
|
|
auto DIFItr = DeInitFunctions.find(NextJD);
|
|
|
|
if (DIFItr != DeInitFunctions.end()) {
|
|
|
|
LookupSymbols[NextJD] = std::move(DIFItr->second);
|
|
|
|
DeInitFunctions.erase(DIFItr);
|
|
|
|
}
|
|
|
|
JDLookupSymbols.add(LLJITRunAtExits,
|
|
|
|
SymbolLookupFlags::WeaklyReferencedSymbol);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
auto LookupResult = Platform::lookupInitSymbols(ES, LookupSymbols);
|
|
|
|
|
|
|
|
if (!LookupResult)
|
|
|
|
return LookupResult.takeError();
|
|
|
|
|
|
|
|
std::vector<JITTargetAddress> DeInitializers;
|
|
|
|
for (auto *NextJD : DFSLinkOrder) {
|
|
|
|
auto DeInitsItr = LookupResult->find(NextJD);
|
|
|
|
assert(DeInitsItr != LookupResult->end() &&
|
|
|
|
"Every JD should have at least __lljit_run_atexits");
|
|
|
|
|
|
|
|
auto RunAtExitsItr = DeInitsItr->second.find(LLJITRunAtExits);
|
|
|
|
if (RunAtExitsItr != DeInitsItr->second.end())
|
|
|
|
DeInitializers.push_back(RunAtExitsItr->second.getAddress());
|
|
|
|
|
|
|
|
for (auto &KV : DeInitsItr->second)
|
|
|
|
if (KV.first != LLJITRunAtExits)
|
|
|
|
DeInitializers.push_back(KV.second.getAddress());
|
|
|
|
}
|
|
|
|
|
|
|
|
return DeInitializers;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns a DFS traversal order of the JITDylibs reachable (via
|
|
|
|
// links-against edges) from JD, starting with JD itself.
|
|
|
|
static std::vector<JITDylib *> getDFSLinkOrder(JITDylib &JD) {
|
|
|
|
std::vector<JITDylib *> DFSLinkOrder;
|
|
|
|
std::vector<JITDylib *> WorkStack({&JD});
|
|
|
|
DenseSet<JITDylib *> Visited;
|
|
|
|
|
|
|
|
while (!WorkStack.empty()) {
|
|
|
|
auto &NextJD = *WorkStack.back();
|
|
|
|
WorkStack.pop_back();
|
|
|
|
if (Visited.count(&NextJD))
|
|
|
|
continue;
|
|
|
|
Visited.insert(&NextJD);
|
|
|
|
DFSLinkOrder.push_back(&NextJD);
|
|
|
|
NextJD.withSearchOrderDo([&](const JITDylibSearchOrder &SearchOrder) {
|
|
|
|
for (auto &KV : SearchOrder)
|
|
|
|
WorkStack.push_back(KV.first);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return DFSLinkOrder;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Issue lookups for all init symbols required to initialize JD (and any
|
|
|
|
/// JITDylibs that it depends on).
|
|
|
|
Error issueInitLookups(JITDylib &JD) {
|
|
|
|
DenseMap<JITDylib *, SymbolLookupSet> RequiredInitSymbols;
|
|
|
|
|
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> Lock(PlatformSupportMutex);
|
|
|
|
|
|
|
|
auto DFSLinkOrder = getDFSLinkOrder(JD);
|
|
|
|
|
|
|
|
for (auto *NextJD : DFSLinkOrder) {
|
|
|
|
auto ISItr = InitSymbols.find(NextJD);
|
|
|
|
if (ISItr != InitSymbols.end()) {
|
|
|
|
RequiredInitSymbols[NextJD] = std::move(ISItr->second);
|
|
|
|
InitSymbols.erase(ISItr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Platform::lookupInitSymbols(getExecutionSession(),
|
|
|
|
RequiredInitSymbols)
|
|
|
|
.takeError();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void registerAtExitHelper(void *Self, void (*F)(void *), void *Ctx,
|
|
|
|
void *DSOHandle) {
|
|
|
|
static_cast<GenericLLVMIRPlatformSupport *>(Self)->AtExitMgr.registerAtExit(
|
|
|
|
F, Ctx, DSOHandle);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void runAtExitsHelper(void *Self, void *DSOHandle) {
|
|
|
|
static_cast<GenericLLVMIRPlatformSupport *>(Self)->AtExitMgr.runAtExits(
|
|
|
|
DSOHandle);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Constructs an LLVM IR module containing platform runtime globals,
|
|
|
|
// functions, and interposes.
|
|
|
|
ThreadSafeModule createPlatformRuntimeModule() {
|
|
|
|
auto Ctx = std::make_unique<LLVMContext>();
|
|
|
|
auto M = std::make_unique<Module>("__standard_lib", *Ctx);
|
|
|
|
M->setDataLayout(J.getDataLayout());
|
|
|
|
|
|
|
|
auto *GenericIRPlatformSupportTy =
|
|
|
|
StructType::create(*Ctx, "lljit.GenericLLJITIRPlatformSupport");
|
|
|
|
|
|
|
|
auto *PlatformInstanceDecl = new GlobalVariable(
|
|
|
|
*M, GenericIRPlatformSupportTy, true, GlobalValue::ExternalLinkage,
|
|
|
|
nullptr, "__lljit.platform_support_instance");
|
|
|
|
|
|
|
|
auto *DSOHandleDecl = addDSOHandleDecl(*M);
|
|
|
|
|
|
|
|
auto *Int8Ty = Type::getInt8Ty(*Ctx);
|
|
|
|
auto *IntTy = Type::getIntNTy(*Ctx, sizeof(int) * CHAR_BIT);
|
|
|
|
auto *VoidTy = Type::getVoidTy(*Ctx);
|
|
|
|
auto *BytePtrTy = PointerType::getUnqual(Int8Ty);
|
|
|
|
auto *AtExitCallbackTy = FunctionType::get(VoidTy, {BytePtrTy}, false);
|
|
|
|
auto *AtExitCallbackPtrTy = PointerType::getUnqual(AtExitCallbackTy);
|
|
|
|
|
|
|
|
addHelperAndWrapper(
|
|
|
|
*M, "__cxa_atexit",
|
|
|
|
FunctionType::get(IntTy, {AtExitCallbackPtrTy, BytePtrTy, BytePtrTy},
|
|
|
|
false),
|
|
|
|
GlobalValue::HiddenVisibility, "__lljit.cxa_atexit_helper",
|
|
|
|
{PlatformInstanceDecl});
|
|
|
|
|
|
|
|
addHelperAndWrapper(
|
|
|
|
*M, "__lljit_run_atexits", FunctionType::get(VoidTy, {}, false),
|
|
|
|
GlobalValue::HiddenVisibility, "__lljit.run_atexits_helper",
|
|
|
|
{PlatformInstanceDecl, DSOHandleDecl});
|
|
|
|
|
|
|
|
return ThreadSafeModule(std::move(M), std::move(Ctx));
|
|
|
|
}
|
|
|
|
|
|
|
|
std::mutex PlatformSupportMutex;
|
|
|
|
LLJIT &J;
|
2020-02-23 01:49:55 +08:00
|
|
|
SymbolStringPtr InitFunctionPrefix;
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
DenseMap<JITDylib *, SymbolLookupSet> InitSymbols;
|
|
|
|
DenseMap<JITDylib *, SymbolLookupSet> InitFunctions;
|
|
|
|
DenseMap<JITDylib *, SymbolLookupSet> DeInitFunctions;
|
|
|
|
ItaniumCXAAtExitSupport AtExitMgr;
|
|
|
|
};
|
|
|
|
|
|
|
|
Error GenericLLVMIRPlatform::setupJITDylib(JITDylib &JD) {
|
|
|
|
return S.setupJITDylib(JD);
|
|
|
|
}
|
|
|
|
|
|
|
|
Error GenericLLVMIRPlatform::notifyAdding(JITDylib &JD,
|
|
|
|
const MaterializationUnit &MU) {
|
|
|
|
return S.notifyAdding(JD, MU);
|
|
|
|
}
|
|
|
|
|
|
|
|
Expected<ThreadSafeModule>
|
|
|
|
GlobalCtorDtorScraper::operator()(ThreadSafeModule TSM,
|
|
|
|
MaterializationResponsibility &R) {
|
|
|
|
auto Err = TSM.withModuleDo([&](Module &M) -> Error {
|
|
|
|
auto &Ctx = M.getContext();
|
|
|
|
auto *GlobalCtors = M.getNamedGlobal("llvm.global_ctors");
|
|
|
|
|
|
|
|
// If there's no llvm.global_ctors or it's just a decl then skip.
|
|
|
|
if (!GlobalCtors || GlobalCtors->isDeclaration())
|
|
|
|
return Error::success();
|
|
|
|
|
|
|
|
std::string InitFunctionName;
|
|
|
|
raw_string_ostream(InitFunctionName)
|
2020-02-23 01:49:55 +08:00
|
|
|
<< InitFunctionPrefix << M.getModuleIdentifier();
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
|
|
|
|
MangleAndInterner Mangle(PS.getExecutionSession(), M.getDataLayout());
|
|
|
|
auto InternedName = Mangle(InitFunctionName);
|
|
|
|
if (auto Err =
|
|
|
|
R.defineMaterializing({{InternedName, JITSymbolFlags::Callable}}))
|
|
|
|
return Err;
|
|
|
|
|
|
|
|
auto *InitFunc =
|
|
|
|
Function::Create(FunctionType::get(Type::getVoidTy(Ctx), {}, false),
|
|
|
|
GlobalValue::ExternalLinkage, InitFunctionName, &M);
|
|
|
|
InitFunc->setVisibility(GlobalValue::HiddenVisibility);
|
|
|
|
std::vector<std::pair<Function *, unsigned>> Inits;
|
|
|
|
for (auto E : getConstructors(M))
|
|
|
|
Inits.push_back(std::make_pair(E.Func, E.Priority));
|
|
|
|
llvm::sort(Inits, [](const std::pair<Function *, unsigned> &LHS,
|
|
|
|
const std::pair<Function *, unsigned> &RHS) {
|
|
|
|
return LHS.first < RHS.first;
|
|
|
|
});
|
|
|
|
auto *EntryBlock = BasicBlock::Create(Ctx, "entry", InitFunc);
|
|
|
|
IRBuilder<> IB(EntryBlock);
|
|
|
|
for (auto &KV : Inits)
|
|
|
|
IB.CreateCall(KV.first);
|
|
|
|
IB.CreateRetVoid();
|
|
|
|
|
|
|
|
PS.registerInitFunc(R.getTargetJITDylib(), InternedName);
|
|
|
|
GlobalCtors->eraseFromParent();
|
|
|
|
return Error::success();
|
|
|
|
});
|
|
|
|
|
|
|
|
if (Err)
|
|
|
|
return std::move(Err);
|
|
|
|
|
2020-02-20 06:27:31 +08:00
|
|
|
return std::move(TSM);
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
class MachOPlatformSupport : public LLJIT::PlatformSupport {
|
|
|
|
public:
|
|
|
|
using DLOpenType = void *(*)(const char *Name, int Mode);
|
|
|
|
using DLCloseType = int (*)(void *Handle);
|
|
|
|
using DLSymType = void *(*)(void *Handle, const char *Name);
|
|
|
|
using DLErrorType = const char *(*)();
|
|
|
|
|
|
|
|
struct DlFcnValues {
|
|
|
|
Optional<void *> RTLDDefault;
|
|
|
|
DLOpenType dlopen = nullptr;
|
|
|
|
DLCloseType dlclose = nullptr;
|
|
|
|
DLSymType dlsym = nullptr;
|
|
|
|
DLErrorType dlerror = nullptr;
|
|
|
|
};
|
|
|
|
|
|
|
|
static Expected<std::unique_ptr<MachOPlatformSupport>>
|
|
|
|
Create(LLJIT &J, JITDylib &PlatformJITDylib) {
|
|
|
|
|
|
|
|
// Make process symbols visible.
|
|
|
|
{
|
|
|
|
std::string ErrMsg;
|
|
|
|
auto Lib = sys::DynamicLibrary::getPermanentLibrary(nullptr, &ErrMsg);
|
|
|
|
if (!Lib.isValid())
|
|
|
|
return make_error<StringError>(std::move(ErrMsg),
|
|
|
|
inconvertibleErrorCode());
|
|
|
|
}
|
|
|
|
|
|
|
|
DlFcnValues DlFcn;
|
|
|
|
|
|
|
|
// Add support for RTLDDefault on known platforms.
|
|
|
|
#ifdef __APPLE__
|
|
|
|
DlFcn.RTLDDefault = reinterpret_cast<void *>(-2);
|
|
|
|
#endif // __APPLE__
|
|
|
|
|
|
|
|
if (auto Err = hookUpFunction(DlFcn.dlopen, "dlopen"))
|
|
|
|
return std::move(Err);
|
|
|
|
if (auto Err = hookUpFunction(DlFcn.dlclose, "dlclose"))
|
|
|
|
return std::move(Err);
|
|
|
|
if (auto Err = hookUpFunction(DlFcn.dlsym, "dlsym"))
|
|
|
|
return std::move(Err);
|
|
|
|
if (auto Err = hookUpFunction(DlFcn.dlerror, "dlerror"))
|
|
|
|
return std::move(Err);
|
|
|
|
|
|
|
|
std::unique_ptr<MachOPlatformSupport> MP(
|
|
|
|
new MachOPlatformSupport(J, PlatformJITDylib, DlFcn));
|
|
|
|
return std::move(MP);
|
|
|
|
}
|
|
|
|
|
|
|
|
Error initialize(JITDylib &JD) override {
|
2020-03-05 04:36:50 +08:00
|
|
|
LLVM_DEBUG({
|
|
|
|
dbgs() << "MachOPlatformSupport initializing \"" << JD.getName()
|
|
|
|
<< "\"\n";
|
|
|
|
});
|
|
|
|
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
if (auto InitSeq = MP.getInitializerSequence(JD)) {
|
|
|
|
for (auto &KV : *InitSeq) {
|
|
|
|
KV.second.registerObjCSelectors();
|
|
|
|
if (auto Err = KV.second.registerObjCClasses()) {
|
|
|
|
// FIXME: Roll back registrations on error?
|
|
|
|
return Err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (auto &KV : *InitSeq)
|
|
|
|
KV.second.runModInits();
|
|
|
|
} else
|
|
|
|
return InitSeq.takeError();
|
|
|
|
return Error::success();
|
|
|
|
}
|
|
|
|
|
|
|
|
Error deinitialize(JITDylib &JD) override {
|
|
|
|
auto &ES = J.getExecutionSession();
|
|
|
|
if (auto DeinitSeq = MP.getDeinitializerSequence(JD)) {
|
|
|
|
for (auto &KV : *DeinitSeq) {
|
|
|
|
auto DSOHandleName = ES.intern("___dso_handle");
|
|
|
|
|
|
|
|
// FIXME: Run DeInits here.
|
|
|
|
auto Result = ES.lookup(
|
|
|
|
{{KV.first, JITDylibLookupFlags::MatchAllSymbols}},
|
|
|
|
SymbolLookupSet(DSOHandleName,
|
|
|
|
SymbolLookupFlags::WeaklyReferencedSymbol));
|
|
|
|
if (!Result)
|
|
|
|
return Result.takeError();
|
|
|
|
if (Result->empty())
|
|
|
|
continue;
|
|
|
|
assert(Result->count(DSOHandleName) &&
|
|
|
|
"Result does not contain __dso_handle");
|
|
|
|
auto *DSOHandle = jitTargetAddressToPointer<void *>(
|
|
|
|
Result->begin()->second.getAddress());
|
|
|
|
AtExitMgr.runAtExits(DSOHandle);
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
return DeinitSeq.takeError();
|
|
|
|
return Error::success();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
template <typename FunctionPtrTy>
|
|
|
|
static Error hookUpFunction(FunctionPtrTy &Fn, const char *Name) {
|
|
|
|
if (auto *FnAddr = sys::DynamicLibrary::SearchForAddressOfSymbol(Name)) {
|
|
|
|
Fn = reinterpret_cast<FunctionPtrTy>(Fn);
|
|
|
|
return Error::success();
|
|
|
|
}
|
|
|
|
|
|
|
|
return make_error<StringError>((Twine("Can not enable MachO JIT Platform: "
|
|
|
|
"missing function: ") +
|
|
|
|
Name)
|
|
|
|
.str(),
|
|
|
|
inconvertibleErrorCode());
|
|
|
|
}
|
|
|
|
|
|
|
|
MachOPlatformSupport(LLJIT &J, JITDylib &PlatformJITDylib, DlFcnValues DlFcn)
|
|
|
|
: J(J), MP(setupPlatform(J)), DlFcn(std::move(DlFcn)) {
|
|
|
|
|
|
|
|
MangleAndInterner Mangle(J.getExecutionSession(), J.getDataLayout());
|
|
|
|
SymbolMap HelperSymbols;
|
|
|
|
|
|
|
|
// platform and atexit helpers.
|
|
|
|
HelperSymbols[Mangle("__lljit.platform_support_instance")] =
|
|
|
|
JITEvaluatedSymbol(pointerToJITTargetAddress(this), JITSymbolFlags());
|
|
|
|
HelperSymbols[Mangle("__lljit.cxa_atexit_helper")] = JITEvaluatedSymbol(
|
|
|
|
pointerToJITTargetAddress(registerAtExitHelper), JITSymbolFlags());
|
|
|
|
HelperSymbols[Mangle("__lljit.run_atexits_helper")] = JITEvaluatedSymbol(
|
|
|
|
pointerToJITTargetAddress(runAtExitsHelper), JITSymbolFlags());
|
|
|
|
|
|
|
|
// dlfcn helpers.
|
|
|
|
HelperSymbols[Mangle("__lljit.dlopen_helper")] = JITEvaluatedSymbol(
|
|
|
|
pointerToJITTargetAddress(dlopenHelper), JITSymbolFlags());
|
|
|
|
HelperSymbols[Mangle("__lljit.dlclose_helper")] = JITEvaluatedSymbol(
|
|
|
|
pointerToJITTargetAddress(dlcloseHelper), JITSymbolFlags());
|
|
|
|
HelperSymbols[Mangle("__lljit.dlsym_helper")] = JITEvaluatedSymbol(
|
|
|
|
pointerToJITTargetAddress(dlsymHelper), JITSymbolFlags());
|
|
|
|
HelperSymbols[Mangle("__lljit.dlerror_helper")] = JITEvaluatedSymbol(
|
|
|
|
pointerToJITTargetAddress(dlerrorHelper), JITSymbolFlags());
|
|
|
|
|
|
|
|
cantFail(
|
|
|
|
PlatformJITDylib.define(absoluteSymbols(std::move(HelperSymbols))));
|
|
|
|
cantFail(MP.setupJITDylib(J.getMainJITDylib()));
|
|
|
|
cantFail(J.addIRModule(PlatformJITDylib, createPlatformRuntimeModule()));
|
|
|
|
}
|
|
|
|
|
|
|
|
static MachOPlatform &setupPlatform(LLJIT &J) {
|
|
|
|
auto Tmp = std::make_unique<MachOPlatform>(
|
|
|
|
J.getExecutionSession(),
|
|
|
|
static_cast<ObjectLinkingLayer &>(J.getObjLinkingLayer()),
|
|
|
|
createStandardSymbolsObject(J));
|
|
|
|
auto &MP = *Tmp;
|
|
|
|
J.getExecutionSession().setPlatform(std::move(Tmp));
|
|
|
|
return MP;
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::unique_ptr<MemoryBuffer> createStandardSymbolsObject(LLJIT &J) {
|
|
|
|
LLVMContext Ctx;
|
|
|
|
Module M("__standard_symbols", Ctx);
|
|
|
|
M.setDataLayout(J.getDataLayout());
|
|
|
|
|
|
|
|
auto *Int64Ty = Type::getInt64Ty(Ctx);
|
|
|
|
|
|
|
|
auto *DSOHandle =
|
|
|
|
new GlobalVariable(M, Int64Ty, true, GlobalValue::ExternalLinkage,
|
|
|
|
ConstantInt::get(Int64Ty, 0), "__dso_handle");
|
|
|
|
DSOHandle->setVisibility(GlobalValue::HiddenVisibility);
|
|
|
|
|
|
|
|
return cantFail(J.getIRCompileLayer().getCompiler()(M));
|
|
|
|
}
|
|
|
|
|
|
|
|
ThreadSafeModule createPlatformRuntimeModule() {
|
|
|
|
auto Ctx = std::make_unique<LLVMContext>();
|
|
|
|
auto M = std::make_unique<Module>("__standard_lib", *Ctx);
|
|
|
|
M->setDataLayout(J.getDataLayout());
|
|
|
|
|
|
|
|
auto *MachOPlatformSupportTy =
|
|
|
|
StructType::create(*Ctx, "lljit.MachOPlatformSupport");
|
|
|
|
|
|
|
|
auto *PlatformInstanceDecl = new GlobalVariable(
|
|
|
|
*M, MachOPlatformSupportTy, true, GlobalValue::ExternalLinkage, nullptr,
|
|
|
|
"__lljit.platform_support_instance");
|
|
|
|
|
|
|
|
auto *Int8Ty = Type::getInt8Ty(*Ctx);
|
|
|
|
auto *IntTy = Type::getIntNTy(*Ctx, sizeof(int) * CHAR_BIT);
|
|
|
|
auto *VoidTy = Type::getVoidTy(*Ctx);
|
|
|
|
auto *BytePtrTy = PointerType::getUnqual(Int8Ty);
|
|
|
|
auto *AtExitCallbackTy = FunctionType::get(VoidTy, {BytePtrTy}, false);
|
|
|
|
auto *AtExitCallbackPtrTy = PointerType::getUnqual(AtExitCallbackTy);
|
|
|
|
|
|
|
|
addHelperAndWrapper(
|
|
|
|
*M, "__cxa_atexit",
|
|
|
|
FunctionType::get(IntTy, {AtExitCallbackPtrTy, BytePtrTy, BytePtrTy},
|
|
|
|
false),
|
|
|
|
GlobalValue::DefaultVisibility, "__lljit.cxa_atexit_helper",
|
|
|
|
{PlatformInstanceDecl});
|
|
|
|
|
|
|
|
addHelperAndWrapper(*M, "dlopen",
|
|
|
|
FunctionType::get(BytePtrTy, {BytePtrTy, IntTy}, false),
|
|
|
|
GlobalValue::DefaultVisibility, "__lljit.dlopen_helper",
|
|
|
|
{PlatformInstanceDecl});
|
|
|
|
|
|
|
|
addHelperAndWrapper(*M, "dlclose",
|
|
|
|
FunctionType::get(IntTy, {BytePtrTy}, false),
|
|
|
|
GlobalValue::DefaultVisibility,
|
|
|
|
"__lljit.dlclose_helper", {PlatformInstanceDecl});
|
|
|
|
|
|
|
|
addHelperAndWrapper(
|
|
|
|
*M, "dlsym",
|
|
|
|
FunctionType::get(BytePtrTy, {BytePtrTy, BytePtrTy}, false),
|
|
|
|
GlobalValue::DefaultVisibility, "__lljit.dlsym_helper",
|
|
|
|
{PlatformInstanceDecl});
|
|
|
|
|
|
|
|
addHelperAndWrapper(*M, "dlerror", FunctionType::get(BytePtrTy, {}, false),
|
|
|
|
GlobalValue::DefaultVisibility,
|
|
|
|
"__lljit.dlerror_helper", {PlatformInstanceDecl});
|
|
|
|
|
|
|
|
return ThreadSafeModule(std::move(M), std::move(Ctx));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void registerAtExitHelper(void *Self, void (*F)(void *), void *Ctx,
|
|
|
|
void *DSOHandle) {
|
|
|
|
static_cast<MachOPlatformSupport *>(Self)->AtExitMgr.registerAtExit(
|
|
|
|
F, Ctx, DSOHandle);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void runAtExitsHelper(void *Self, void *DSOHandle) {
|
|
|
|
static_cast<MachOPlatformSupport *>(Self)->AtExitMgr.runAtExits(DSOHandle);
|
|
|
|
}
|
|
|
|
|
|
|
|
void *jit_dlopen(const char *Path, int Mode) {
|
|
|
|
JITDylib *JDToOpen = nullptr;
|
|
|
|
// FIXME: Do the right thing with Mode flags.
|
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> Lock(PlatformSupportMutex);
|
|
|
|
|
|
|
|
// Clear any existing error messages.
|
|
|
|
dlErrorMsgs.erase(std::this_thread::get_id());
|
|
|
|
|
|
|
|
if (auto *JD = J.getExecutionSession().getJITDylibByName(Path)) {
|
|
|
|
auto I = JDRefCounts.find(JD);
|
|
|
|
if (I != JDRefCounts.end()) {
|
|
|
|
++I->second;
|
|
|
|
return JD;
|
|
|
|
}
|
|
|
|
|
|
|
|
JDRefCounts[JD] = 1;
|
|
|
|
JDToOpen = JD;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (JDToOpen) {
|
|
|
|
if (auto Err = initialize(*JDToOpen)) {
|
|
|
|
recordError(std::move(Err));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fall through to dlopen if no JITDylib found for Path.
|
|
|
|
return DlFcn.dlopen(Path, Mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *dlopenHelper(void *Self, const char *Path, int Mode) {
|
|
|
|
return static_cast<MachOPlatformSupport *>(Self)->jit_dlopen(Path, Mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
int jit_dlclose(void *Handle) {
|
|
|
|
JITDylib *JDToClose = nullptr;
|
|
|
|
|
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> Lock(PlatformSupportMutex);
|
|
|
|
|
|
|
|
// Clear any existing error messages.
|
|
|
|
dlErrorMsgs.erase(std::this_thread::get_id());
|
|
|
|
|
|
|
|
auto I = JDRefCounts.find(Handle);
|
|
|
|
if (I != JDRefCounts.end()) {
|
|
|
|
--I->second;
|
|
|
|
if (I->second == 0) {
|
|
|
|
JDRefCounts.erase(I);
|
|
|
|
JDToClose = static_cast<JITDylib *>(Handle);
|
|
|
|
} else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (JDToClose) {
|
|
|
|
if (auto Err = deinitialize(*JDToClose)) {
|
|
|
|
recordError(std::move(Err));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fall through to dlclose if no JITDylib found for Path.
|
|
|
|
return DlFcn.dlclose(Handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int dlcloseHelper(void *Self, void *Handle) {
|
|
|
|
return static_cast<MachOPlatformSupport *>(Self)->jit_dlclose(Handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
void *jit_dlsym(void *Handle, const char *Name) {
|
|
|
|
JITDylibSearchOrder JITSymSearchOrder;
|
|
|
|
|
|
|
|
// FIXME: RTLD_NEXT, RTLD_SELF not supported.
|
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> Lock(PlatformSupportMutex);
|
|
|
|
|
|
|
|
// Clear any existing error messages.
|
|
|
|
dlErrorMsgs.erase(std::this_thread::get_id());
|
|
|
|
|
|
|
|
if (JDRefCounts.count(Handle)) {
|
|
|
|
JITSymSearchOrder.push_back(
|
|
|
|
{static_cast<JITDylib *>(Handle),
|
|
|
|
JITDylibLookupFlags::MatchExportedSymbolsOnly});
|
|
|
|
} else if (Handle == DlFcn.RTLDDefault) {
|
|
|
|
for (auto &KV : JDRefCounts)
|
|
|
|
JITSymSearchOrder.push_back(
|
|
|
|
{static_cast<JITDylib *>(KV.first),
|
|
|
|
JITDylibLookupFlags::MatchExportedSymbolsOnly});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!JITSymSearchOrder.empty()) {
|
|
|
|
MangleAndInterner Mangle(J.getExecutionSession(), J.getDataLayout());
|
|
|
|
auto MangledName = Mangle(Name);
|
|
|
|
SymbolLookupSet Syms(MangledName,
|
|
|
|
SymbolLookupFlags::WeaklyReferencedSymbol);
|
|
|
|
if (auto Result = J.getExecutionSession().lookup(JITSymSearchOrder, Syms,
|
|
|
|
LookupKind::DLSym)) {
|
|
|
|
auto I = Result->find(MangledName);
|
|
|
|
if (I != Result->end())
|
|
|
|
return jitTargetAddressToPointer<void *>(I->second.getAddress());
|
|
|
|
} else {
|
|
|
|
recordError(Result.takeError());
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fall through to dlsym.
|
|
|
|
return DlFcn.dlsym(Handle, Name);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *dlsymHelper(void *Self, void *Handle, const char *Name) {
|
|
|
|
return static_cast<MachOPlatformSupport *>(Self)->jit_dlsym(Handle, Name);
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *jit_dlerror() {
|
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> Lock(PlatformSupportMutex);
|
|
|
|
auto I = dlErrorMsgs.find(std::this_thread::get_id());
|
|
|
|
if (I != dlErrorMsgs.end())
|
|
|
|
return I->second->c_str();
|
|
|
|
}
|
|
|
|
return DlFcn.dlerror();
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char *dlerrorHelper(void *Self) {
|
|
|
|
return static_cast<MachOPlatformSupport *>(Self)->jit_dlerror();
|
|
|
|
}
|
|
|
|
|
|
|
|
void recordError(Error Err) {
|
|
|
|
std::lock_guard<std::mutex> Lock(PlatformSupportMutex);
|
|
|
|
dlErrorMsgs[std::this_thread::get_id()] =
|
|
|
|
std::make_unique<std::string>(toString(std::move(Err)));
|
|
|
|
}
|
|
|
|
|
|
|
|
std::mutex PlatformSupportMutex;
|
|
|
|
LLJIT &J;
|
|
|
|
MachOPlatform &MP;
|
|
|
|
DlFcnValues DlFcn;
|
|
|
|
ItaniumCXAAtExitSupport AtExitMgr;
|
|
|
|
DenseMap<void *, unsigned> JDRefCounts;
|
|
|
|
std::map<std::thread::id, std::unique_ptr<std::string>> dlErrorMsgs;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // end anonymous namespace
|
2018-06-27 05:35:48 +08:00
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
namespace orc {
|
|
|
|
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
void LLJIT::PlatformSupport::setInitTransform(
|
|
|
|
LLJIT &J, IRTransformLayer::TransformFunction T) {
|
|
|
|
J.InitHelperTransformLayer->setTransform(std::move(T));
|
|
|
|
}
|
|
|
|
|
|
|
|
LLJIT::PlatformSupport::~PlatformSupport() {}
|
|
|
|
|
2019-04-30 06:37:27 +08:00
|
|
|
Error LLJITBuilderState::prepareForConstruction() {
|
2018-09-27 00:26:59 +08:00
|
|
|
|
2019-04-30 06:37:27 +08:00
|
|
|
if (!JTMB) {
|
|
|
|
if (auto JTMBOrErr = JITTargetMachineBuilder::detectHost())
|
|
|
|
JTMB = std::move(*JTMBOrErr);
|
|
|
|
else
|
|
|
|
return JTMBOrErr.takeError();
|
2019-12-20 12:42:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// If the client didn't configure any linker options then auto-configure the
|
|
|
|
// JIT linker.
|
|
|
|
if (!CreateObjectLinkingLayer && JTMB->getCodeModel() == None &&
|
|
|
|
JTMB->getRelocationModel() == None) {
|
2019-12-16 11:51:35 +08:00
|
|
|
|
|
|
|
auto &TT = JTMB->getTargetTriple();
|
2019-12-20 12:42:26 +08:00
|
|
|
if (TT.isOSBinFormatMachO() &&
|
2019-12-16 11:51:35 +08:00
|
|
|
(TT.getArch() == Triple::aarch64 || TT.getArch() == Triple::x86_64)) {
|
|
|
|
|
|
|
|
JTMB->setRelocationModel(Reloc::PIC_);
|
|
|
|
JTMB->setCodeModel(CodeModel::Small);
|
|
|
|
CreateObjectLinkingLayer =
|
|
|
|
[](ExecutionSession &ES,
|
|
|
|
const Triple &) -> std::unique_ptr<ObjectLayer> {
|
|
|
|
return std::make_unique<ObjectLinkingLayer>(
|
|
|
|
ES, std::make_unique<jitlink::InProcessMemoryManager>());
|
|
|
|
};
|
|
|
|
}
|
2018-09-27 00:26:59 +08:00
|
|
|
}
|
|
|
|
|
2019-04-30 06:37:27 +08:00
|
|
|
return Error::success();
|
|
|
|
}
|
|
|
|
|
|
|
|
LLJIT::~LLJIT() {
|
|
|
|
if (CompileThreads)
|
|
|
|
CompileThreads->wait();
|
2018-06-27 05:35:48 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Error LLJIT::defineAbsolute(StringRef Name, JITEvaluatedSymbol Sym) {
|
2018-10-01 07:18:24 +08:00
|
|
|
auto InternedName = ES->intern(Name);
|
2018-06-27 05:35:48 +08:00
|
|
|
SymbolMap Symbols({{InternedName, Sym}});
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
return Main->define(absoluteSymbols(std::move(Symbols)));
|
2018-06-27 05:35:48 +08:00
|
|
|
}
|
|
|
|
|
2018-09-26 09:24:12 +08:00
|
|
|
Error LLJIT::addIRModule(JITDylib &JD, ThreadSafeModule TSM) {
|
|
|
|
assert(TSM && "Can not add null module");
|
2018-06-27 05:35:48 +08:00
|
|
|
|
2019-08-02 23:21:37 +08:00
|
|
|
if (auto Err =
|
|
|
|
TSM.withModuleDo([&](Module &M) { return applyDataLayout(M); }))
|
2018-06-27 05:35:48 +08:00
|
|
|
return Err;
|
|
|
|
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
return InitHelperTransformLayer->add(JD, std::move(TSM),
|
|
|
|
ES->allocateVModule());
|
2018-06-27 05:35:48 +08:00
|
|
|
}
|
|
|
|
|
2018-08-29 04:20:31 +08:00
|
|
|
Error LLJIT::addObjectFile(JITDylib &JD, std::unique_ptr<MemoryBuffer> Obj) {
|
|
|
|
assert(Obj && "Can not add null object");
|
|
|
|
|
2019-11-15 07:58:21 +08:00
|
|
|
return ObjTransformLayer.add(JD, std::move(Obj), ES->allocateVModule());
|
2018-08-29 04:20:31 +08:00
|
|
|
}
|
|
|
|
|
2018-08-18 05:18:18 +08:00
|
|
|
Expected<JITEvaluatedSymbol> LLJIT::lookupLinkerMangled(JITDylib &JD,
|
2018-06-27 05:35:48 +08:00
|
|
|
StringRef Name) {
|
2019-11-26 13:57:27 +08:00
|
|
|
return ES->lookup(
|
|
|
|
makeJITDylibSearchOrder(&JD, JITDylibLookupFlags::MatchAllSymbols),
|
|
|
|
ES->intern(Name));
|
2018-06-27 05:35:48 +08:00
|
|
|
}
|
|
|
|
|
2019-04-30 06:37:27 +08:00
|
|
|
std::unique_ptr<ObjectLayer>
|
|
|
|
LLJIT::createObjectLinkingLayer(LLJITBuilderState &S, ExecutionSession &ES) {
|
|
|
|
|
|
|
|
// If the config state provided an ObjectLinkingLayer factory then use it.
|
|
|
|
if (S.CreateObjectLinkingLayer)
|
2019-08-03 03:43:20 +08:00
|
|
|
return S.CreateObjectLinkingLayer(ES, S.JTMB->getTargetTriple());
|
2019-04-30 06:37:27 +08:00
|
|
|
|
|
|
|
// Otherwise default to creating an RTDyldObjectLinkingLayer that constructs
|
|
|
|
// a new SectionMemoryManager for each object.
|
2019-08-15 23:54:37 +08:00
|
|
|
auto GetMemMgr = []() { return std::make_unique<SectionMemoryManager>(); };
|
2019-08-03 03:43:20 +08:00
|
|
|
auto ObjLinkingLayer =
|
2019-08-15 23:54:37 +08:00
|
|
|
std::make_unique<RTDyldObjectLinkingLayer>(ES, std::move(GetMemMgr));
|
2019-08-03 03:43:20 +08:00
|
|
|
|
2020-01-18 06:48:48 +08:00
|
|
|
if (S.JTMB->getTargetTriple().isOSBinFormatCOFF()) {
|
2019-08-03 03:43:20 +08:00
|
|
|
ObjLinkingLayer->setOverrideObjectFlagsWithResponsibilityFlags(true);
|
2020-01-18 06:48:48 +08:00
|
|
|
ObjLinkingLayer->setAutoClaimResponsibilityForObjectSymbols(true);
|
|
|
|
}
|
2019-08-03 03:43:20 +08:00
|
|
|
|
2019-08-06 04:30:35 +08:00
|
|
|
// FIXME: Explicit conversion to std::unique_ptr<ObjectLayer> added to silence
|
|
|
|
// errors from some GCC / libstdc++ bots. Remove this conversion (i.e.
|
|
|
|
// just return ObjLinkingLayer) once those bots are upgraded.
|
|
|
|
return std::unique_ptr<ObjectLayer>(std::move(ObjLinkingLayer));
|
2019-04-30 06:37:27 +08:00
|
|
|
}
|
|
|
|
|
2020-01-22 08:28:30 +08:00
|
|
|
Expected<std::unique_ptr<IRCompileLayer::IRCompiler>>
|
2019-07-11 01:24:24 +08:00
|
|
|
LLJIT::createCompileFunction(LLJITBuilderState &S,
|
|
|
|
JITTargetMachineBuilder JTMB) {
|
|
|
|
|
|
|
|
/// If there is a custom compile function creator set then use it.
|
|
|
|
if (S.CreateCompileFunction)
|
|
|
|
return S.CreateCompileFunction(std::move(JTMB));
|
|
|
|
|
|
|
|
// Otherwise default to creating a SimpleCompiler, or ConcurrentIRCompiler,
|
|
|
|
// depending on the number of threads requested.
|
|
|
|
if (S.NumCompileThreads > 0)
|
2020-01-22 08:28:30 +08:00
|
|
|
return std::make_unique<ConcurrentIRCompiler>(std::move(JTMB));
|
2019-07-11 01:24:24 +08:00
|
|
|
|
|
|
|
auto TM = JTMB.createTargetMachine();
|
|
|
|
if (!TM)
|
|
|
|
return TM.takeError();
|
|
|
|
|
2020-01-22 08:28:30 +08:00
|
|
|
return std::make_unique<TMOwningSimpleCompiler>(std::move(*TM));
|
2019-07-11 01:24:24 +08:00
|
|
|
}
|
|
|
|
|
2019-04-30 06:37:27 +08:00
|
|
|
LLJIT::LLJIT(LLJITBuilderState &S, Error &Err)
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
: ES(S.ES ? std::move(S.ES) : std::make_unique<ExecutionSession>()), Main(),
|
|
|
|
DL(""), TT(S.JTMB->getTargetTriple()),
|
2019-11-15 07:58:21 +08:00
|
|
|
ObjLinkingLayer(createObjectLinkingLayer(S, *ES)),
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
ObjTransformLayer(*this->ES, *ObjLinkingLayer) {
|
2019-04-30 06:37:27 +08:00
|
|
|
|
|
|
|
ErrorAsOutParameter _(&Err);
|
|
|
|
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
if (auto MainOrErr = this->ES->createJITDylib("main"))
|
|
|
|
Main = &*MainOrErr;
|
|
|
|
else {
|
|
|
|
Err = MainOrErr.takeError();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (S.DL)
|
|
|
|
DL = std::move(*S.DL);
|
|
|
|
else if (auto DLOrErr = S.JTMB->getDefaultDataLayoutForTarget())
|
2019-07-11 01:24:24 +08:00
|
|
|
DL = std::move(*DLOrErr);
|
|
|
|
else {
|
|
|
|
Err = DLOrErr.takeError();
|
|
|
|
return;
|
|
|
|
}
|
2019-04-30 06:37:27 +08:00
|
|
|
|
2019-07-11 01:24:24 +08:00
|
|
|
{
|
|
|
|
auto CompileFunction = createCompileFunction(S, std::move(*S.JTMB));
|
|
|
|
if (!CompileFunction) {
|
|
|
|
Err = CompileFunction.takeError();
|
2019-04-30 06:37:27 +08:00
|
|
|
return;
|
|
|
|
}
|
2019-08-15 23:54:37 +08:00
|
|
|
CompileLayer = std::make_unique<IRCompileLayer>(
|
2019-11-15 07:58:21 +08:00
|
|
|
*ES, ObjTransformLayer, std::move(*CompileFunction));
|
2020-01-16 02:20:10 +08:00
|
|
|
TransformLayer = std::make_unique<IRTransformLayer>(*ES, *CompileLayer);
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
InitHelperTransformLayer =
|
|
|
|
std::make_unique<IRTransformLayer>(*ES, *TransformLayer);
|
2019-07-11 01:24:24 +08:00
|
|
|
}
|
2019-04-30 06:37:27 +08:00
|
|
|
|
2019-07-11 01:24:24 +08:00
|
|
|
if (S.NumCompileThreads > 0) {
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
InitHelperTransformLayer->setCloneToNewContextOnEmit(true);
|
[Support] On Windows, ensure hardware_concurrency() extends to all CPU sockets and all NUMA groups
The goal of this patch is to maximize CPU utilization on multi-socket or high core count systems, so that parallel computations such as LLD/ThinLTO can use all hardware threads in the system. Before this patch, on Windows, a maximum of 64 hardware threads could be used at most, in some cases dispatched only on one CPU socket.
== Background ==
Windows doesn't have a flat cpu_set_t like Linux. Instead, it projects hardware CPUs (or NUMA nodes) to applications through a concept of "processor groups". A "processor" is the smallest unit of execution on a CPU, that is, an hyper-thread if SMT is active; a core otherwise. There's a limit of 32-bit processors on older 32-bit versions of Windows, which later was raised to 64-processors with 64-bit versions of Windows. This limit comes from the affinity mask, which historically is represented by the sizeof(void*). Consequently, the concept of "processor groups" was introduced for dealing with systems with more than 64 hyper-threads.
By default, the Windows OS assigns only one "processor group" to each starting application, in a round-robin manner. If the application wants to use more processors, it needs to programmatically enable it, by assigning threads to other "processor groups". This also means that affinity cannot cross "processor group" boundaries; one can only specify a "preferred" group on start-up, but the application is free to allocate more groups if it wants to.
This creates a peculiar situation, where newer CPUs like the AMD EPYC 7702P (64-cores, 128-hyperthreads) are projected by the OS as two (2) "processor groups". This means that by default, an application can only use half of the cores. This situation could only get worse in the years to come, as dies with more cores will appear on the market.
== The problem ==
The heavyweight_hardware_concurrency() API was introduced so that only *one hardware thread per core* was used. Once that API returns, that original intention is lost, only the number of threads is retained. Consider a situation, on Windows, where the system has 2 CPU sockets, 18 cores each, each core having 2 hyper-threads, for a total of 72 hyper-threads. Both heavyweight_hardware_concurrency() and hardware_concurrency() currently return 36, because on Windows they are simply wrappers over std::thread::hardware_concurrency() -- which can only return processors from the current "processor group".
== The changes in this patch ==
To solve this situation, we capture (and retain) the initial intention until the point of usage, through a new ThreadPoolStrategy class. The number of threads to use is deferred as late as possible, until the moment where the std::threads are created (ThreadPool in the case of ThinLTO).
When using hardware_concurrency(), setting ThreadCount to 0 now means to use all the possible hardware CPU (SMT) threads. Providing a ThreadCount above to the maximum number of threads will have no effect, the maximum will be used instead.
The heavyweight_hardware_concurrency() is similar to hardware_concurrency(), except that only one thread per hardware *core* will be used.
When LLVM_ENABLE_THREADS is OFF, the threading APIs will always return 1, to ensure any caller loops will be exercised at least once.
Differential Revision: https://reviews.llvm.org/D71775
2020-02-14 11:49:57 +08:00
|
|
|
CompileThreads =
|
|
|
|
std::make_unique<ThreadPool>(hardware_concurrency(S.NumCompileThreads));
|
2019-04-30 06:37:27 +08:00
|
|
|
ES->setDispatchMaterialization(
|
|
|
|
[this](JITDylib &JD, std::unique_ptr<MaterializationUnit> MU) {
|
2019-09-13 19:59:51 +08:00
|
|
|
// FIXME: Switch to move capture once we have c++14.
|
|
|
|
auto SharedMU = std::shared_ptr<MaterializationUnit>(std::move(MU));
|
|
|
|
auto Work = [SharedMU, &JD]() { SharedMU->doMaterialize(JD); };
|
2019-04-30 06:37:27 +08:00
|
|
|
CompileThreads->async(std::move(Work));
|
|
|
|
});
|
|
|
|
}
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
|
|
|
|
if (S.SetUpPlatform)
|
|
|
|
Err = S.SetUpPlatform(*this);
|
|
|
|
else
|
|
|
|
setUpGenericLLVMIRPlatform(*this);
|
2018-09-27 00:26:59 +08:00
|
|
|
}
|
|
|
|
|
2018-06-27 05:35:48 +08:00
|
|
|
std::string LLJIT::mangle(StringRef UnmangledName) {
|
|
|
|
std::string MangledName;
|
|
|
|
{
|
|
|
|
raw_string_ostream MangledNameStream(MangledName);
|
|
|
|
Mangler::getNameWithPrefix(MangledNameStream, UnmangledName, DL);
|
|
|
|
}
|
|
|
|
return MangledName;
|
|
|
|
}
|
|
|
|
|
|
|
|
Error LLJIT::applyDataLayout(Module &M) {
|
|
|
|
if (M.getDataLayout().isDefault())
|
|
|
|
M.setDataLayout(DL);
|
|
|
|
|
|
|
|
if (M.getDataLayout() != DL)
|
|
|
|
return make_error<StringError>(
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
"Added modules have incompatible data layouts: " +
|
|
|
|
M.getDataLayout().getStringRepresentation() + " (module) vs " +
|
|
|
|
DL.getStringRepresentation() + " (jit)",
|
2018-06-27 05:35:48 +08:00
|
|
|
inconvertibleErrorCode());
|
|
|
|
|
|
|
|
return Error::success();
|
|
|
|
}
|
|
|
|
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
void setUpGenericLLVMIRPlatform(LLJIT &J) {
|
2020-03-05 04:36:50 +08:00
|
|
|
LLVM_DEBUG(
|
|
|
|
{ dbgs() << "Setting up GenericLLVMIRPlatform support for LLJIT\n"; });
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
J.setPlatformSupport(std::make_unique<GenericLLVMIRPlatformSupport>(J));
|
|
|
|
}
|
|
|
|
|
|
|
|
Error setUpMachOPlatform(LLJIT &J) {
|
2020-03-05 04:36:50 +08:00
|
|
|
LLVM_DEBUG({ dbgs() << "Setting up MachOPlatform support for LLJIT\n"; });
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
auto MP = MachOPlatformSupport::Create(J, J.getMainJITDylib());
|
|
|
|
if (!MP)
|
|
|
|
return MP.takeError();
|
|
|
|
J.setPlatformSupport(std::move(*MP));
|
|
|
|
return Error::success();
|
2018-06-27 05:35:48 +08:00
|
|
|
}
|
|
|
|
|
2019-04-30 06:37:27 +08:00
|
|
|
Error LLLazyJITBuilderState::prepareForConstruction() {
|
|
|
|
if (auto Err = LLJITBuilderState::prepareForConstruction())
|
|
|
|
return Err;
|
|
|
|
TT = JTMB->getTargetTriple();
|
|
|
|
return Error::success();
|
2018-06-27 05:35:48 +08:00
|
|
|
}
|
|
|
|
|
2018-09-26 09:24:12 +08:00
|
|
|
Error LLLazyJIT::addLazyIRModule(JITDylib &JD, ThreadSafeModule TSM) {
|
|
|
|
assert(TSM && "Can not add null module");
|
2018-06-27 05:35:48 +08:00
|
|
|
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
if (auto Err = TSM.withModuleDo(
|
|
|
|
[&](Module &M) -> Error { return applyDataLayout(M); }))
|
2019-08-02 23:21:37 +08:00
|
|
|
return Err;
|
2018-06-27 05:35:48 +08:00
|
|
|
|
2019-04-30 06:37:27 +08:00
|
|
|
return CODLayer->add(JD, std::move(TSM), ES->allocateVModule());
|
2018-06-27 05:35:48 +08:00
|
|
|
}
|
|
|
|
|
2019-04-30 06:37:27 +08:00
|
|
|
LLLazyJIT::LLLazyJIT(LLLazyJITBuilderState &S, Error &Err) : LLJIT(S, Err) {
|
|
|
|
|
|
|
|
// If LLJIT construction failed then bail out.
|
|
|
|
if (Err)
|
|
|
|
return;
|
|
|
|
|
|
|
|
ErrorAsOutParameter _(&Err);
|
|
|
|
|
|
|
|
/// Take/Create the lazy-compile callthrough manager.
|
|
|
|
if (S.LCTMgr)
|
|
|
|
LCTMgr = std::move(S.LCTMgr);
|
|
|
|
else {
|
|
|
|
if (auto LCTMgrOrErr = createLocalLazyCallThroughManager(
|
|
|
|
S.TT, *ES, S.LazyCompileFailureAddr))
|
|
|
|
LCTMgr = std::move(*LCTMgrOrErr);
|
|
|
|
else {
|
|
|
|
Err = LCTMgrOrErr.takeError();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Take/Create the indirect stubs manager builder.
|
|
|
|
auto ISMBuilder = std::move(S.ISMBuilder);
|
|
|
|
|
|
|
|
// If none was provided, try to build one.
|
|
|
|
if (!ISMBuilder)
|
|
|
|
ISMBuilder = createLocalIndirectStubsManagerBuilder(S.TT);
|
|
|
|
|
|
|
|
// No luck. Bail out.
|
|
|
|
if (!ISMBuilder) {
|
|
|
|
Err = make_error<StringError>("Could not construct "
|
|
|
|
"IndirectStubsManagerBuilder for target " +
|
|
|
|
S.TT.str(),
|
|
|
|
inconvertibleErrorCode());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create the COD layer.
|
2019-08-15 23:54:37 +08:00
|
|
|
CODLayer = std::make_unique<CompileOnDemandLayer>(
|
[ORC] Add generic initializer/deinitializer support.
Initializers and deinitializers are used to implement C++ static constructors
and destructors, runtime registration for some languages (e.g. with the
Objective-C runtime for Objective-C/C++ code) and other tasks that would
typically be performed when a shared-object/dylib is loaded or unloaded by a
statically compiled program.
MCJIT and ORC have historically provided limited support for discovering and
running initializers/deinitializers by scanning the llvm.global_ctors and
llvm.global_dtors variables and recording the functions to be run. This approach
suffers from several drawbacks: (1) It only works for IR inputs, not for object
files (including cached JIT'd objects). (2) It only works for initializers
described by llvm.global_ctors and llvm.global_dtors, however not all
initializers are described in this way (Objective-C, for example, describes
initializers via specially named metadata sections). (3) To make the
initializer/deinitializer functions described by llvm.global_ctors and
llvm.global_dtors searchable they must be promoted to extern linkage, polluting
the JIT symbol table (extra care must be taken to ensure this promotion does
not result in symbol name clashes).
This patch introduces several interdependent changes to ORCv2 to support the
construction of new initialization schemes, and includes an implementation of a
backwards-compatible llvm.global_ctor/llvm.global_dtor scanning scheme, and a
MachO specific scheme that handles Objective-C runtime registration (if the
Objective-C runtime is available) enabling execution of LLVM IR compiled from
Objective-C and Swift.
The major changes included in this patch are:
(1) The MaterializationUnit and MaterializationResponsibility classes are
extended to describe an optional "initializer" symbol for the module (see the
getInitializerSymbol method on each class). The presence or absence of this
symbol indicates whether the module contains any initializers or
deinitializers. The initializer symbol otherwise behaves like any other:
searching for it triggers materialization.
(2) A new Platform interface is introduced in llvm/ExecutionEngine/Orc/Core.h
which provides the following callback interface:
- Error setupJITDylib(JITDylib &JD): Can be used to install standard symbols
in JITDylibs upon creation. E.g. __dso_handle.
- Error notifyAdding(JITDylib &JD, const MaterializationUnit &MU): Generally
used to record initializer symbols.
- Error notifyRemoving(JITDylib &JD, VModuleKey K): Used to notify a platform
that a module is being removed.
Platform implementations can use these callbacks to track outstanding
initializers and implement a platform-specific approach for executing them. For
example, the MachOPlatform installs a plugin in the JIT linker to scan for both
__mod_inits sections (for C++ static constructors) and ObjC metadata sections.
If discovered, these are processed in the usual platform order: Objective-C
registration is carried out first, then static initializers are executed,
ensuring that calls to Objective-C from static initializers will be safe.
This patch updates LLJIT to use the new scheme for initialization. Two
LLJIT::PlatformSupport classes are implemented: A GenericIR platform and a MachO
platform. The GenericIR platform implements a modified version of the previous
llvm.global-ctor scraping scheme to provide support for Windows and
Linux. LLJIT's MachO platform uses the MachOPlatform class to provide MachO
specific initialization as described above.
Reviewers: sgraenitz, dblaikie
Subscribers: mgorny, hiraditya, mgrang, ributzka, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D74300
2019-12-16 18:50:40 +08:00
|
|
|
*ES, *InitHelperTransformLayer, *LCTMgr, std::move(ISMBuilder));
|
2019-04-30 06:37:27 +08:00
|
|
|
|
|
|
|
if (S.NumCompileThreads > 0)
|
|
|
|
CODLayer->setCloneToNewContextOnEmit(true);
|
2018-09-28 05:13:07 +08:00
|
|
|
}
|
2018-09-27 00:26:59 +08:00
|
|
|
|
2018-06-27 05:35:48 +08:00
|
|
|
} // End namespace orc.
|
|
|
|
} // End namespace llvm.
|