//===-- AMDGPUCtorDtorLowering.cpp - Handle global ctors and dtors --------===// // // 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 // //===----------------------------------------------------------------------===// /// /// \file /// This pass creates a unified init and fini kernel with the required metadata //===----------------------------------------------------------------------===// #include "AMDGPU.h" #include "llvm/IR/Constants.h" #include "llvm/IR/Function.h" #include "llvm/IR/GlobalVariable.h" #include "llvm/IR/IRBuilder.h" #include "llvm/IR/Module.h" #include "llvm/IR/Value.h" #include "llvm/Pass.h" #include "llvm/Transforms/Utils/ModuleUtils.h" using namespace llvm; #define DEBUG_TYPE "amdgpu-lower-ctor-dtor" namespace { class AMDGPUCtorDtorLowering final : public ModulePass { bool runOnModule(Module &M) override; public: Function *createInitOrFiniKernelFunction(Module &M, bool IsCtor) { StringRef InitOrFiniKernelName = "amdgcn.device.init"; if (!IsCtor) InitOrFiniKernelName = "amdgcn.device.fini"; Function *InitOrFiniKernel = Function::createWithDefaultAttr( FunctionType::get(Type::getVoidTy(M.getContext()), false), GlobalValue::ExternalLinkage, 0, InitOrFiniKernelName, &M); BasicBlock *InitOrFiniKernelBB = BasicBlock::Create(M.getContext(), "", InitOrFiniKernel); ReturnInst::Create(M.getContext(), InitOrFiniKernelBB); InitOrFiniKernel->setCallingConv(CallingConv::AMDGPU_KERNEL); if (IsCtor) InitOrFiniKernel->addFnAttr("device-init"); else InitOrFiniKernel->addFnAttr("device-fini"); return InitOrFiniKernel; } bool createInitOrFiniKernel(Module &M, GlobalVariable *GV, bool IsCtor) { if (!GV || !GV->hasInitializer()) return false; ConstantArray *GA = dyn_cast(GV->getInitializer()); if (!GA || GA->getNumOperands() == 0) return false; Function *InitOrFiniKernel = createInitOrFiniKernelFunction(M, IsCtor); IRBuilder<> IRB(InitOrFiniKernel->getEntryBlock().getTerminator()); for (Value *V : GA->operands()) { auto *CS = cast(V); if (Function *F = dyn_cast(CS->getOperand(1))) { FunctionCallee Ctor = M.getOrInsertFunction(F->getName(), IRB.getVoidTy()); IRB.CreateCall(Ctor); } } appendToUsed(M, {InitOrFiniKernel}); return true; } static char ID; AMDGPUCtorDtorLowering() : ModulePass(ID) {} }; } // End anonymous namespace char AMDGPUCtorDtorLowering::ID = 0; char &llvm::AMDGPUCtorDtorLoweringID = AMDGPUCtorDtorLowering::ID; INITIALIZE_PASS(AMDGPUCtorDtorLowering, DEBUG_TYPE, "Lower ctors and dtors for AMDGPU", false, false) ModulePass *llvm::createAMDGPUCtorDtorLoweringPass() { return new AMDGPUCtorDtorLowering(); } bool AMDGPUCtorDtorLowering::runOnModule(Module &M) { bool Modified = false; Modified |= createInitOrFiniKernel(M, M.getGlobalVariable("llvm.global_ctors"), /*IsCtor =*/true); Modified |= createInitOrFiniKernel(M, M.getGlobalVariable("llvm.global_dtors"), /*IsCtor =*/false); return Modified; }