CGCUDANV.cpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317
  1. //===----- CGCUDANV.cpp - Interface to NVIDIA CUDA Runtime ----------------===//
  2. //
  3. // The LLVM Compiler Infrastructure
  4. //
  5. // This file is distributed under the University of Illinois Open Source
  6. // License. See LICENSE.TXT for details.
  7. //
  8. //===----------------------------------------------------------------------===//
  9. //
  10. // This provides a class for CUDA code generation targeting the NVIDIA CUDA
  11. // runtime library.
  12. //
  13. //===----------------------------------------------------------------------===//
  14. #include "CGCUDARuntime.h"
  15. #include "CodeGenFunction.h"
  16. #include "CodeGenModule.h"
  17. #include "clang/AST/Decl.h"
  18. #include "llvm/IR/BasicBlock.h"
  19. #include "llvm/IR/CallSite.h"
  20. #include "llvm/IR/Constants.h"
  21. #include "llvm/IR/DerivedTypes.h"
  22. using namespace clang;
  23. using namespace CodeGen;
  24. namespace {
  25. class CGNVCUDARuntime : public CGCUDARuntime {
  26. private:
  27. llvm::Type *IntTy, *SizeTy, *VoidTy;
  28. llvm::PointerType *CharPtrTy, *VoidPtrTy, *VoidPtrPtrTy;
  29. /// Convenience reference to LLVM Context
  30. llvm::LLVMContext &Context;
  31. /// Convenience reference to the current module
  32. llvm::Module &TheModule;
  33. /// Keeps track of kernel launch stubs emitted in this module
  34. llvm::SmallVector<llvm::Function *, 16> EmittedKernels;
  35. /// Keeps track of variables containing handles of GPU binaries. Populated by
  36. /// ModuleCtorFunction() and used to create corresponding cleanup calls in
  37. /// ModuleDtorFunction()
  38. llvm::SmallVector<llvm::GlobalVariable *, 16> GpuBinaryHandles;
  39. llvm::Constant *getSetupArgumentFn() const;
  40. llvm::Constant *getLaunchFn() const;
  41. /// Creates a function to register all kernel stubs generated in this module.
  42. llvm::Function *makeRegisterKernelsFn();
  43. /// Helper function that generates a constant string and returns a pointer to
  44. /// the start of the string. The result of this function can be used anywhere
  45. /// where the C code specifies const char*.
  46. llvm::Constant *makeConstantString(const std::string &Str,
  47. const std::string &Name = "",
  48. unsigned Alignment = 0) {
  49. llvm::Constant *Zeros[] = {llvm::ConstantInt::get(SizeTy, 0),
  50. llvm::ConstantInt::get(SizeTy, 0)};
  51. auto *ConstStr = CGM.GetAddrOfConstantCString(Str, Name.c_str());
  52. return llvm::ConstantExpr::getGetElementPtr(ConstStr->getValueType(),
  53. ConstStr, Zeros);
  54. }
  55. void emitDeviceStubBody(CodeGenFunction &CGF, FunctionArgList &Args);
  56. public:
  57. CGNVCUDARuntime(CodeGenModule &CGM);
  58. void emitDeviceStub(CodeGenFunction &CGF, FunctionArgList &Args) override;
  59. /// Creates module constructor function
  60. llvm::Function *makeModuleCtorFunction() override;
  61. /// Creates module destructor function
  62. llvm::Function *makeModuleDtorFunction() override;
  63. };
  64. }
  65. CGNVCUDARuntime::CGNVCUDARuntime(CodeGenModule &CGM)
  66. : CGCUDARuntime(CGM), Context(CGM.getLLVMContext()),
  67. TheModule(CGM.getModule()) {
  68. CodeGen::CodeGenTypes &Types = CGM.getTypes();
  69. ASTContext &Ctx = CGM.getContext();
  70. IntTy = Types.ConvertType(Ctx.IntTy);
  71. SizeTy = Types.ConvertType(Ctx.getSizeType());
  72. VoidTy = llvm::Type::getVoidTy(Context);
  73. CharPtrTy = llvm::PointerType::getUnqual(Types.ConvertType(Ctx.CharTy));
  74. VoidPtrTy = cast<llvm::PointerType>(Types.ConvertType(Ctx.VoidPtrTy));
  75. VoidPtrPtrTy = VoidPtrTy->getPointerTo();
  76. }
  77. llvm::Constant *CGNVCUDARuntime::getSetupArgumentFn() const {
  78. // cudaError_t cudaSetupArgument(void *, size_t, size_t)
  79. std::vector<llvm::Type*> Params;
  80. Params.push_back(VoidPtrTy);
  81. Params.push_back(SizeTy);
  82. Params.push_back(SizeTy);
  83. return CGM.CreateRuntimeFunction(llvm::FunctionType::get(IntTy,
  84. Params, false),
  85. "cudaSetupArgument");
  86. }
  87. llvm::Constant *CGNVCUDARuntime::getLaunchFn() const {
  88. // cudaError_t cudaLaunch(char *)
  89. return CGM.CreateRuntimeFunction(
  90. llvm::FunctionType::get(IntTy, CharPtrTy, false), "cudaLaunch");
  91. }
  92. void CGNVCUDARuntime::emitDeviceStub(CodeGenFunction &CGF,
  93. FunctionArgList &Args) {
  94. EmittedKernels.push_back(CGF.CurFn);
  95. emitDeviceStubBody(CGF, Args);
  96. }
  97. void CGNVCUDARuntime::emitDeviceStubBody(CodeGenFunction &CGF,
  98. FunctionArgList &Args) {
  99. // Build the argument value list and the argument stack struct type.
  100. SmallVector<llvm::Value *, 16> ArgValues;
  101. std::vector<llvm::Type *> ArgTypes;
  102. for (FunctionArgList::const_iterator I = Args.begin(), E = Args.end();
  103. I != E; ++I) {
  104. llvm::Value *V = CGF.GetAddrOfLocalVar(*I);
  105. ArgValues.push_back(V);
  106. assert(isa<llvm::PointerType>(V->getType()) && "Arg type not PointerType");
  107. ArgTypes.push_back(cast<llvm::PointerType>(V->getType())->getElementType());
  108. }
  109. llvm::StructType *ArgStackTy = llvm::StructType::get(Context, ArgTypes);
  110. llvm::BasicBlock *EndBlock = CGF.createBasicBlock("setup.end");
  111. // Emit the calls to cudaSetupArgument
  112. llvm::Constant *cudaSetupArgFn = getSetupArgumentFn();
  113. for (unsigned I = 0, E = Args.size(); I != E; ++I) {
  114. llvm::Value *Args[3];
  115. llvm::BasicBlock *NextBlock = CGF.createBasicBlock("setup.next");
  116. Args[0] = CGF.Builder.CreatePointerCast(ArgValues[I], VoidPtrTy);
  117. Args[1] = CGF.Builder.CreateIntCast(
  118. llvm::ConstantExpr::getSizeOf(ArgTypes[I]),
  119. SizeTy, false);
  120. Args[2] = CGF.Builder.CreateIntCast(
  121. llvm::ConstantExpr::getOffsetOf(ArgStackTy, I),
  122. SizeTy, false);
  123. llvm::CallSite CS = CGF.EmitRuntimeCallOrInvoke(cudaSetupArgFn, Args);
  124. llvm::Constant *Zero = llvm::ConstantInt::get(IntTy, 0);
  125. llvm::Value *CSZero = CGF.Builder.CreateICmpEQ(CS.getInstruction(), Zero);
  126. CGF.Builder.CreateCondBr(CSZero, NextBlock, EndBlock);
  127. CGF.EmitBlock(NextBlock);
  128. }
  129. // Emit the call to cudaLaunch
  130. llvm::Constant *cudaLaunchFn = getLaunchFn();
  131. llvm::Value *Arg = CGF.Builder.CreatePointerCast(CGF.CurFn, CharPtrTy);
  132. CGF.EmitRuntimeCallOrInvoke(cudaLaunchFn, Arg);
  133. CGF.EmitBranch(EndBlock);
  134. CGF.EmitBlock(EndBlock);
  135. }
  136. /// Creates internal function to register all kernel stubs generated in this
  137. /// module with the CUDA runtime.
  138. /// \code
  139. /// void __cuda_register_kernels(void** GpuBinaryHandle) {
  140. /// __cudaRegisterFunction(GpuBinaryHandle,Kernel0,...);
  141. /// ...
  142. /// __cudaRegisterFunction(GpuBinaryHandle,KernelM,...);
  143. /// }
  144. /// \endcode
  145. llvm::Function *CGNVCUDARuntime::makeRegisterKernelsFn() {
  146. llvm::Function *RegisterKernelsFunc = llvm::Function::Create(
  147. llvm::FunctionType::get(VoidTy, VoidPtrPtrTy, false),
  148. llvm::GlobalValue::InternalLinkage, "__cuda_register_kernels", &TheModule);
  149. llvm::BasicBlock *EntryBB =
  150. llvm::BasicBlock::Create(Context, "entry", RegisterKernelsFunc);
  151. CGBuilderTy Builder(Context);
  152. Builder.SetInsertPoint(EntryBB);
  153. // void __cudaRegisterFunction(void **, const char *, char *, const char *,
  154. // int, uint3*, uint3*, dim3*, dim3*, int*)
  155. std::vector<llvm::Type *> RegisterFuncParams = {
  156. VoidPtrPtrTy, CharPtrTy, CharPtrTy, CharPtrTy, IntTy,
  157. VoidPtrTy, VoidPtrTy, VoidPtrTy, VoidPtrTy, IntTy->getPointerTo()};
  158. llvm::Constant *RegisterFunc = CGM.CreateRuntimeFunction(
  159. llvm::FunctionType::get(IntTy, RegisterFuncParams, false),
  160. "__cudaRegisterFunction");
  161. // Extract GpuBinaryHandle passed as the first argument passed to
  162. // __cuda_register_kernels() and generate __cudaRegisterFunction() call for
  163. // each emitted kernel.
  164. llvm::Argument &GpuBinaryHandlePtr = *RegisterKernelsFunc->arg_begin();
  165. for (llvm::Function *Kernel : EmittedKernels) {
  166. llvm::Constant *KernelName = makeConstantString(Kernel->getName());
  167. llvm::Constant *NullPtr = llvm::ConstantPointerNull::get(VoidPtrTy);
  168. llvm::Value *args[] = {
  169. &GpuBinaryHandlePtr, Builder.CreateBitCast(Kernel, VoidPtrTy),
  170. KernelName, KernelName, llvm::ConstantInt::get(IntTy, -1), NullPtr,
  171. NullPtr, NullPtr, NullPtr,
  172. llvm::ConstantPointerNull::get(IntTy->getPointerTo())};
  173. Builder.CreateCall(RegisterFunc, args);
  174. }
  175. Builder.CreateRetVoid();
  176. return RegisterKernelsFunc;
  177. }
  178. /// Creates a global constructor function for the module:
  179. /// \code
  180. /// void __cuda_module_ctor(void*) {
  181. /// Handle0 = __cudaRegisterFatBinary(GpuBinaryBlob0);
  182. /// __cuda_register_kernels(Handle0);
  183. /// ...
  184. /// HandleN = __cudaRegisterFatBinary(GpuBinaryBlobN);
  185. /// __cuda_register_kernels(HandleN);
  186. /// }
  187. /// \endcode
  188. llvm::Function *CGNVCUDARuntime::makeModuleCtorFunction() {
  189. // void __cuda_register_kernels(void* handle);
  190. llvm::Function *RegisterKernelsFunc = makeRegisterKernelsFn();
  191. // void ** __cudaRegisterFatBinary(void *);
  192. llvm::Constant *RegisterFatbinFunc = CGM.CreateRuntimeFunction(
  193. llvm::FunctionType::get(VoidPtrPtrTy, VoidPtrTy, false),
  194. "__cudaRegisterFatBinary");
  195. // struct { int magic, int version, void * gpu_binary, void * dont_care };
  196. llvm::StructType *FatbinWrapperTy =
  197. llvm::StructType::get(IntTy, IntTy, VoidPtrTy, VoidPtrTy, nullptr);
  198. llvm::Function *ModuleCtorFunc = llvm::Function::Create(
  199. llvm::FunctionType::get(VoidTy, VoidPtrTy, false),
  200. llvm::GlobalValue::InternalLinkage, "__cuda_module_ctor", &TheModule);
  201. llvm::BasicBlock *CtorEntryBB =
  202. llvm::BasicBlock::Create(Context, "entry", ModuleCtorFunc);
  203. CGBuilderTy CtorBuilder(Context);
  204. CtorBuilder.SetInsertPoint(CtorEntryBB);
  205. // For each GPU binary, register it with the CUDA runtime and store returned
  206. // handle in a global variable and save the handle in GpuBinaryHandles vector
  207. // to be cleaned up in destructor on exit. Then associate all known kernels
  208. // with the GPU binary handle so CUDA runtime can figure out what to call on
  209. // the GPU side.
  210. for (const std::string &GpuBinaryFileName :
  211. CGM.getCodeGenOpts().CudaGpuBinaryFileNames) {
  212. llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> GpuBinaryOrErr =
  213. llvm::MemoryBuffer::getFileOrSTDIN(GpuBinaryFileName);
  214. if (std::error_code EC = GpuBinaryOrErr.getError()) {
  215. CGM.getDiags().Report(diag::err_cannot_open_file) << GpuBinaryFileName
  216. << EC.message();
  217. continue;
  218. }
  219. // Create initialized wrapper structure that points to the loaded GPU binary
  220. llvm::Constant *Values[] = {
  221. llvm::ConstantInt::get(IntTy, 0x466243b1), // Fatbin wrapper magic.
  222. llvm::ConstantInt::get(IntTy, 1), // Fatbin version.
  223. makeConstantString(GpuBinaryOrErr.get()->getBuffer(), "", 16), // Data.
  224. llvm::ConstantPointerNull::get(VoidPtrTy)}; // Unused in fatbin v1.
  225. llvm::GlobalVariable *FatbinWrapper = new llvm::GlobalVariable(
  226. TheModule, FatbinWrapperTy, true, llvm::GlobalValue::InternalLinkage,
  227. llvm::ConstantStruct::get(FatbinWrapperTy, Values),
  228. "__cuda_fatbin_wrapper");
  229. // GpuBinaryHandle = __cudaRegisterFatBinary(&FatbinWrapper);
  230. llvm::CallInst *RegisterFatbinCall = CtorBuilder.CreateCall(
  231. RegisterFatbinFunc,
  232. CtorBuilder.CreateBitCast(FatbinWrapper, VoidPtrTy));
  233. llvm::GlobalVariable *GpuBinaryHandle = new llvm::GlobalVariable(
  234. TheModule, VoidPtrPtrTy, false, llvm::GlobalValue::InternalLinkage,
  235. llvm::ConstantPointerNull::get(VoidPtrPtrTy), "__cuda_gpubin_handle");
  236. CtorBuilder.CreateStore(RegisterFatbinCall, GpuBinaryHandle, false);
  237. // Call __cuda_register_kernels(GpuBinaryHandle);
  238. CtorBuilder.CreateCall(RegisterKernelsFunc, RegisterFatbinCall);
  239. // Save GpuBinaryHandle so we can unregister it in destructor.
  240. GpuBinaryHandles.push_back(GpuBinaryHandle);
  241. }
  242. CtorBuilder.CreateRetVoid();
  243. return ModuleCtorFunc;
  244. }
  245. /// Creates a global destructor function that unregisters all GPU code blobs
  246. /// registered by constructor.
  247. /// \code
  248. /// void __cuda_module_dtor(void*) {
  249. /// __cudaUnregisterFatBinary(Handle0);
  250. /// ...
  251. /// __cudaUnregisterFatBinary(HandleN);
  252. /// }
  253. /// \endcode
  254. llvm::Function *CGNVCUDARuntime::makeModuleDtorFunction() {
  255. // void __cudaUnregisterFatBinary(void ** handle);
  256. llvm::Constant *UnregisterFatbinFunc = CGM.CreateRuntimeFunction(
  257. llvm::FunctionType::get(VoidTy, VoidPtrPtrTy, false),
  258. "__cudaUnregisterFatBinary");
  259. llvm::Function *ModuleDtorFunc = llvm::Function::Create(
  260. llvm::FunctionType::get(VoidTy, VoidPtrTy, false),
  261. llvm::GlobalValue::InternalLinkage, "__cuda_module_dtor", &TheModule);
  262. llvm::BasicBlock *DtorEntryBB =
  263. llvm::BasicBlock::Create(Context, "entry", ModuleDtorFunc);
  264. CGBuilderTy DtorBuilder(Context);
  265. DtorBuilder.SetInsertPoint(DtorEntryBB);
  266. for (llvm::GlobalVariable *GpuBinaryHandle : GpuBinaryHandles) {
  267. DtorBuilder.CreateCall(UnregisterFatbinFunc,
  268. DtorBuilder.CreateLoad(GpuBinaryHandle, false));
  269. }
  270. DtorBuilder.CreateRetVoid();
  271. return ModuleDtorFunc;
  272. }
  273. CGCUDARuntime *CodeGen::CreateNVCUDARuntime(CodeGenModule &CGM) {
  274. return new CGNVCUDARuntime(CGM);
  275. }