32#define DEBUG_TYPE "nvptx-lower-ctor-dtor"
36 cl::desc(
"Override unique ID of ctor/dtor globals."),
41 cl::desc(
"Emit kernels to call ctor/dtor globals."),
46static std::string getHash(
StringRef Str) {
60static Function *createInitOrFiniKernelFunction(
Module &M,
bool IsCtor) {
62 IsCtor ?
"nvptx$device$init" :
"nvptx$device$fini";
63 if (M.getFunction(InitOrFiniKernelName))
69 addKernelAttrs(InitOrFiniKernel);
71 return InitOrFiniKernel;
97static void createInitOrFiniCalls(
Function &
F,
bool IsCtor) {
106 auto *Begin = M.getOrInsertGlobal(
107 IsCtor ?
"__init_array_start" :
"__fini_array_start",
113 IsCtor ?
"__init_array_start" :
"__fini_array_start",
119 auto *End = M.getOrInsertGlobal(
126 IsCtor ?
"__init_array_end" :
"__fini_array_end",
139 Value *BeginVal = IRB.CreateLoad(Begin->getType(), Begin,
"begin");
140 Value *EndVal = IRB.CreateLoad(Begin->getType(), End,
"stop");
142 Value *OldBeginVal = BeginVal;
148 EndVal = OldBeginVal;
154 IRB.SetInsertPoint(LoopBB);
155 auto *CallBackPHI = IRB.CreatePHI(PtrTy, 2,
"ptr");
156 auto *CallBack = IRB.CreateLoad(IRB.getPtrTy(
F.getAddressSpace()),
157 CallBackPHI,
"callback");
158 IRB.CreateCall(CallBackTy, CallBack);
160 IRB.CreateConstGEP1_64(PtrTy, CallBackPHI, IsCtor ? 1 : -1,
"next");
162 NewCallBack, EndVal,
"end");
163 CallBackPHI->addIncoming(BeginVal, &
F.getEntryBlock());
164 CallBackPHI->addIncoming(NewCallBack, LoopBB);
165 IRB.CreateCondBr(EndCmp, ExitBB, LoopBB);
166 IRB.SetInsertPoint(ExitBB);
183 std::string PriorityStr =
"." + std::to_string(Priority);
187 std::string NameStr =
188 ((IsCtor ?
"__init_array_object_" :
"__fini_array_object_") +
189 F->getName() +
"_" + GlobalID +
"_" + std::to_string(Priority))
193 [](
char c) { return c ==
'.' ?
'_' : c; });
200 GV->
setSection(IsCtor ?
".init_array" + PriorityStr
201 :
".fini_array" + PriorityStr);
215 if (!createInitOrFiniGlobals(M, GV, IsCtor))
221 Function *InitOrFiniKernel = createInitOrFiniKernelFunction(M, IsCtor);
222 if (!InitOrFiniKernel)
225 createInitOrFiniCalls(*InitOrFiniKernel, IsCtor);
231static bool lowerCtorsAndDtors(
Module &M) {
233 Modified |= createInitOrFiniKernel(M,
"llvm.global_ctors",
true);
234 Modified |= createInitOrFiniKernel(M,
"llvm.global_dtors",
false);
238class NVPTXCtorDtorLoweringLegacy final :
public ModulePass {
241 NVPTXCtorDtorLoweringLegacy() : ModulePass(ID) {}
242 bool runOnModule(
Module &M)
override {
return lowerCtorsAndDtors(M); }
253char NVPTXCtorDtorLoweringLegacy::ID = 0;
255 "Lower ctors and dtors for NVPTX",
false,
false)
258 return new NVPTXCtorDtorLoweringLegacy();
This file contains the declarations for the subclasses of Constant, which represent the different fla...
Module.h This file contains the declarations for the Module class.
Machine Check Debug Module
static cl::opt< bool > CreateKernels("nvptx-emit-init-fini-kernel", cl::desc("Emit kernels to call ctor/dtor globals."), cl::init(true), cl::Hidden)
static cl::opt< std::string > GlobalStr("nvptx-lower-global-ctor-dtor-id", cl::desc("Override unique ID of ctor/dtor globals."), cl::init(""), cl::Hidden)
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
ArrayRef - Represent a constant reference to an array (0 or more elements consecutively in memory),...
static BasicBlock * Create(LLVMContext &Context, const Twine &Name="", Function *Parent=nullptr, BasicBlock *InsertBefore=nullptr)
Creates a new BasicBlock.
@ ICMP_UGE
unsigned greater or equal
@ ICMP_ULT
unsigned less than
ConstantArray - Constant Array Declarations.
static LLVM_ABI Constant * getAllOnesValue(Type *Ty)
static LLVM_ABI Constant * getNullValue(Type *Ty)
Constructor to create a '0' constant of arbitrary type.
static LLVM_ABI FunctionType * get(Type *Result, ArrayRef< Type * > Params, bool isVarArg)
This static method is the primary way of constructing a FunctionType.
static Function * createWithDefaultAttr(FunctionType *Ty, LinkageTypes Linkage, unsigned AddrSpace, const Twine &N="", Module *M=nullptr)
Creates a function with some attributes recorded in llvm.module.flags and the LLVMContext applied.
LLVM_ABI void setSection(StringRef S)
Change the section for this global.
@ ProtectedVisibility
The GV is protected.
void setVisibility(VisibilityTypes V)
@ WeakODRLinkage
Same, but only replaced by something equivalent.
@ ExternalLinkage
Externally visible function.
@ WeakAnyLinkage
Keep one copy of named function when linking (weak)
const Constant * getInitializer() const
getInitializer - Return the initializer for this global variable.
bool hasInitializer() const
Definitions have initializers, declarations don't.
LLVM_ABI void eraseFromParent()
eraseFromParent - This method unlinks 'this' from the containing module and deletes it.
This provides a uniform API for creating instructions and inserting them into a basic block: either a...
This is an important class for using LLVM in a threaded context.
LLVM_ABI void update(ArrayRef< uint8_t > Data)
Updates the hash for the byte stream provided.
LLVM_ABI void final(MD5Result &Result)
Finishes off the hash and puts the result in result.
ModulePass class - This class is used to implement unstructured interprocedural optimizations and ana...
A Module instance is used to store all the information related to an LLVM module.
PreservedAnalyses run(Module &M, ModuleAnalysisManager &AM)
static LLVM_ABI PointerType * get(Type *ElementType, unsigned AddressSpace)
This constructs a pointer to an object of the specified type in a numbered address space.
A set of analyses that are preserved following a run of a transformation pass.
static PreservedAnalyses none()
Convenience factory function for the empty preserved set.
static PreservedAnalyses all()
Construct a special preserved set that preserves all passes.
StringRef - Represent a constant reference to a string, i.e.
The instances of the Type class are immutable: once they are created, they are never changed.
static LLVM_ABI IntegerType * getInt64Ty(LLVMContext &C)
static LLVM_ABI Type * getVoidTy(LLVMContext &C)
unsigned getNumOperands() const
LLVM Value Representation.
@ PTX_Kernel
Call to a PTX kernel. Passes all arguments in parameter space.
@ C
The default llvm calling convention, compatible with C.
constexpr StringLiteral MaxNTID("nvvm.maxntid")
constexpr StringLiteral MaxClusterRank("nvvm.maxclusterrank")
initializer< Ty > init(const Ty &Val)
This is an optimization pass for GlobalISel generic memory operations.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
std::string utohexstr(uint64_t X, bool LowerCase=false, unsigned Width=0)
OutputIt transform(R &&Range, OutputIt d_first, UnaryFunction F)
Wrapper function around std::transform to apply a function to a range and store the result elsewhere.
ModulePass * createNVPTXCtorDtorLoweringLegacyPass()
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
LLVM_ABI void appendToUsed(Module &M, ArrayRef< GlobalValue * > Values)
Adds global values to the llvm.used list.
AnalysisManager< Module > ModuleAnalysisManager
Convenience typedef for the Module analysis manager.