xref: /llvm-project/clang/lib/CodeGen/Targets/SPIR.cpp (revision 66acb2694655321b37a1ee3ff19207a111756562)
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