1992cb984SSergei Barannikov //===- SPIR.cpp -----------------------------------------------------------===// 2992cb984SSergei Barannikov // 3992cb984SSergei Barannikov // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4992cb984SSergei Barannikov // See https://llvm.org/LICENSE.txt for license information. 5992cb984SSergei Barannikov // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6992cb984SSergei Barannikov // 7992cb984SSergei Barannikov //===----------------------------------------------------------------------===// 8992cb984SSergei Barannikov 9992cb984SSergei Barannikov #include "ABIInfoImpl.h" 10992cb984SSergei Barannikov #include "TargetInfo.h" 11992cb984SSergei Barannikov 12992cb984SSergei Barannikov using namespace clang; 13992cb984SSergei Barannikov using namespace clang::CodeGen; 14992cb984SSergei Barannikov 15992cb984SSergei Barannikov //===----------------------------------------------------------------------===// 16992cb984SSergei Barannikov // Base ABI and target codegen info implementation common between SPIR and 17992cb984SSergei Barannikov // SPIR-V. 18992cb984SSergei Barannikov //===----------------------------------------------------------------------===// 19992cb984SSergei Barannikov 20992cb984SSergei Barannikov namespace { 21992cb984SSergei Barannikov class CommonSPIRABIInfo : public DefaultABIInfo { 22992cb984SSergei Barannikov public: 23992cb984SSergei Barannikov CommonSPIRABIInfo(CodeGenTypes &CGT) : DefaultABIInfo(CGT) { setCCs(); } 24992cb984SSergei Barannikov 25992cb984SSergei Barannikov private: 26992cb984SSergei Barannikov void setCCs(); 27992cb984SSergei Barannikov }; 28992cb984SSergei Barannikov 29992cb984SSergei Barannikov class SPIRVABIInfo : public CommonSPIRABIInfo { 30992cb984SSergei Barannikov public: 31992cb984SSergei Barannikov SPIRVABIInfo(CodeGenTypes &CGT) : CommonSPIRABIInfo(CGT) {} 32992cb984SSergei Barannikov void computeInfo(CGFunctionInfo &FI) const override; 33992cb984SSergei Barannikov 34992cb984SSergei Barannikov private: 35ad435bccSAlex Voicu ABIArgInfo classifyReturnType(QualType RetTy) const; 36992cb984SSergei Barannikov ABIArgInfo classifyKernelArgumentType(QualType Ty) const; 37ad435bccSAlex Voicu ABIArgInfo classifyArgumentType(QualType Ty) const; 38992cb984SSergei Barannikov }; 39992cb984SSergei Barannikov } // end anonymous namespace 40992cb984SSergei Barannikov namespace { 41992cb984SSergei Barannikov class CommonSPIRTargetCodeGenInfo : public TargetCodeGenInfo { 42992cb984SSergei Barannikov public: 43992cb984SSergei Barannikov CommonSPIRTargetCodeGenInfo(CodeGen::CodeGenTypes &CGT) 44992cb984SSergei Barannikov : TargetCodeGenInfo(std::make_unique<CommonSPIRABIInfo>(CGT)) {} 45992cb984SSergei Barannikov CommonSPIRTargetCodeGenInfo(std::unique_ptr<ABIInfo> ABIInfo) 46992cb984SSergei Barannikov : TargetCodeGenInfo(std::move(ABIInfo)) {} 47992cb984SSergei Barannikov 48992cb984SSergei Barannikov LangAS getASTAllocaAddressSpace() const override { 49992cb984SSergei Barannikov return getLangASFromTargetAS( 50992cb984SSergei Barannikov getABIInfo().getDataLayout().getAllocaAddrSpace()); 51992cb984SSergei Barannikov } 52992cb984SSergei Barannikov 53992cb984SSergei Barannikov unsigned getOpenCLKernelCallingConv() const override; 54992cb984SSergei Barannikov llvm::Type *getOpenCLType(CodeGenModule &CGM, const Type *T) const override; 55d6344c1cSSteven Perron llvm::Type *getHLSLType(CodeGenModule &CGM, const Type *Ty) const override; 56d6344c1cSSteven Perron llvm::Type *getSPIRVImageTypeFromHLSLResource( 57d6344c1cSSteven Perron const HLSLAttributedResourceType::Attributes &attributes, 58d6344c1cSSteven Perron llvm::Type *ElementType, llvm::LLVMContext &Ctx) const; 59992cb984SSergei Barannikov }; 60992cb984SSergei Barannikov class SPIRVTargetCodeGenInfo : public CommonSPIRTargetCodeGenInfo { 61992cb984SSergei Barannikov public: 62992cb984SSergei Barannikov SPIRVTargetCodeGenInfo(CodeGen::CodeGenTypes &CGT) 63992cb984SSergei Barannikov : CommonSPIRTargetCodeGenInfo(std::make_unique<SPIRVABIInfo>(CGT)) {} 64992cb984SSergei Barannikov void setCUDAKernelCallingConvention(const FunctionType *&FT) const override; 65e13cbacaSAlex Voicu LangAS getGlobalVarAddressSpace(CodeGenModule &CGM, 66e13cbacaSAlex Voicu const VarDecl *D) const override; 67*66acb269SAlex Voicu void setTargetAttributes(const Decl *D, llvm::GlobalValue *GV, 68*66acb269SAlex Voicu CodeGen::CodeGenModule &M) const override; 693cfd0c0dSAlex Voicu llvm::SyncScope::ID getLLVMSyncScopeID(const LangOptions &LangOpts, 703cfd0c0dSAlex Voicu SyncScope Scope, 713cfd0c0dSAlex Voicu llvm::AtomicOrdering Ordering, 723cfd0c0dSAlex Voicu llvm::LLVMContext &Ctx) const override; 73992cb984SSergei Barannikov }; 743cfd0c0dSAlex Voicu 753cfd0c0dSAlex Voicu inline StringRef mapClangSyncScopeToLLVM(SyncScope Scope) { 763cfd0c0dSAlex Voicu switch (Scope) { 773cfd0c0dSAlex Voicu case SyncScope::HIPSingleThread: 783cfd0c0dSAlex Voicu case SyncScope::SingleScope: 793cfd0c0dSAlex Voicu return "singlethread"; 803cfd0c0dSAlex Voicu case SyncScope::HIPWavefront: 813cfd0c0dSAlex Voicu case SyncScope::OpenCLSubGroup: 823cfd0c0dSAlex Voicu case SyncScope::WavefrontScope: 833cfd0c0dSAlex Voicu return "subgroup"; 843cfd0c0dSAlex Voicu case SyncScope::HIPWorkgroup: 853cfd0c0dSAlex Voicu case SyncScope::OpenCLWorkGroup: 863cfd0c0dSAlex Voicu case SyncScope::WorkgroupScope: 873cfd0c0dSAlex Voicu return "workgroup"; 883cfd0c0dSAlex Voicu case SyncScope::HIPAgent: 893cfd0c0dSAlex Voicu case SyncScope::OpenCLDevice: 903cfd0c0dSAlex Voicu case SyncScope::DeviceScope: 913cfd0c0dSAlex Voicu return "device"; 923cfd0c0dSAlex Voicu case SyncScope::SystemScope: 933cfd0c0dSAlex Voicu case SyncScope::HIPSystem: 943cfd0c0dSAlex Voicu case SyncScope::OpenCLAllSVMDevices: 953cfd0c0dSAlex Voicu return ""; 963cfd0c0dSAlex Voicu } 973cfd0c0dSAlex Voicu return ""; 983cfd0c0dSAlex Voicu } 99992cb984SSergei Barannikov } // End anonymous namespace. 100992cb984SSergei Barannikov 101992cb984SSergei Barannikov void CommonSPIRABIInfo::setCCs() { 102992cb984SSergei Barannikov assert(getRuntimeCC() == llvm::CallingConv::C); 103992cb984SSergei Barannikov RuntimeCC = llvm::CallingConv::SPIR_FUNC; 104992cb984SSergei Barannikov } 105992cb984SSergei Barannikov 106ad435bccSAlex Voicu ABIArgInfo SPIRVABIInfo::classifyReturnType(QualType RetTy) const { 107ad435bccSAlex Voicu if (getTarget().getTriple().getVendor() != llvm::Triple::AMD) 108ad435bccSAlex Voicu return DefaultABIInfo::classifyReturnType(RetTy); 109ad435bccSAlex Voicu if (!isAggregateTypeForABI(RetTy) || getRecordArgABI(RetTy, getCXXABI())) 110ad435bccSAlex Voicu return DefaultABIInfo::classifyReturnType(RetTy); 111ad435bccSAlex Voicu 112ad435bccSAlex Voicu if (const RecordType *RT = RetTy->getAs<RecordType>()) { 113ad435bccSAlex Voicu const RecordDecl *RD = RT->getDecl(); 114ad435bccSAlex Voicu if (RD->hasFlexibleArrayMember()) 115ad435bccSAlex Voicu return DefaultABIInfo::classifyReturnType(RetTy); 116ad435bccSAlex Voicu } 117ad435bccSAlex Voicu 118ad435bccSAlex Voicu // TODO: The AMDGPU ABI is non-trivial to represent in SPIR-V; in order to 119ad435bccSAlex Voicu // avoid encoding various architecture specific bits here we return everything 120ad435bccSAlex Voicu // as direct to retain type info for things like aggregates, for later perusal 121ad435bccSAlex Voicu // when translating back to LLVM/lowering in the BE. This is also why we 122ad435bccSAlex Voicu // disable flattening as the outcomes can mismatch between SPIR-V and AMDGPU. 123ad435bccSAlex Voicu // This will be revisited / optimised in the future. 124ad435bccSAlex Voicu return ABIArgInfo::getDirect(CGT.ConvertType(RetTy), 0u, nullptr, false); 125ad435bccSAlex Voicu } 126ad435bccSAlex Voicu 127992cb984SSergei Barannikov ABIArgInfo SPIRVABIInfo::classifyKernelArgumentType(QualType Ty) const { 128992cb984SSergei Barannikov if (getContext().getLangOpts().CUDAIsDevice) { 129992cb984SSergei Barannikov // Coerce pointer arguments with default address space to CrossWorkGroup 130992cb984SSergei Barannikov // pointers for HIPSPV/CUDASPV. When the language mode is HIP/CUDA, the 131992cb984SSergei Barannikov // SPIRTargetInfo maps cuda_device to SPIR-V's CrossWorkGroup address space. 132992cb984SSergei Barannikov llvm::Type *LTy = CGT.ConvertType(Ty); 133992cb984SSergei Barannikov auto DefaultAS = getContext().getTargetAddressSpace(LangAS::Default); 134992cb984SSergei Barannikov auto GlobalAS = getContext().getTargetAddressSpace(LangAS::cuda_device); 135992cb984SSergei Barannikov auto *PtrTy = llvm::dyn_cast<llvm::PointerType>(LTy); 136992cb984SSergei Barannikov if (PtrTy && PtrTy->getAddressSpace() == DefaultAS) { 137992cb984SSergei Barannikov LTy = llvm::PointerType::get(PtrTy->getContext(), GlobalAS); 138992cb984SSergei Barannikov return ABIArgInfo::getDirect(LTy, 0, nullptr, false); 139992cb984SSergei Barannikov } 140992cb984SSergei Barannikov 141ad435bccSAlex Voicu if (isAggregateTypeForABI(Ty)) { 142ad435bccSAlex Voicu if (getTarget().getTriple().getVendor() == llvm::Triple::AMD) 143ad435bccSAlex Voicu // TODO: The AMDGPU kernel ABI passes aggregates byref, which is not 144ad435bccSAlex Voicu // currently expressible in SPIR-V; SPIR-V passes aggregates byval, 145ad435bccSAlex Voicu // which the AMDGPU kernel ABI does not allow. Passing aggregates as 146ad435bccSAlex Voicu // direct works around this impedance mismatch, as it retains type info 147ad435bccSAlex Voicu // and can be correctly handled, post reverse-translation, by the AMDGPU 148ad435bccSAlex Voicu // BE, which has to support this CC for legacy OpenCL purposes. It can 149ad435bccSAlex Voicu // be brittle and does lead to performance degradation in certain 150ad435bccSAlex Voicu // pathological cases. This will be revisited / optimised in the future, 151ad435bccSAlex Voicu // once a way to deal with the byref/byval impedance mismatch is 152ad435bccSAlex Voicu // identified. 153ad435bccSAlex Voicu return ABIArgInfo::getDirect(LTy, 0, nullptr, false); 154992cb984SSergei Barannikov // Force copying aggregate type in kernel arguments by value when 155992cb984SSergei Barannikov // compiling CUDA targeting SPIR-V. This is required for the object 156992cb984SSergei Barannikov // copied to be valid on the device. 157992cb984SSergei Barannikov // This behavior follows the CUDA spec 158992cb984SSergei Barannikov // https://docs.nvidia.com/cuda/cuda-c-programming-guide/index.html#global-function-argument-processing, 159992cb984SSergei Barannikov // and matches the NVPTX implementation. 160992cb984SSergei Barannikov return getNaturalAlignIndirect(Ty, /* byval */ true); 161992cb984SSergei Barannikov } 162ad435bccSAlex Voicu } 163992cb984SSergei Barannikov return classifyArgumentType(Ty); 164992cb984SSergei Barannikov } 165992cb984SSergei Barannikov 166ad435bccSAlex Voicu ABIArgInfo SPIRVABIInfo::classifyArgumentType(QualType Ty) const { 167ad435bccSAlex Voicu if (getTarget().getTriple().getVendor() != llvm::Triple::AMD) 168ad435bccSAlex Voicu return DefaultABIInfo::classifyArgumentType(Ty); 169ad435bccSAlex Voicu if (!isAggregateTypeForABI(Ty)) 170ad435bccSAlex Voicu return DefaultABIInfo::classifyArgumentType(Ty); 171ad435bccSAlex Voicu 172ad435bccSAlex Voicu // Records with non-trivial destructors/copy-constructors should not be 173ad435bccSAlex Voicu // passed by value. 174ad435bccSAlex Voicu if (auto RAA = getRecordArgABI(Ty, getCXXABI())) 175ad435bccSAlex Voicu return getNaturalAlignIndirect(Ty, RAA == CGCXXABI::RAA_DirectInMemory); 176ad435bccSAlex Voicu 177ad435bccSAlex Voicu if (const RecordType *RT = Ty->getAs<RecordType>()) { 178ad435bccSAlex Voicu const RecordDecl *RD = RT->getDecl(); 179ad435bccSAlex Voicu if (RD->hasFlexibleArrayMember()) 180ad435bccSAlex Voicu return DefaultABIInfo::classifyArgumentType(Ty); 181ad435bccSAlex Voicu } 182ad435bccSAlex Voicu 183ad435bccSAlex Voicu return ABIArgInfo::getDirect(CGT.ConvertType(Ty), 0u, nullptr, false); 184ad435bccSAlex Voicu } 185ad435bccSAlex Voicu 186992cb984SSergei Barannikov void SPIRVABIInfo::computeInfo(CGFunctionInfo &FI) const { 187992cb984SSergei Barannikov // The logic is same as in DefaultABIInfo with an exception on the kernel 188992cb984SSergei Barannikov // arguments handling. 189992cb984SSergei Barannikov llvm::CallingConv::ID CC = FI.getCallingConvention(); 190992cb984SSergei Barannikov 191992cb984SSergei Barannikov if (!getCXXABI().classifyReturnType(FI)) 192992cb984SSergei Barannikov FI.getReturnInfo() = classifyReturnType(FI.getReturnType()); 193992cb984SSergei Barannikov 194992cb984SSergei Barannikov for (auto &I : FI.arguments()) { 195992cb984SSergei Barannikov if (CC == llvm::CallingConv::SPIR_KERNEL) { 196992cb984SSergei Barannikov I.info = classifyKernelArgumentType(I.type); 197992cb984SSergei Barannikov } else { 198992cb984SSergei Barannikov I.info = classifyArgumentType(I.type); 199992cb984SSergei Barannikov } 200992cb984SSergei Barannikov } 201992cb984SSergei Barannikov } 202992cb984SSergei Barannikov 203992cb984SSergei Barannikov namespace clang { 204992cb984SSergei Barannikov namespace CodeGen { 205992cb984SSergei Barannikov void computeSPIRKernelABIInfo(CodeGenModule &CGM, CGFunctionInfo &FI) { 206992cb984SSergei Barannikov if (CGM.getTarget().getTriple().isSPIRV()) 207992cb984SSergei Barannikov SPIRVABIInfo(CGM.getTypes()).computeInfo(FI); 208992cb984SSergei Barannikov else 209992cb984SSergei Barannikov CommonSPIRABIInfo(CGM.getTypes()).computeInfo(FI); 210992cb984SSergei Barannikov } 211992cb984SSergei Barannikov } 212992cb984SSergei Barannikov } 213992cb984SSergei Barannikov 214992cb984SSergei Barannikov unsigned CommonSPIRTargetCodeGenInfo::getOpenCLKernelCallingConv() const { 215992cb984SSergei Barannikov return llvm::CallingConv::SPIR_KERNEL; 216992cb984SSergei Barannikov } 217992cb984SSergei Barannikov 218992cb984SSergei Barannikov void SPIRVTargetCodeGenInfo::setCUDAKernelCallingConvention( 219992cb984SSergei Barannikov const FunctionType *&FT) const { 220992cb984SSergei Barannikov // Convert HIP kernels to SPIR-V kernels. 221992cb984SSergei Barannikov if (getABIInfo().getContext().getLangOpts().HIP) { 222992cb984SSergei Barannikov FT = getABIInfo().getContext().adjustFunctionType( 223992cb984SSergei Barannikov FT, FT->getExtInfo().withCallingConv(CC_OpenCLKernel)); 224992cb984SSergei Barannikov return; 225992cb984SSergei Barannikov } 226992cb984SSergei Barannikov } 227992cb984SSergei Barannikov 228e13cbacaSAlex Voicu LangAS 229e13cbacaSAlex Voicu SPIRVTargetCodeGenInfo::getGlobalVarAddressSpace(CodeGenModule &CGM, 230e13cbacaSAlex Voicu const VarDecl *D) const { 231e13cbacaSAlex Voicu assert(!CGM.getLangOpts().OpenCL && 232e13cbacaSAlex Voicu !(CGM.getLangOpts().CUDA && CGM.getLangOpts().CUDAIsDevice) && 233e13cbacaSAlex Voicu "Address space agnostic languages only"); 234e13cbacaSAlex Voicu // If we're here it means that we're using the SPIRDefIsGen ASMap, hence for 235e13cbacaSAlex Voicu // the global AS we can rely on either cuda_device or sycl_global to be 236e13cbacaSAlex Voicu // correct; however, since this is not a CUDA Device context, we use 237e13cbacaSAlex Voicu // sycl_global to prevent confusion with the assertion. 238e13cbacaSAlex Voicu LangAS DefaultGlobalAS = getLangASFromTargetAS( 239e13cbacaSAlex Voicu CGM.getContext().getTargetAddressSpace(LangAS::sycl_global)); 240e13cbacaSAlex Voicu if (!D) 241e13cbacaSAlex Voicu return DefaultGlobalAS; 242e13cbacaSAlex Voicu 243e13cbacaSAlex Voicu LangAS AddrSpace = D->getType().getAddressSpace(); 244e13cbacaSAlex Voicu if (AddrSpace != LangAS::Default) 245e13cbacaSAlex Voicu return AddrSpace; 246e13cbacaSAlex Voicu 247e13cbacaSAlex Voicu return DefaultGlobalAS; 248e13cbacaSAlex Voicu } 249e13cbacaSAlex Voicu 250*66acb269SAlex Voicu void SPIRVTargetCodeGenInfo::setTargetAttributes( 251*66acb269SAlex Voicu const Decl *D, llvm::GlobalValue *GV, CodeGen::CodeGenModule &M) const { 252*66acb269SAlex Voicu if (!M.getLangOpts().HIP || 253*66acb269SAlex Voicu M.getTarget().getTriple().getVendor() != llvm::Triple::AMD) 254*66acb269SAlex Voicu return; 255*66acb269SAlex Voicu if (GV->isDeclaration()) 256*66acb269SAlex Voicu return; 257*66acb269SAlex Voicu 258*66acb269SAlex Voicu auto F = dyn_cast<llvm::Function>(GV); 259*66acb269SAlex Voicu if (!F) 260*66acb269SAlex Voicu return; 261*66acb269SAlex Voicu 262*66acb269SAlex Voicu auto FD = dyn_cast_or_null<FunctionDecl>(D); 263*66acb269SAlex Voicu if (!FD) 264*66acb269SAlex Voicu return; 265*66acb269SAlex Voicu if (!FD->hasAttr<CUDAGlobalAttr>()) 266*66acb269SAlex Voicu return; 267*66acb269SAlex Voicu 268*66acb269SAlex Voicu unsigned N = M.getLangOpts().GPUMaxThreadsPerBlock; 269*66acb269SAlex Voicu if (auto FlatWGS = FD->getAttr<AMDGPUFlatWorkGroupSizeAttr>()) 270*66acb269SAlex Voicu N = FlatWGS->getMax()->EvaluateKnownConstInt(M.getContext()).getExtValue(); 271*66acb269SAlex Voicu 272*66acb269SAlex Voicu // We encode the maximum flat WG size in the first component of the 3D 273*66acb269SAlex Voicu // max_work_group_size attribute, which will get reverse translated into the 274*66acb269SAlex Voicu // original AMDGPU attribute when targeting AMDGPU. 275*66acb269SAlex Voicu auto Int32Ty = llvm::IntegerType::getInt32Ty(M.getLLVMContext()); 276*66acb269SAlex Voicu llvm::Metadata *AttrMDArgs[] = { 277*66acb269SAlex Voicu llvm::ConstantAsMetadata::get(llvm::ConstantInt::get(Int32Ty, N)), 278*66acb269SAlex Voicu llvm::ConstantAsMetadata::get(llvm::ConstantInt::get(Int32Ty, 1)), 279*66acb269SAlex Voicu llvm::ConstantAsMetadata::get(llvm::ConstantInt::get(Int32Ty, 1))}; 280*66acb269SAlex Voicu 281*66acb269SAlex Voicu F->setMetadata("max_work_group_size", 282*66acb269SAlex Voicu llvm::MDNode::get(M.getLLVMContext(), AttrMDArgs)); 283*66acb269SAlex Voicu } 284*66acb269SAlex Voicu 2853cfd0c0dSAlex Voicu llvm::SyncScope::ID 2863cfd0c0dSAlex Voicu SPIRVTargetCodeGenInfo::getLLVMSyncScopeID(const LangOptions &, SyncScope Scope, 2873cfd0c0dSAlex Voicu llvm::AtomicOrdering, 2883cfd0c0dSAlex Voicu llvm::LLVMContext &Ctx) const { 2893cfd0c0dSAlex Voicu return Ctx.getOrInsertSyncScopeID(mapClangSyncScopeToLLVM(Scope)); 2903cfd0c0dSAlex Voicu } 2913cfd0c0dSAlex Voicu 292992cb984SSergei Barannikov /// Construct a SPIR-V target extension type for the given OpenCL image type. 293992cb984SSergei Barannikov static llvm::Type *getSPIRVImageType(llvm::LLVMContext &Ctx, StringRef BaseType, 294992cb984SSergei Barannikov StringRef OpenCLName, 295992cb984SSergei Barannikov unsigned AccessQualifier) { 296992cb984SSergei Barannikov // These parameters compare to the operands of OpTypeImage (see 297992cb984SSergei Barannikov // https://registry.khronos.org/SPIR-V/specs/unified1/SPIRV.html#OpTypeImage 298992cb984SSergei Barannikov // for more details). The first 6 integer parameters all default to 0, and 299992cb984SSergei Barannikov // will be changed to 1 only for the image type(s) that set the parameter to 300992cb984SSergei Barannikov // one. The 7th integer parameter is the access qualifier, which is tacked on 301992cb984SSergei Barannikov // at the end. 302992cb984SSergei Barannikov SmallVector<unsigned, 7> IntParams = {0, 0, 0, 0, 0, 0}; 303992cb984SSergei Barannikov 304992cb984SSergei Barannikov // Choose the dimension of the image--this corresponds to the Dim enum in 305992cb984SSergei Barannikov // SPIR-V (first integer parameter of OpTypeImage). 306f3dcc235SKazu Hirata if (OpenCLName.starts_with("image2d")) 307992cb984SSergei Barannikov IntParams[0] = 1; // 1D 308f3dcc235SKazu Hirata else if (OpenCLName.starts_with("image3d")) 309992cb984SSergei Barannikov IntParams[0] = 2; // 2D 310992cb984SSergei Barannikov else if (OpenCLName == "image1d_buffer") 311992cb984SSergei Barannikov IntParams[0] = 5; // Buffer 312992cb984SSergei Barannikov else 313f3dcc235SKazu Hirata assert(OpenCLName.starts_with("image1d") && "Unknown image type"); 314992cb984SSergei Barannikov 315992cb984SSergei Barannikov // Set the other integer parameters of OpTypeImage if necessary. Note that the 316992cb984SSergei Barannikov // OpenCL image types don't provide any information for the Sampled or 317992cb984SSergei Barannikov // Image Format parameters. 318992cb984SSergei Barannikov if (OpenCLName.contains("_depth")) 319992cb984SSergei Barannikov IntParams[1] = 1; 320992cb984SSergei Barannikov if (OpenCLName.contains("_array")) 321992cb984SSergei Barannikov IntParams[2] = 1; 322992cb984SSergei Barannikov if (OpenCLName.contains("_msaa")) 323992cb984SSergei Barannikov IntParams[3] = 1; 324992cb984SSergei Barannikov 325992cb984SSergei Barannikov // Access qualifier 326992cb984SSergei Barannikov IntParams.push_back(AccessQualifier); 327992cb984SSergei Barannikov 328992cb984SSergei Barannikov return llvm::TargetExtType::get(Ctx, BaseType, {llvm::Type::getVoidTy(Ctx)}, 329992cb984SSergei Barannikov IntParams); 330992cb984SSergei Barannikov } 331992cb984SSergei Barannikov 332992cb984SSergei Barannikov llvm::Type *CommonSPIRTargetCodeGenInfo::getOpenCLType(CodeGenModule &CGM, 333992cb984SSergei Barannikov const Type *Ty) const { 334992cb984SSergei Barannikov llvm::LLVMContext &Ctx = CGM.getLLVMContext(); 335992cb984SSergei Barannikov if (auto *PipeTy = dyn_cast<PipeType>(Ty)) 336992cb984SSergei Barannikov return llvm::TargetExtType::get(Ctx, "spirv.Pipe", {}, 337992cb984SSergei Barannikov {!PipeTy->isReadOnly()}); 338992cb984SSergei Barannikov if (auto *BuiltinTy = dyn_cast<BuiltinType>(Ty)) { 339992cb984SSergei Barannikov enum AccessQualifier : unsigned { AQ_ro = 0, AQ_wo = 1, AQ_rw = 2 }; 340992cb984SSergei Barannikov switch (BuiltinTy->getKind()) { 341992cb984SSergei Barannikov #define IMAGE_TYPE(ImgType, Id, SingletonId, Access, Suffix) \ 342992cb984SSergei Barannikov case BuiltinType::Id: \ 343992cb984SSergei Barannikov return getSPIRVImageType(Ctx, "spirv.Image", #ImgType, AQ_##Suffix); 344992cb984SSergei Barannikov #include "clang/Basic/OpenCLImageTypes.def" 345992cb984SSergei Barannikov case BuiltinType::OCLSampler: 346992cb984SSergei Barannikov return llvm::TargetExtType::get(Ctx, "spirv.Sampler"); 347992cb984SSergei Barannikov case BuiltinType::OCLEvent: 348992cb984SSergei Barannikov return llvm::TargetExtType::get(Ctx, "spirv.Event"); 349992cb984SSergei Barannikov case BuiltinType::OCLClkEvent: 350992cb984SSergei Barannikov return llvm::TargetExtType::get(Ctx, "spirv.DeviceEvent"); 351992cb984SSergei Barannikov case BuiltinType::OCLQueue: 352992cb984SSergei Barannikov return llvm::TargetExtType::get(Ctx, "spirv.Queue"); 353992cb984SSergei Barannikov case BuiltinType::OCLReserveID: 354992cb984SSergei Barannikov return llvm::TargetExtType::get(Ctx, "spirv.ReserveId"); 355992cb984SSergei Barannikov #define INTEL_SUBGROUP_AVC_TYPE(Name, Id) \ 356992cb984SSergei Barannikov case BuiltinType::OCLIntelSubgroupAVC##Id: \ 357992cb984SSergei Barannikov return llvm::TargetExtType::get(Ctx, "spirv.Avc" #Id "INTEL"); 358992cb984SSergei Barannikov #include "clang/Basic/OpenCLExtensionTypes.def" 359992cb984SSergei Barannikov default: 360992cb984SSergei Barannikov return nullptr; 361992cb984SSergei Barannikov } 362992cb984SSergei Barannikov } 363992cb984SSergei Barannikov 364992cb984SSergei Barannikov return nullptr; 365992cb984SSergei Barannikov } 366992cb984SSergei Barannikov 367d6344c1cSSteven Perron llvm::Type *CommonSPIRTargetCodeGenInfo::getHLSLType(CodeGenModule &CGM, 368d6344c1cSSteven Perron const Type *Ty) const { 369d6344c1cSSteven Perron auto *ResType = dyn_cast<HLSLAttributedResourceType>(Ty); 370d6344c1cSSteven Perron if (!ResType) 371d6344c1cSSteven Perron return nullptr; 372d6344c1cSSteven Perron 373d6344c1cSSteven Perron llvm::LLVMContext &Ctx = CGM.getLLVMContext(); 374d6344c1cSSteven Perron const HLSLAttributedResourceType::Attributes &ResAttrs = ResType->getAttrs(); 375d6344c1cSSteven Perron switch (ResAttrs.ResourceClass) { 376d6344c1cSSteven Perron case llvm::dxil::ResourceClass::UAV: 377d6344c1cSSteven Perron case llvm::dxil::ResourceClass::SRV: { 378d6344c1cSSteven Perron // TypedBuffer and RawBuffer both need element type 379d6344c1cSSteven Perron QualType ContainedTy = ResType->getContainedType(); 380d6344c1cSSteven Perron if (ContainedTy.isNull()) 381d6344c1cSSteven Perron return nullptr; 382d6344c1cSSteven Perron 383d6344c1cSSteven Perron assert(!ResAttrs.RawBuffer && 384d6344c1cSSteven Perron "Raw buffers handles are not implemented for SPIR-V yet"); 385d6344c1cSSteven Perron assert(!ResAttrs.IsROV && 386d6344c1cSSteven Perron "Rasterizer order views not implemented for SPIR-V yet"); 387d6344c1cSSteven Perron 388d6344c1cSSteven Perron // convert element type 389d6344c1cSSteven Perron llvm::Type *ElemType = CGM.getTypes().ConvertType(ContainedTy); 390d6344c1cSSteven Perron return getSPIRVImageTypeFromHLSLResource(ResAttrs, ElemType, Ctx); 391d6344c1cSSteven Perron } 392d6344c1cSSteven Perron case llvm::dxil::ResourceClass::CBuffer: 393d6344c1cSSteven Perron llvm_unreachable("CBuffer handles are not implemented for SPIR-V yet"); 394d6344c1cSSteven Perron break; 395d6344c1cSSteven Perron case llvm::dxil::ResourceClass::Sampler: 396d6344c1cSSteven Perron return llvm::TargetExtType::get(Ctx, "spirv.Sampler"); 397d6344c1cSSteven Perron } 398d6344c1cSSteven Perron return nullptr; 399d6344c1cSSteven Perron } 400d6344c1cSSteven Perron 401d6344c1cSSteven Perron llvm::Type *CommonSPIRTargetCodeGenInfo::getSPIRVImageTypeFromHLSLResource( 402d6344c1cSSteven Perron const HLSLAttributedResourceType::Attributes &attributes, 403d6344c1cSSteven Perron llvm::Type *ElementType, llvm::LLVMContext &Ctx) const { 404d6344c1cSSteven Perron 405d6344c1cSSteven Perron if (ElementType->isVectorTy()) 406d6344c1cSSteven Perron ElementType = ElementType->getScalarType(); 407d6344c1cSSteven Perron 408d6344c1cSSteven Perron assert((ElementType->isIntegerTy() || ElementType->isFloatingPointTy()) && 409d6344c1cSSteven Perron "The element type for a SPIR-V resource must be a scalar integer or " 410d6344c1cSSteven Perron "floating point type."); 411d6344c1cSSteven Perron 412d6344c1cSSteven Perron // These parameters correspond to the operands to the OpTypeImage SPIR-V 413d6344c1cSSteven Perron // instruction. See 414d6344c1cSSteven Perron // https://registry.khronos.org/SPIR-V/specs/unified1/SPIRV.html#OpTypeImage. 415d6344c1cSSteven Perron SmallVector<unsigned, 6> IntParams(6, 0); 416d6344c1cSSteven Perron 417d6344c1cSSteven Perron // Dim 418d6344c1cSSteven Perron // For now we assume everything is a buffer. 419d6344c1cSSteven Perron IntParams[0] = 5; 420d6344c1cSSteven Perron 421d6344c1cSSteven Perron // Depth 422d6344c1cSSteven Perron // HLSL does not indicate if it is a depth texture or not, so we use unknown. 423d6344c1cSSteven Perron IntParams[1] = 2; 424d6344c1cSSteven Perron 425d6344c1cSSteven Perron // Arrayed 426d6344c1cSSteven Perron IntParams[2] = 0; 427d6344c1cSSteven Perron 428d6344c1cSSteven Perron // MS 429d6344c1cSSteven Perron IntParams[3] = 0; 430d6344c1cSSteven Perron 431d6344c1cSSteven Perron // Sampled 432d6344c1cSSteven Perron IntParams[4] = 433d6344c1cSSteven Perron attributes.ResourceClass == llvm::dxil::ResourceClass::UAV ? 2 : 1; 434d6344c1cSSteven Perron 435d6344c1cSSteven Perron // Image format. 436d6344c1cSSteven Perron // Setting to unknown for now. 437d6344c1cSSteven Perron IntParams[5] = 0; 438d6344c1cSSteven Perron 439d6344c1cSSteven Perron return llvm::TargetExtType::get(Ctx, "spirv.Image", {ElementType}, IntParams); 440d6344c1cSSteven Perron } 441d6344c1cSSteven Perron 442992cb984SSergei Barannikov std::unique_ptr<TargetCodeGenInfo> 443992cb984SSergei Barannikov CodeGen::createCommonSPIRTargetCodeGenInfo(CodeGenModule &CGM) { 444992cb984SSergei Barannikov return std::make_unique<CommonSPIRTargetCodeGenInfo>(CGM.getTypes()); 445992cb984SSergei Barannikov } 446992cb984SSergei Barannikov 447992cb984SSergei Barannikov std::unique_ptr<TargetCodeGenInfo> 448992cb984SSergei Barannikov CodeGen::createSPIRVTargetCodeGenInfo(CodeGenModule &CGM) { 449992cb984SSergei Barannikov return std::make_unique<SPIRVTargetCodeGenInfo>(CGM.getTypes()); 450992cb984SSergei Barannikov } 451