xref: /aosp_15_r20/external/mesa3d/src/compiler/clc/clc_helpers.cpp (revision 6104692788411f58d303aa86923a9ff6ecaded22)
1*61046927SAndroid Build Coastguard Worker //
2*61046927SAndroid Build Coastguard Worker // Copyright 2012-2016 Francisco Jerez
3*61046927SAndroid Build Coastguard Worker // Copyright 2012-2016 Advanced Micro Devices, Inc.
4*61046927SAndroid Build Coastguard Worker // Copyright 2014-2016 Jan Vesely
5*61046927SAndroid Build Coastguard Worker // Copyright 2014-2015 Serge Martin
6*61046927SAndroid Build Coastguard Worker // Copyright 2015 Zoltan Gilian
7*61046927SAndroid Build Coastguard Worker //
8*61046927SAndroid Build Coastguard Worker // Permission is hereby granted, free of charge, to any person obtaining a
9*61046927SAndroid Build Coastguard Worker // copy of this software and associated documentation files (the "Software"),
10*61046927SAndroid Build Coastguard Worker // to deal in the Software without restriction, including without limitation
11*61046927SAndroid Build Coastguard Worker // the rights to use, copy, modify, merge, publish, distribute, sublicense,
12*61046927SAndroid Build Coastguard Worker // and/or sell copies of the Software, and to permit persons to whom the
13*61046927SAndroid Build Coastguard Worker // Software is furnished to do so, subject to the following conditions:
14*61046927SAndroid Build Coastguard Worker //
15*61046927SAndroid Build Coastguard Worker // The above copyright notice and this permission notice shall be included in
16*61046927SAndroid Build Coastguard Worker // all copies or substantial portions of the Software.
17*61046927SAndroid Build Coastguard Worker //
18*61046927SAndroid Build Coastguard Worker // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19*61046927SAndroid Build Coastguard Worker // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20*61046927SAndroid Build Coastguard Worker // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
21*61046927SAndroid Build Coastguard Worker // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
22*61046927SAndroid Build Coastguard Worker // OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
23*61046927SAndroid Build Coastguard Worker // ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
24*61046927SAndroid Build Coastguard Worker // OTHER DEALINGS IN THE SOFTWARE.
25*61046927SAndroid Build Coastguard Worker 
26*61046927SAndroid Build Coastguard Worker #include <cstdlib>
27*61046927SAndroid Build Coastguard Worker #include <filesystem>
28*61046927SAndroid Build Coastguard Worker #include <sstream>
29*61046927SAndroid Build Coastguard Worker #include <mutex>
30*61046927SAndroid Build Coastguard Worker 
31*61046927SAndroid Build Coastguard Worker #include <llvm/ADT/ArrayRef.h>
32*61046927SAndroid Build Coastguard Worker #include <llvm/IR/DiagnosticPrinter.h>
33*61046927SAndroid Build Coastguard Worker #include <llvm/IR/DiagnosticInfo.h>
34*61046927SAndroid Build Coastguard Worker #include <llvm/IR/LegacyPassManager.h>
35*61046927SAndroid Build Coastguard Worker #include <llvm/IR/LLVMContext.h>
36*61046927SAndroid Build Coastguard Worker #include <llvm/IR/Type.h>
37*61046927SAndroid Build Coastguard Worker #include <llvm/MC/TargetRegistry.h>
38*61046927SAndroid Build Coastguard Worker #include <llvm/Target/TargetMachine.h>
39*61046927SAndroid Build Coastguard Worker #include <llvm/Support/raw_ostream.h>
40*61046927SAndroid Build Coastguard Worker #include <llvm/Bitcode/BitcodeWriter.h>
41*61046927SAndroid Build Coastguard Worker #include <llvm/Bitcode/BitcodeReader.h>
42*61046927SAndroid Build Coastguard Worker #include <llvm-c/Core.h>
43*61046927SAndroid Build Coastguard Worker #include <llvm-c/Target.h>
44*61046927SAndroid Build Coastguard Worker #include <LLVMSPIRVLib/LLVMSPIRVLib.h>
45*61046927SAndroid Build Coastguard Worker 
46*61046927SAndroid Build Coastguard Worker #include <clang/Config/config.h>
47*61046927SAndroid Build Coastguard Worker #include <clang/Driver/Driver.h>
48*61046927SAndroid Build Coastguard Worker #include <clang/CodeGen/CodeGenAction.h>
49*61046927SAndroid Build Coastguard Worker #include <clang/Lex/PreprocessorOptions.h>
50*61046927SAndroid Build Coastguard Worker #include <clang/Frontend/CompilerInstance.h>
51*61046927SAndroid Build Coastguard Worker #include <clang/Frontend/TextDiagnosticBuffer.h>
52*61046927SAndroid Build Coastguard Worker #include <clang/Frontend/TextDiagnosticPrinter.h>
53*61046927SAndroid Build Coastguard Worker #include <clang/Basic/TargetInfo.h>
54*61046927SAndroid Build Coastguard Worker 
55*61046927SAndroid Build Coastguard Worker #include <spirv-tools/libspirv.hpp>
56*61046927SAndroid Build Coastguard Worker #include <spirv-tools/linker.hpp>
57*61046927SAndroid Build Coastguard Worker #include <spirv-tools/optimizer.hpp>
58*61046927SAndroid Build Coastguard Worker 
59*61046927SAndroid Build Coastguard Worker #include "util/macros.h"
60*61046927SAndroid Build Coastguard Worker #include "glsl_types.h"
61*61046927SAndroid Build Coastguard Worker 
62*61046927SAndroid Build Coastguard Worker #include "spirv.h"
63*61046927SAndroid Build Coastguard Worker 
64*61046927SAndroid Build Coastguard Worker #if DETECT_OS_POSIX
65*61046927SAndroid Build Coastguard Worker #include <dlfcn.h>
66*61046927SAndroid Build Coastguard Worker #endif
67*61046927SAndroid Build Coastguard Worker 
68*61046927SAndroid Build Coastguard Worker #ifdef USE_STATIC_OPENCL_C_H
69*61046927SAndroid Build Coastguard Worker #include "opencl-c-base.h.h"
70*61046927SAndroid Build Coastguard Worker #include "opencl-c.h.h"
71*61046927SAndroid Build Coastguard Worker #endif
72*61046927SAndroid Build Coastguard Worker 
73*61046927SAndroid Build Coastguard Worker #include "clc_helpers.h"
74*61046927SAndroid Build Coastguard Worker 
75*61046927SAndroid Build Coastguard Worker namespace fs = std::filesystem;
76*61046927SAndroid Build Coastguard Worker 
77*61046927SAndroid Build Coastguard Worker /* Use the highest version of SPIRV supported by SPIRV-Tools. */
78*61046927SAndroid Build Coastguard Worker constexpr spv_target_env spirv_target = SPV_ENV_UNIVERSAL_1_5;
79*61046927SAndroid Build Coastguard Worker 
80*61046927SAndroid Build Coastguard Worker constexpr SPIRV::VersionNumber invalid_spirv_trans_version = static_cast<SPIRV::VersionNumber>(0);
81*61046927SAndroid Build Coastguard Worker 
82*61046927SAndroid Build Coastguard Worker using ::llvm::Function;
83*61046927SAndroid Build Coastguard Worker using ::llvm::legacy::PassManager;
84*61046927SAndroid Build Coastguard Worker using ::llvm::LLVMContext;
85*61046927SAndroid Build Coastguard Worker using ::llvm::Module;
86*61046927SAndroid Build Coastguard Worker using ::llvm::raw_string_ostream;
87*61046927SAndroid Build Coastguard Worker using ::llvm::TargetRegistry;
88*61046927SAndroid Build Coastguard Worker using ::clang::driver::Driver;
89*61046927SAndroid Build Coastguard Worker 
90*61046927SAndroid Build Coastguard Worker static void
91*61046927SAndroid Build Coastguard Worker clc_dump_llvm(const llvm::Module *mod, FILE *f);
92*61046927SAndroid Build Coastguard Worker 
93*61046927SAndroid Build Coastguard Worker static void
94*61046927SAndroid Build Coastguard Worker #if LLVM_VERSION_MAJOR >= 19
llvm_log_handler(const::llvm::DiagnosticInfo * di,void * data)95*61046927SAndroid Build Coastguard Worker llvm_log_handler(const ::llvm::DiagnosticInfo *di, void *data) {
96*61046927SAndroid Build Coastguard Worker #else
97*61046927SAndroid Build Coastguard Worker llvm_log_handler(const ::llvm::DiagnosticInfo &di, void *data) {
98*61046927SAndroid Build Coastguard Worker #endif
99*61046927SAndroid Build Coastguard Worker    const clc_logger *logger = static_cast<clc_logger *>(data);
100*61046927SAndroid Build Coastguard Worker 
101*61046927SAndroid Build Coastguard Worker    std::string log;
102*61046927SAndroid Build Coastguard Worker    raw_string_ostream os { log };
103*61046927SAndroid Build Coastguard Worker    ::llvm::DiagnosticPrinterRawOStream printer { os };
104*61046927SAndroid Build Coastguard Worker #if LLVM_VERSION_MAJOR >= 19
105*61046927SAndroid Build Coastguard Worker    di->print(printer);
106*61046927SAndroid Build Coastguard Worker #else
107*61046927SAndroid Build Coastguard Worker    di.print(printer);
108*61046927SAndroid Build Coastguard Worker #endif
109*61046927SAndroid Build Coastguard Worker 
110*61046927SAndroid Build Coastguard Worker    clc_error(logger, "%s", log.c_str());
111*61046927SAndroid Build Coastguard Worker }
112*61046927SAndroid Build Coastguard Worker 
113*61046927SAndroid Build Coastguard Worker class SPIRVKernelArg {
114*61046927SAndroid Build Coastguard Worker public:
115*61046927SAndroid Build Coastguard Worker    SPIRVKernelArg(uint32_t id, uint32_t typeId) : id(id), typeId(typeId),
116*61046927SAndroid Build Coastguard Worker                                                   addrQualifier(CLC_KERNEL_ARG_ADDRESS_PRIVATE),
117*61046927SAndroid Build Coastguard Worker                                                   accessQualifier(0),
118*61046927SAndroid Build Coastguard Worker                                                   typeQualifier(0) { }
119*61046927SAndroid Build Coastguard Worker    ~SPIRVKernelArg() { }
120*61046927SAndroid Build Coastguard Worker 
121*61046927SAndroid Build Coastguard Worker    uint32_t id;
122*61046927SAndroid Build Coastguard Worker    uint32_t typeId;
123*61046927SAndroid Build Coastguard Worker    std::string name;
124*61046927SAndroid Build Coastguard Worker    std::string typeName;
125*61046927SAndroid Build Coastguard Worker    enum clc_kernel_arg_address_qualifier addrQualifier;
126*61046927SAndroid Build Coastguard Worker    unsigned accessQualifier;
127*61046927SAndroid Build Coastguard Worker    unsigned typeQualifier;
128*61046927SAndroid Build Coastguard Worker };
129*61046927SAndroid Build Coastguard Worker 
130*61046927SAndroid Build Coastguard Worker class SPIRVKernelInfo {
131*61046927SAndroid Build Coastguard Worker public:
132*61046927SAndroid Build Coastguard Worker    SPIRVKernelInfo(uint32_t fid, const char *nm)
133*61046927SAndroid Build Coastguard Worker       : funcId(fid), name(nm), vecHint(0), localSize(), localSizeHint() { }
134*61046927SAndroid Build Coastguard Worker    ~SPIRVKernelInfo() { }
135*61046927SAndroid Build Coastguard Worker 
136*61046927SAndroid Build Coastguard Worker    uint32_t funcId;
137*61046927SAndroid Build Coastguard Worker    std::string name;
138*61046927SAndroid Build Coastguard Worker    std::vector<SPIRVKernelArg> args;
139*61046927SAndroid Build Coastguard Worker    unsigned vecHint;
140*61046927SAndroid Build Coastguard Worker    unsigned localSize[3];
141*61046927SAndroid Build Coastguard Worker    unsigned localSizeHint[3];
142*61046927SAndroid Build Coastguard Worker };
143*61046927SAndroid Build Coastguard Worker 
144*61046927SAndroid Build Coastguard Worker class SPIRVKernelParser {
145*61046927SAndroid Build Coastguard Worker public:
146*61046927SAndroid Build Coastguard Worker    SPIRVKernelParser() : curKernel(NULL)
147*61046927SAndroid Build Coastguard Worker    {
148*61046927SAndroid Build Coastguard Worker       ctx = spvContextCreate(spirv_target);
149*61046927SAndroid Build Coastguard Worker    }
150*61046927SAndroid Build Coastguard Worker 
151*61046927SAndroid Build Coastguard Worker    ~SPIRVKernelParser()
152*61046927SAndroid Build Coastguard Worker    {
153*61046927SAndroid Build Coastguard Worker      spvContextDestroy(ctx);
154*61046927SAndroid Build Coastguard Worker    }
155*61046927SAndroid Build Coastguard Worker 
156*61046927SAndroid Build Coastguard Worker    void parseEntryPoint(const spv_parsed_instruction_t *ins)
157*61046927SAndroid Build Coastguard Worker    {
158*61046927SAndroid Build Coastguard Worker       assert(ins->num_operands >= 3);
159*61046927SAndroid Build Coastguard Worker 
160*61046927SAndroid Build Coastguard Worker       const spv_parsed_operand_t *op = &ins->operands[1];
161*61046927SAndroid Build Coastguard Worker 
162*61046927SAndroid Build Coastguard Worker       assert(op->type == SPV_OPERAND_TYPE_ID);
163*61046927SAndroid Build Coastguard Worker 
164*61046927SAndroid Build Coastguard Worker       uint32_t funcId = ins->words[op->offset];
165*61046927SAndroid Build Coastguard Worker 
166*61046927SAndroid Build Coastguard Worker       for (auto &iter : kernels) {
167*61046927SAndroid Build Coastguard Worker          if (funcId == iter.funcId)
168*61046927SAndroid Build Coastguard Worker             return;
169*61046927SAndroid Build Coastguard Worker       }
170*61046927SAndroid Build Coastguard Worker 
171*61046927SAndroid Build Coastguard Worker       op = &ins->operands[2];
172*61046927SAndroid Build Coastguard Worker       assert(op->type == SPV_OPERAND_TYPE_LITERAL_STRING);
173*61046927SAndroid Build Coastguard Worker       const char *name = reinterpret_cast<const char *>(ins->words + op->offset);
174*61046927SAndroid Build Coastguard Worker 
175*61046927SAndroid Build Coastguard Worker       kernels.push_back(SPIRVKernelInfo(funcId, name));
176*61046927SAndroid Build Coastguard Worker    }
177*61046927SAndroid Build Coastguard Worker 
178*61046927SAndroid Build Coastguard Worker    void parseFunction(const spv_parsed_instruction_t *ins)
179*61046927SAndroid Build Coastguard Worker    {
180*61046927SAndroid Build Coastguard Worker       assert(ins->num_operands == 4);
181*61046927SAndroid Build Coastguard Worker 
182*61046927SAndroid Build Coastguard Worker       const spv_parsed_operand_t *op = &ins->operands[1];
183*61046927SAndroid Build Coastguard Worker 
184*61046927SAndroid Build Coastguard Worker       assert(op->type == SPV_OPERAND_TYPE_RESULT_ID);
185*61046927SAndroid Build Coastguard Worker 
186*61046927SAndroid Build Coastguard Worker       uint32_t funcId = ins->words[op->offset];
187*61046927SAndroid Build Coastguard Worker 
188*61046927SAndroid Build Coastguard Worker       for (auto &kernel : kernels) {
189*61046927SAndroid Build Coastguard Worker          if (funcId == kernel.funcId && !kernel.args.size()) {
190*61046927SAndroid Build Coastguard Worker             curKernel = &kernel;
191*61046927SAndroid Build Coastguard Worker 	    return;
192*61046927SAndroid Build Coastguard Worker          }
193*61046927SAndroid Build Coastguard Worker       }
194*61046927SAndroid Build Coastguard Worker    }
195*61046927SAndroid Build Coastguard Worker 
196*61046927SAndroid Build Coastguard Worker    void parseFunctionParam(const spv_parsed_instruction_t *ins)
197*61046927SAndroid Build Coastguard Worker    {
198*61046927SAndroid Build Coastguard Worker       const spv_parsed_operand_t *op;
199*61046927SAndroid Build Coastguard Worker       uint32_t id, typeId;
200*61046927SAndroid Build Coastguard Worker 
201*61046927SAndroid Build Coastguard Worker       if (!curKernel)
202*61046927SAndroid Build Coastguard Worker          return;
203*61046927SAndroid Build Coastguard Worker 
204*61046927SAndroid Build Coastguard Worker       assert(ins->num_operands == 2);
205*61046927SAndroid Build Coastguard Worker       op = &ins->operands[0];
206*61046927SAndroid Build Coastguard Worker       assert(op->type == SPV_OPERAND_TYPE_TYPE_ID);
207*61046927SAndroid Build Coastguard Worker       typeId = ins->words[op->offset];
208*61046927SAndroid Build Coastguard Worker       op = &ins->operands[1];
209*61046927SAndroid Build Coastguard Worker       assert(op->type == SPV_OPERAND_TYPE_RESULT_ID);
210*61046927SAndroid Build Coastguard Worker       id = ins->words[op->offset];
211*61046927SAndroid Build Coastguard Worker       curKernel->args.push_back(SPIRVKernelArg(id, typeId));
212*61046927SAndroid Build Coastguard Worker    }
213*61046927SAndroid Build Coastguard Worker 
214*61046927SAndroid Build Coastguard Worker    void parseName(const spv_parsed_instruction_t *ins)
215*61046927SAndroid Build Coastguard Worker    {
216*61046927SAndroid Build Coastguard Worker       const spv_parsed_operand_t *op;
217*61046927SAndroid Build Coastguard Worker       const char *name;
218*61046927SAndroid Build Coastguard Worker       uint32_t id;
219*61046927SAndroid Build Coastguard Worker 
220*61046927SAndroid Build Coastguard Worker       assert(ins->num_operands == 2);
221*61046927SAndroid Build Coastguard Worker 
222*61046927SAndroid Build Coastguard Worker       op = &ins->operands[0];
223*61046927SAndroid Build Coastguard Worker       assert(op->type == SPV_OPERAND_TYPE_ID);
224*61046927SAndroid Build Coastguard Worker       id = ins->words[op->offset];
225*61046927SAndroid Build Coastguard Worker       op = &ins->operands[1];
226*61046927SAndroid Build Coastguard Worker       assert(op->type == SPV_OPERAND_TYPE_LITERAL_STRING);
227*61046927SAndroid Build Coastguard Worker       name = reinterpret_cast<const char *>(ins->words + op->offset);
228*61046927SAndroid Build Coastguard Worker 
229*61046927SAndroid Build Coastguard Worker       for (auto &kernel : kernels) {
230*61046927SAndroid Build Coastguard Worker          for (auto &arg : kernel.args) {
231*61046927SAndroid Build Coastguard Worker             if (arg.id == id && arg.name.empty()) {
232*61046927SAndroid Build Coastguard Worker               arg.name = name;
233*61046927SAndroid Build Coastguard Worker               break;
234*61046927SAndroid Build Coastguard Worker 	    }
235*61046927SAndroid Build Coastguard Worker          }
236*61046927SAndroid Build Coastguard Worker       }
237*61046927SAndroid Build Coastguard Worker    }
238*61046927SAndroid Build Coastguard Worker 
239*61046927SAndroid Build Coastguard Worker    void parseTypePointer(const spv_parsed_instruction_t *ins)
240*61046927SAndroid Build Coastguard Worker    {
241*61046927SAndroid Build Coastguard Worker       enum clc_kernel_arg_address_qualifier addrQualifier;
242*61046927SAndroid Build Coastguard Worker       uint32_t typeId, storageClass;
243*61046927SAndroid Build Coastguard Worker       const spv_parsed_operand_t *op;
244*61046927SAndroid Build Coastguard Worker 
245*61046927SAndroid Build Coastguard Worker       assert(ins->num_operands == 3);
246*61046927SAndroid Build Coastguard Worker 
247*61046927SAndroid Build Coastguard Worker       op = &ins->operands[0];
248*61046927SAndroid Build Coastguard Worker       assert(op->type == SPV_OPERAND_TYPE_RESULT_ID);
249*61046927SAndroid Build Coastguard Worker       typeId = ins->words[op->offset];
250*61046927SAndroid Build Coastguard Worker 
251*61046927SAndroid Build Coastguard Worker       op = &ins->operands[1];
252*61046927SAndroid Build Coastguard Worker       assert(op->type == SPV_OPERAND_TYPE_STORAGE_CLASS);
253*61046927SAndroid Build Coastguard Worker       storageClass = ins->words[op->offset];
254*61046927SAndroid Build Coastguard Worker       switch (storageClass) {
255*61046927SAndroid Build Coastguard Worker       case SpvStorageClassCrossWorkgroup:
256*61046927SAndroid Build Coastguard Worker          addrQualifier = CLC_KERNEL_ARG_ADDRESS_GLOBAL;
257*61046927SAndroid Build Coastguard Worker          break;
258*61046927SAndroid Build Coastguard Worker       case SpvStorageClassWorkgroup:
259*61046927SAndroid Build Coastguard Worker          addrQualifier = CLC_KERNEL_ARG_ADDRESS_LOCAL;
260*61046927SAndroid Build Coastguard Worker          break;
261*61046927SAndroid Build Coastguard Worker       case SpvStorageClassUniformConstant:
262*61046927SAndroid Build Coastguard Worker          addrQualifier = CLC_KERNEL_ARG_ADDRESS_CONSTANT;
263*61046927SAndroid Build Coastguard Worker          break;
264*61046927SAndroid Build Coastguard Worker       default:
265*61046927SAndroid Build Coastguard Worker          addrQualifier = CLC_KERNEL_ARG_ADDRESS_PRIVATE;
266*61046927SAndroid Build Coastguard Worker          break;
267*61046927SAndroid Build Coastguard Worker       }
268*61046927SAndroid Build Coastguard Worker 
269*61046927SAndroid Build Coastguard Worker       for (auto &kernel : kernels) {
270*61046927SAndroid Build Coastguard Worker 	 for (auto &arg : kernel.args) {
271*61046927SAndroid Build Coastguard Worker             if (arg.typeId == typeId) {
272*61046927SAndroid Build Coastguard Worker                arg.addrQualifier = addrQualifier;
273*61046927SAndroid Build Coastguard Worker                if (addrQualifier == CLC_KERNEL_ARG_ADDRESS_CONSTANT)
274*61046927SAndroid Build Coastguard Worker                   arg.typeQualifier |= CLC_KERNEL_ARG_TYPE_CONST;
275*61046927SAndroid Build Coastguard Worker             }
276*61046927SAndroid Build Coastguard Worker          }
277*61046927SAndroid Build Coastguard Worker       }
278*61046927SAndroid Build Coastguard Worker    }
279*61046927SAndroid Build Coastguard Worker 
280*61046927SAndroid Build Coastguard Worker    void parseOpString(const spv_parsed_instruction_t *ins)
281*61046927SAndroid Build Coastguard Worker    {
282*61046927SAndroid Build Coastguard Worker       const spv_parsed_operand_t *op;
283*61046927SAndroid Build Coastguard Worker       std::string str;
284*61046927SAndroid Build Coastguard Worker 
285*61046927SAndroid Build Coastguard Worker       assert(ins->num_operands == 2);
286*61046927SAndroid Build Coastguard Worker 
287*61046927SAndroid Build Coastguard Worker       op = &ins->operands[1];
288*61046927SAndroid Build Coastguard Worker       assert(op->type == SPV_OPERAND_TYPE_LITERAL_STRING);
289*61046927SAndroid Build Coastguard Worker       str = reinterpret_cast<const char *>(ins->words + op->offset);
290*61046927SAndroid Build Coastguard Worker 
291*61046927SAndroid Build Coastguard Worker       size_t start = 0;
292*61046927SAndroid Build Coastguard Worker       enum class string_type {
293*61046927SAndroid Build Coastguard Worker          arg_type,
294*61046927SAndroid Build Coastguard Worker          arg_type_qual,
295*61046927SAndroid Build Coastguard Worker       } str_type;
296*61046927SAndroid Build Coastguard Worker 
297*61046927SAndroid Build Coastguard Worker       if (str.find("kernel_arg_type.") == 0) {
298*61046927SAndroid Build Coastguard Worker          start = sizeof("kernel_arg_type.") - 1;
299*61046927SAndroid Build Coastguard Worker          str_type = string_type::arg_type;
300*61046927SAndroid Build Coastguard Worker       } else if (str.find("kernel_arg_type_qual.") == 0) {
301*61046927SAndroid Build Coastguard Worker          start = sizeof("kernel_arg_type_qual.") - 1;
302*61046927SAndroid Build Coastguard Worker          str_type = string_type::arg_type_qual;
303*61046927SAndroid Build Coastguard Worker       } else {
304*61046927SAndroid Build Coastguard Worker          return;
305*61046927SAndroid Build Coastguard Worker       }
306*61046927SAndroid Build Coastguard Worker 
307*61046927SAndroid Build Coastguard Worker       for (auto &kernel : kernels) {
308*61046927SAndroid Build Coastguard Worker          size_t pos;
309*61046927SAndroid Build Coastguard Worker 
310*61046927SAndroid Build Coastguard Worker 	 pos = str.find(kernel.name, start);
311*61046927SAndroid Build Coastguard Worker          if (pos == std::string::npos ||
312*61046927SAndroid Build Coastguard Worker              pos != start || str[start + kernel.name.size()] != '.')
313*61046927SAndroid Build Coastguard Worker             continue;
314*61046927SAndroid Build Coastguard Worker 
315*61046927SAndroid Build Coastguard Worker 	 pos = start + kernel.name.size();
316*61046927SAndroid Build Coastguard Worker          if (str[pos++] != '.')
317*61046927SAndroid Build Coastguard Worker             continue;
318*61046927SAndroid Build Coastguard Worker 
319*61046927SAndroid Build Coastguard Worker          for (auto &arg : kernel.args) {
320*61046927SAndroid Build Coastguard Worker             if (arg.name.empty())
321*61046927SAndroid Build Coastguard Worker                break;
322*61046927SAndroid Build Coastguard Worker 
323*61046927SAndroid Build Coastguard Worker             size_t entryEnd = str.find(',', pos);
324*61046927SAndroid Build Coastguard Worker 	    if (entryEnd == std::string::npos)
325*61046927SAndroid Build Coastguard Worker                break;
326*61046927SAndroid Build Coastguard Worker 
327*61046927SAndroid Build Coastguard Worker             std::string entryVal = str.substr(pos, entryEnd - pos);
328*61046927SAndroid Build Coastguard Worker             pos = entryEnd + 1;
329*61046927SAndroid Build Coastguard Worker 
330*61046927SAndroid Build Coastguard Worker             if (str_type == string_type::arg_type) {
331*61046927SAndroid Build Coastguard Worker                arg.typeName = std::move(entryVal);
332*61046927SAndroid Build Coastguard Worker             } else if (str_type == string_type::arg_type_qual) {
333*61046927SAndroid Build Coastguard Worker                if (entryVal.find("const") != std::string::npos)
334*61046927SAndroid Build Coastguard Worker                   arg.typeQualifier |= CLC_KERNEL_ARG_TYPE_CONST;
335*61046927SAndroid Build Coastguard Worker             }
336*61046927SAndroid Build Coastguard Worker          }
337*61046927SAndroid Build Coastguard Worker       }
338*61046927SAndroid Build Coastguard Worker    }
339*61046927SAndroid Build Coastguard Worker 
340*61046927SAndroid Build Coastguard Worker    void applyDecoration(uint32_t id, const spv_parsed_instruction_t *ins)
341*61046927SAndroid Build Coastguard Worker    {
342*61046927SAndroid Build Coastguard Worker       auto iter = decorationGroups.find(id);
343*61046927SAndroid Build Coastguard Worker       if (iter != decorationGroups.end()) {
344*61046927SAndroid Build Coastguard Worker          for (uint32_t entry : iter->second)
345*61046927SAndroid Build Coastguard Worker             applyDecoration(entry, ins);
346*61046927SAndroid Build Coastguard Worker          return;
347*61046927SAndroid Build Coastguard Worker       }
348*61046927SAndroid Build Coastguard Worker 
349*61046927SAndroid Build Coastguard Worker       const spv_parsed_operand_t *op;
350*61046927SAndroid Build Coastguard Worker       uint32_t decoration;
351*61046927SAndroid Build Coastguard Worker 
352*61046927SAndroid Build Coastguard Worker       assert(ins->num_operands >= 2);
353*61046927SAndroid Build Coastguard Worker 
354*61046927SAndroid Build Coastguard Worker       op = &ins->operands[1];
355*61046927SAndroid Build Coastguard Worker       assert(op->type == SPV_OPERAND_TYPE_DECORATION);
356*61046927SAndroid Build Coastguard Worker       decoration = ins->words[op->offset];
357*61046927SAndroid Build Coastguard Worker 
358*61046927SAndroid Build Coastguard Worker       if (decoration == SpvDecorationSpecId) {
359*61046927SAndroid Build Coastguard Worker          uint32_t spec_id = ins->words[ins->operands[2].offset];
360*61046927SAndroid Build Coastguard Worker          for (auto &c : specConstants) {
361*61046927SAndroid Build Coastguard Worker             if (c.second.id == spec_id) {
362*61046927SAndroid Build Coastguard Worker                return;
363*61046927SAndroid Build Coastguard Worker             }
364*61046927SAndroid Build Coastguard Worker          }
365*61046927SAndroid Build Coastguard Worker          specConstants.emplace_back(id, clc_parsed_spec_constant{ spec_id });
366*61046927SAndroid Build Coastguard Worker          return;
367*61046927SAndroid Build Coastguard Worker       }
368*61046927SAndroid Build Coastguard Worker 
369*61046927SAndroid Build Coastguard Worker       for (auto &kernel : kernels) {
370*61046927SAndroid Build Coastguard Worker          for (auto &arg : kernel.args) {
371*61046927SAndroid Build Coastguard Worker             if (arg.id == id) {
372*61046927SAndroid Build Coastguard Worker                switch (decoration) {
373*61046927SAndroid Build Coastguard Worker                case SpvDecorationVolatile:
374*61046927SAndroid Build Coastguard Worker                   arg.typeQualifier |= CLC_KERNEL_ARG_TYPE_VOLATILE;
375*61046927SAndroid Build Coastguard Worker                   break;
376*61046927SAndroid Build Coastguard Worker                case SpvDecorationConstant:
377*61046927SAndroid Build Coastguard Worker                   arg.typeQualifier |= CLC_KERNEL_ARG_TYPE_CONST;
378*61046927SAndroid Build Coastguard Worker                   break;
379*61046927SAndroid Build Coastguard Worker                case SpvDecorationRestrict:
380*61046927SAndroid Build Coastguard Worker                   arg.typeQualifier |= CLC_KERNEL_ARG_TYPE_RESTRICT;
381*61046927SAndroid Build Coastguard Worker                   break;
382*61046927SAndroid Build Coastguard Worker                case SpvDecorationFuncParamAttr:
383*61046927SAndroid Build Coastguard Worker                   op = &ins->operands[2];
384*61046927SAndroid Build Coastguard Worker                   assert(op->type == SPV_OPERAND_TYPE_FUNCTION_PARAMETER_ATTRIBUTE);
385*61046927SAndroid Build Coastguard Worker                   switch (ins->words[op->offset]) {
386*61046927SAndroid Build Coastguard Worker                   case SpvFunctionParameterAttributeNoAlias:
387*61046927SAndroid Build Coastguard Worker                      arg.typeQualifier |= CLC_KERNEL_ARG_TYPE_RESTRICT;
388*61046927SAndroid Build Coastguard Worker                      break;
389*61046927SAndroid Build Coastguard Worker                   case SpvFunctionParameterAttributeNoWrite:
390*61046927SAndroid Build Coastguard Worker                      arg.typeQualifier |= CLC_KERNEL_ARG_TYPE_CONST;
391*61046927SAndroid Build Coastguard Worker                      break;
392*61046927SAndroid Build Coastguard Worker                   }
393*61046927SAndroid Build Coastguard Worker                   break;
394*61046927SAndroid Build Coastguard Worker                }
395*61046927SAndroid Build Coastguard Worker             }
396*61046927SAndroid Build Coastguard Worker 
397*61046927SAndroid Build Coastguard Worker          }
398*61046927SAndroid Build Coastguard Worker       }
399*61046927SAndroid Build Coastguard Worker    }
400*61046927SAndroid Build Coastguard Worker 
401*61046927SAndroid Build Coastguard Worker    void parseOpDecorate(const spv_parsed_instruction_t *ins)
402*61046927SAndroid Build Coastguard Worker    {
403*61046927SAndroid Build Coastguard Worker       const spv_parsed_operand_t *op;
404*61046927SAndroid Build Coastguard Worker       uint32_t id;
405*61046927SAndroid Build Coastguard Worker 
406*61046927SAndroid Build Coastguard Worker       assert(ins->num_operands >= 2);
407*61046927SAndroid Build Coastguard Worker 
408*61046927SAndroid Build Coastguard Worker       op = &ins->operands[0];
409*61046927SAndroid Build Coastguard Worker       assert(op->type == SPV_OPERAND_TYPE_ID);
410*61046927SAndroid Build Coastguard Worker       id = ins->words[op->offset];
411*61046927SAndroid Build Coastguard Worker 
412*61046927SAndroid Build Coastguard Worker       applyDecoration(id, ins);
413*61046927SAndroid Build Coastguard Worker    }
414*61046927SAndroid Build Coastguard Worker 
415*61046927SAndroid Build Coastguard Worker    void parseOpGroupDecorate(const spv_parsed_instruction_t *ins)
416*61046927SAndroid Build Coastguard Worker    {
417*61046927SAndroid Build Coastguard Worker       assert(ins->num_operands >= 2);
418*61046927SAndroid Build Coastguard Worker 
419*61046927SAndroid Build Coastguard Worker       const spv_parsed_operand_t *op = &ins->operands[0];
420*61046927SAndroid Build Coastguard Worker       assert(op->type == SPV_OPERAND_TYPE_ID);
421*61046927SAndroid Build Coastguard Worker       uint32_t groupId = ins->words[op->offset];
422*61046927SAndroid Build Coastguard Worker 
423*61046927SAndroid Build Coastguard Worker       auto lowerBound = decorationGroups.lower_bound(groupId);
424*61046927SAndroid Build Coastguard Worker       if (lowerBound != decorationGroups.end() &&
425*61046927SAndroid Build Coastguard Worker           lowerBound->first == groupId)
426*61046927SAndroid Build Coastguard Worker          // Group already filled out
427*61046927SAndroid Build Coastguard Worker          return;
428*61046927SAndroid Build Coastguard Worker 
429*61046927SAndroid Build Coastguard Worker       auto iter = decorationGroups.emplace_hint(lowerBound, groupId, std::vector<uint32_t>{});
430*61046927SAndroid Build Coastguard Worker       auto& vec = iter->second;
431*61046927SAndroid Build Coastguard Worker       vec.reserve(ins->num_operands - 1);
432*61046927SAndroid Build Coastguard Worker       for (uint32_t i = 1; i < ins->num_operands; ++i) {
433*61046927SAndroid Build Coastguard Worker          op = &ins->operands[i];
434*61046927SAndroid Build Coastguard Worker          assert(op->type == SPV_OPERAND_TYPE_ID);
435*61046927SAndroid Build Coastguard Worker          vec.push_back(ins->words[op->offset]);
436*61046927SAndroid Build Coastguard Worker       }
437*61046927SAndroid Build Coastguard Worker    }
438*61046927SAndroid Build Coastguard Worker 
439*61046927SAndroid Build Coastguard Worker    void parseOpTypeImage(const spv_parsed_instruction_t *ins)
440*61046927SAndroid Build Coastguard Worker    {
441*61046927SAndroid Build Coastguard Worker       const spv_parsed_operand_t *op;
442*61046927SAndroid Build Coastguard Worker       uint32_t typeId;
443*61046927SAndroid Build Coastguard Worker       unsigned accessQualifier = CLC_KERNEL_ARG_ACCESS_READ;
444*61046927SAndroid Build Coastguard Worker 
445*61046927SAndroid Build Coastguard Worker       op = &ins->operands[0];
446*61046927SAndroid Build Coastguard Worker       assert(op->type == SPV_OPERAND_TYPE_RESULT_ID);
447*61046927SAndroid Build Coastguard Worker       typeId = ins->words[op->offset];
448*61046927SAndroid Build Coastguard Worker 
449*61046927SAndroid Build Coastguard Worker       if (ins->num_operands >= 9) {
450*61046927SAndroid Build Coastguard Worker          op = &ins->operands[8];
451*61046927SAndroid Build Coastguard Worker          assert(op->type == SPV_OPERAND_TYPE_ACCESS_QUALIFIER);
452*61046927SAndroid Build Coastguard Worker          switch (ins->words[op->offset]) {
453*61046927SAndroid Build Coastguard Worker          case SpvAccessQualifierReadOnly:
454*61046927SAndroid Build Coastguard Worker             accessQualifier = CLC_KERNEL_ARG_ACCESS_READ;
455*61046927SAndroid Build Coastguard Worker             break;
456*61046927SAndroid Build Coastguard Worker          case SpvAccessQualifierWriteOnly:
457*61046927SAndroid Build Coastguard Worker             accessQualifier = CLC_KERNEL_ARG_ACCESS_WRITE;
458*61046927SAndroid Build Coastguard Worker             break;
459*61046927SAndroid Build Coastguard Worker          case SpvAccessQualifierReadWrite:
460*61046927SAndroid Build Coastguard Worker             accessQualifier = CLC_KERNEL_ARG_ACCESS_WRITE |
461*61046927SAndroid Build Coastguard Worker                CLC_KERNEL_ARG_ACCESS_READ;
462*61046927SAndroid Build Coastguard Worker             break;
463*61046927SAndroid Build Coastguard Worker          }
464*61046927SAndroid Build Coastguard Worker       }
465*61046927SAndroid Build Coastguard Worker 
466*61046927SAndroid Build Coastguard Worker       for (auto &kernel : kernels) {
467*61046927SAndroid Build Coastguard Worker 	 for (auto &arg : kernel.args) {
468*61046927SAndroid Build Coastguard Worker             if (arg.typeId == typeId) {
469*61046927SAndroid Build Coastguard Worker                arg.accessQualifier = accessQualifier;
470*61046927SAndroid Build Coastguard Worker                arg.addrQualifier = CLC_KERNEL_ARG_ADDRESS_GLOBAL;
471*61046927SAndroid Build Coastguard Worker             }
472*61046927SAndroid Build Coastguard Worker          }
473*61046927SAndroid Build Coastguard Worker       }
474*61046927SAndroid Build Coastguard Worker    }
475*61046927SAndroid Build Coastguard Worker 
476*61046927SAndroid Build Coastguard Worker    void parseExecutionMode(const spv_parsed_instruction_t *ins)
477*61046927SAndroid Build Coastguard Worker    {
478*61046927SAndroid Build Coastguard Worker       uint32_t executionMode = ins->words[ins->operands[1].offset];
479*61046927SAndroid Build Coastguard Worker       uint32_t funcId = ins->words[ins->operands[0].offset];
480*61046927SAndroid Build Coastguard Worker 
481*61046927SAndroid Build Coastguard Worker       for (auto& kernel : kernels) {
482*61046927SAndroid Build Coastguard Worker          if (kernel.funcId == funcId) {
483*61046927SAndroid Build Coastguard Worker             switch (executionMode) {
484*61046927SAndroid Build Coastguard Worker             case SpvExecutionModeVecTypeHint:
485*61046927SAndroid Build Coastguard Worker                kernel.vecHint = ins->words[ins->operands[2].offset];
486*61046927SAndroid Build Coastguard Worker                break;
487*61046927SAndroid Build Coastguard Worker             case SpvExecutionModeLocalSize:
488*61046927SAndroid Build Coastguard Worker                kernel.localSize[0] = ins->words[ins->operands[2].offset];
489*61046927SAndroid Build Coastguard Worker                kernel.localSize[1] = ins->words[ins->operands[3].offset];
490*61046927SAndroid Build Coastguard Worker                kernel.localSize[2] = ins->words[ins->operands[4].offset];
491*61046927SAndroid Build Coastguard Worker                break;
492*61046927SAndroid Build Coastguard Worker             case SpvExecutionModeLocalSizeHint:
493*61046927SAndroid Build Coastguard Worker                kernel.localSizeHint[0] = ins->words[ins->operands[2].offset];
494*61046927SAndroid Build Coastguard Worker                kernel.localSizeHint[1] = ins->words[ins->operands[3].offset];
495*61046927SAndroid Build Coastguard Worker                kernel.localSizeHint[2] = ins->words[ins->operands[4].offset];
496*61046927SAndroid Build Coastguard Worker                break;
497*61046927SAndroid Build Coastguard Worker             default:
498*61046927SAndroid Build Coastguard Worker                return;
499*61046927SAndroid Build Coastguard Worker             }
500*61046927SAndroid Build Coastguard Worker          }
501*61046927SAndroid Build Coastguard Worker       }
502*61046927SAndroid Build Coastguard Worker    }
503*61046927SAndroid Build Coastguard Worker 
504*61046927SAndroid Build Coastguard Worker    void parseLiteralType(const spv_parsed_instruction_t *ins)
505*61046927SAndroid Build Coastguard Worker    {
506*61046927SAndroid Build Coastguard Worker       uint32_t typeId = ins->words[ins->operands[0].offset];
507*61046927SAndroid Build Coastguard Worker       auto& literalType = literalTypes[typeId];
508*61046927SAndroid Build Coastguard Worker       switch (ins->opcode) {
509*61046927SAndroid Build Coastguard Worker       case SpvOpTypeBool:
510*61046927SAndroid Build Coastguard Worker          literalType = CLC_SPEC_CONSTANT_BOOL;
511*61046927SAndroid Build Coastguard Worker          break;
512*61046927SAndroid Build Coastguard Worker       case SpvOpTypeFloat: {
513*61046927SAndroid Build Coastguard Worker          uint32_t sizeInBits = ins->words[ins->operands[1].offset];
514*61046927SAndroid Build Coastguard Worker          switch (sizeInBits) {
515*61046927SAndroid Build Coastguard Worker          case 32:
516*61046927SAndroid Build Coastguard Worker             literalType = CLC_SPEC_CONSTANT_FLOAT;
517*61046927SAndroid Build Coastguard Worker             break;
518*61046927SAndroid Build Coastguard Worker          case 64:
519*61046927SAndroid Build Coastguard Worker             literalType = CLC_SPEC_CONSTANT_DOUBLE;
520*61046927SAndroid Build Coastguard Worker             break;
521*61046927SAndroid Build Coastguard Worker          case 16:
522*61046927SAndroid Build Coastguard Worker             /* Can't be used for a spec constant */
523*61046927SAndroid Build Coastguard Worker             break;
524*61046927SAndroid Build Coastguard Worker          default:
525*61046927SAndroid Build Coastguard Worker             unreachable("Unexpected float bit size");
526*61046927SAndroid Build Coastguard Worker          }
527*61046927SAndroid Build Coastguard Worker          break;
528*61046927SAndroid Build Coastguard Worker       }
529*61046927SAndroid Build Coastguard Worker       case SpvOpTypeInt: {
530*61046927SAndroid Build Coastguard Worker          uint32_t sizeInBits = ins->words[ins->operands[1].offset];
531*61046927SAndroid Build Coastguard Worker          bool isSigned = ins->words[ins->operands[2].offset];
532*61046927SAndroid Build Coastguard Worker          if (isSigned) {
533*61046927SAndroid Build Coastguard Worker             switch (sizeInBits) {
534*61046927SAndroid Build Coastguard Worker             case 8:
535*61046927SAndroid Build Coastguard Worker                literalType = CLC_SPEC_CONSTANT_INT8;
536*61046927SAndroid Build Coastguard Worker                break;
537*61046927SAndroid Build Coastguard Worker             case 16:
538*61046927SAndroid Build Coastguard Worker                literalType = CLC_SPEC_CONSTANT_INT16;
539*61046927SAndroid Build Coastguard Worker                break;
540*61046927SAndroid Build Coastguard Worker             case 32:
541*61046927SAndroid Build Coastguard Worker                literalType = CLC_SPEC_CONSTANT_INT32;
542*61046927SAndroid Build Coastguard Worker                break;
543*61046927SAndroid Build Coastguard Worker             case 64:
544*61046927SAndroid Build Coastguard Worker                literalType = CLC_SPEC_CONSTANT_INT64;
545*61046927SAndroid Build Coastguard Worker                break;
546*61046927SAndroid Build Coastguard Worker             default:
547*61046927SAndroid Build Coastguard Worker                unreachable("Unexpected int bit size");
548*61046927SAndroid Build Coastguard Worker             }
549*61046927SAndroid Build Coastguard Worker          } else {
550*61046927SAndroid Build Coastguard Worker             switch (sizeInBits) {
551*61046927SAndroid Build Coastguard Worker             case 8:
552*61046927SAndroid Build Coastguard Worker                literalType = CLC_SPEC_CONSTANT_UINT8;
553*61046927SAndroid Build Coastguard Worker                break;
554*61046927SAndroid Build Coastguard Worker             case 16:
555*61046927SAndroid Build Coastguard Worker                literalType = CLC_SPEC_CONSTANT_UINT16;
556*61046927SAndroid Build Coastguard Worker                break;
557*61046927SAndroid Build Coastguard Worker             case 32:
558*61046927SAndroid Build Coastguard Worker                literalType = CLC_SPEC_CONSTANT_UINT32;
559*61046927SAndroid Build Coastguard Worker                break;
560*61046927SAndroid Build Coastguard Worker             case 64:
561*61046927SAndroid Build Coastguard Worker                literalType = CLC_SPEC_CONSTANT_UINT64;
562*61046927SAndroid Build Coastguard Worker                break;
563*61046927SAndroid Build Coastguard Worker             default:
564*61046927SAndroid Build Coastguard Worker                unreachable("Unexpected uint bit size");
565*61046927SAndroid Build Coastguard Worker             }
566*61046927SAndroid Build Coastguard Worker          }
567*61046927SAndroid Build Coastguard Worker          break;
568*61046927SAndroid Build Coastguard Worker       }
569*61046927SAndroid Build Coastguard Worker       default:
570*61046927SAndroid Build Coastguard Worker          unreachable("Unexpected type opcode");
571*61046927SAndroid Build Coastguard Worker       }
572*61046927SAndroid Build Coastguard Worker    }
573*61046927SAndroid Build Coastguard Worker 
574*61046927SAndroid Build Coastguard Worker    void parseSpecConstant(const spv_parsed_instruction_t *ins)
575*61046927SAndroid Build Coastguard Worker    {
576*61046927SAndroid Build Coastguard Worker       uint32_t id = ins->result_id;
577*61046927SAndroid Build Coastguard Worker       for (auto& c : specConstants) {
578*61046927SAndroid Build Coastguard Worker          if (c.first == id) {
579*61046927SAndroid Build Coastguard Worker             auto& data = c.second;
580*61046927SAndroid Build Coastguard Worker             switch (ins->opcode) {
581*61046927SAndroid Build Coastguard Worker             case SpvOpSpecConstant: {
582*61046927SAndroid Build Coastguard Worker                uint32_t typeId = ins->words[ins->operands[0].offset];
583*61046927SAndroid Build Coastguard Worker 
584*61046927SAndroid Build Coastguard Worker                // This better be an integer or float type
585*61046927SAndroid Build Coastguard Worker                auto typeIter = literalTypes.find(typeId);
586*61046927SAndroid Build Coastguard Worker                assert(typeIter != literalTypes.end());
587*61046927SAndroid Build Coastguard Worker 
588*61046927SAndroid Build Coastguard Worker                data.type = typeIter->second;
589*61046927SAndroid Build Coastguard Worker                break;
590*61046927SAndroid Build Coastguard Worker             }
591*61046927SAndroid Build Coastguard Worker             case SpvOpSpecConstantFalse:
592*61046927SAndroid Build Coastguard Worker             case SpvOpSpecConstantTrue:
593*61046927SAndroid Build Coastguard Worker                data.type = CLC_SPEC_CONSTANT_BOOL;
594*61046927SAndroid Build Coastguard Worker                break;
595*61046927SAndroid Build Coastguard Worker             default:
596*61046927SAndroid Build Coastguard Worker                unreachable("Composites and Ops are not directly specializable.");
597*61046927SAndroid Build Coastguard Worker             }
598*61046927SAndroid Build Coastguard Worker          }
599*61046927SAndroid Build Coastguard Worker       }
600*61046927SAndroid Build Coastguard Worker    }
601*61046927SAndroid Build Coastguard Worker 
602*61046927SAndroid Build Coastguard Worker    static spv_result_t
603*61046927SAndroid Build Coastguard Worker    parseInstruction(void *data, const spv_parsed_instruction_t *ins)
604*61046927SAndroid Build Coastguard Worker    {
605*61046927SAndroid Build Coastguard Worker       SPIRVKernelParser *parser = reinterpret_cast<SPIRVKernelParser *>(data);
606*61046927SAndroid Build Coastguard Worker 
607*61046927SAndroid Build Coastguard Worker       switch (ins->opcode) {
608*61046927SAndroid Build Coastguard Worker       case SpvOpName:
609*61046927SAndroid Build Coastguard Worker          parser->parseName(ins);
610*61046927SAndroid Build Coastguard Worker          break;
611*61046927SAndroid Build Coastguard Worker       case SpvOpEntryPoint:
612*61046927SAndroid Build Coastguard Worker          parser->parseEntryPoint(ins);
613*61046927SAndroid Build Coastguard Worker          break;
614*61046927SAndroid Build Coastguard Worker       case SpvOpFunction:
615*61046927SAndroid Build Coastguard Worker          parser->parseFunction(ins);
616*61046927SAndroid Build Coastguard Worker          break;
617*61046927SAndroid Build Coastguard Worker       case SpvOpFunctionParameter:
618*61046927SAndroid Build Coastguard Worker          parser->parseFunctionParam(ins);
619*61046927SAndroid Build Coastguard Worker          break;
620*61046927SAndroid Build Coastguard Worker       case SpvOpFunctionEnd:
621*61046927SAndroid Build Coastguard Worker       case SpvOpLabel:
622*61046927SAndroid Build Coastguard Worker          parser->curKernel = NULL;
623*61046927SAndroid Build Coastguard Worker          break;
624*61046927SAndroid Build Coastguard Worker       case SpvOpTypePointer:
625*61046927SAndroid Build Coastguard Worker          parser->parseTypePointer(ins);
626*61046927SAndroid Build Coastguard Worker          break;
627*61046927SAndroid Build Coastguard Worker       case SpvOpTypeImage:
628*61046927SAndroid Build Coastguard Worker          parser->parseOpTypeImage(ins);
629*61046927SAndroid Build Coastguard Worker          break;
630*61046927SAndroid Build Coastguard Worker       case SpvOpString:
631*61046927SAndroid Build Coastguard Worker          parser->parseOpString(ins);
632*61046927SAndroid Build Coastguard Worker          break;
633*61046927SAndroid Build Coastguard Worker       case SpvOpDecorate:
634*61046927SAndroid Build Coastguard Worker          parser->parseOpDecorate(ins);
635*61046927SAndroid Build Coastguard Worker          break;
636*61046927SAndroid Build Coastguard Worker       case SpvOpGroupDecorate:
637*61046927SAndroid Build Coastguard Worker          parser->parseOpGroupDecorate(ins);
638*61046927SAndroid Build Coastguard Worker          break;
639*61046927SAndroid Build Coastguard Worker       case SpvOpExecutionMode:
640*61046927SAndroid Build Coastguard Worker          parser->parseExecutionMode(ins);
641*61046927SAndroid Build Coastguard Worker          break;
642*61046927SAndroid Build Coastguard Worker       case SpvOpTypeBool:
643*61046927SAndroid Build Coastguard Worker       case SpvOpTypeInt:
644*61046927SAndroid Build Coastguard Worker       case SpvOpTypeFloat:
645*61046927SAndroid Build Coastguard Worker          parser->parseLiteralType(ins);
646*61046927SAndroid Build Coastguard Worker          break;
647*61046927SAndroid Build Coastguard Worker       case SpvOpSpecConstant:
648*61046927SAndroid Build Coastguard Worker       case SpvOpSpecConstantFalse:
649*61046927SAndroid Build Coastguard Worker       case SpvOpSpecConstantTrue:
650*61046927SAndroid Build Coastguard Worker          parser->parseSpecConstant(ins);
651*61046927SAndroid Build Coastguard Worker          break;
652*61046927SAndroid Build Coastguard Worker       default:
653*61046927SAndroid Build Coastguard Worker          break;
654*61046927SAndroid Build Coastguard Worker       }
655*61046927SAndroid Build Coastguard Worker 
656*61046927SAndroid Build Coastguard Worker       return SPV_SUCCESS;
657*61046927SAndroid Build Coastguard Worker    }
658*61046927SAndroid Build Coastguard Worker 
659*61046927SAndroid Build Coastguard Worker    bool parseBinary(const struct clc_binary &spvbin, const struct clc_logger *logger)
660*61046927SAndroid Build Coastguard Worker    {
661*61046927SAndroid Build Coastguard Worker       /* 3 passes should be enough to retrieve all kernel information:
662*61046927SAndroid Build Coastguard Worker        * 1st pass: all entry point name and number of args
663*61046927SAndroid Build Coastguard Worker        * 2nd pass: argument names and type names
664*61046927SAndroid Build Coastguard Worker        * 3rd pass: pointer type names
665*61046927SAndroid Build Coastguard Worker        */
666*61046927SAndroid Build Coastguard Worker       for (unsigned pass = 0; pass < 3; pass++) {
667*61046927SAndroid Build Coastguard Worker          spv_diagnostic diagnostic = NULL;
668*61046927SAndroid Build Coastguard Worker          auto result = spvBinaryParse(ctx, reinterpret_cast<void *>(this),
669*61046927SAndroid Build Coastguard Worker                                       static_cast<uint32_t*>(spvbin.data), spvbin.size / 4,
670*61046927SAndroid Build Coastguard Worker                                       NULL, parseInstruction, &diagnostic);
671*61046927SAndroid Build Coastguard Worker 
672*61046927SAndroid Build Coastguard Worker          if (result != SPV_SUCCESS) {
673*61046927SAndroid Build Coastguard Worker             if (diagnostic && logger)
674*61046927SAndroid Build Coastguard Worker                logger->error(logger->priv, diagnostic->error);
675*61046927SAndroid Build Coastguard Worker             return false;
676*61046927SAndroid Build Coastguard Worker          }
677*61046927SAndroid Build Coastguard Worker       }
678*61046927SAndroid Build Coastguard Worker 
679*61046927SAndroid Build Coastguard Worker       return true;
680*61046927SAndroid Build Coastguard Worker    }
681*61046927SAndroid Build Coastguard Worker 
682*61046927SAndroid Build Coastguard Worker    std::vector<SPIRVKernelInfo> kernels;
683*61046927SAndroid Build Coastguard Worker    std::vector<std::pair<uint32_t, clc_parsed_spec_constant>> specConstants;
684*61046927SAndroid Build Coastguard Worker    std::map<uint32_t, enum clc_spec_constant_type> literalTypes;
685*61046927SAndroid Build Coastguard Worker    std::map<uint32_t, std::vector<uint32_t>> decorationGroups;
686*61046927SAndroid Build Coastguard Worker    SPIRVKernelInfo *curKernel;
687*61046927SAndroid Build Coastguard Worker    spv_context ctx;
688*61046927SAndroid Build Coastguard Worker };
689*61046927SAndroid Build Coastguard Worker 
690*61046927SAndroid Build Coastguard Worker bool
691*61046927SAndroid Build Coastguard Worker clc_spirv_get_kernels_info(const struct clc_binary *spvbin,
692*61046927SAndroid Build Coastguard Worker                            const struct clc_kernel_info **out_kernels,
693*61046927SAndroid Build Coastguard Worker                            unsigned *num_kernels,
694*61046927SAndroid Build Coastguard Worker                            const struct clc_parsed_spec_constant **out_spec_constants,
695*61046927SAndroid Build Coastguard Worker                            unsigned *num_spec_constants,
696*61046927SAndroid Build Coastguard Worker                            const struct clc_logger *logger)
697*61046927SAndroid Build Coastguard Worker {
698*61046927SAndroid Build Coastguard Worker    struct clc_kernel_info *kernels = NULL;
699*61046927SAndroid Build Coastguard Worker    struct clc_parsed_spec_constant *spec_constants = NULL;
700*61046927SAndroid Build Coastguard Worker 
701*61046927SAndroid Build Coastguard Worker    SPIRVKernelParser parser;
702*61046927SAndroid Build Coastguard Worker 
703*61046927SAndroid Build Coastguard Worker    if (!parser.parseBinary(*spvbin, logger))
704*61046927SAndroid Build Coastguard Worker       return false;
705*61046927SAndroid Build Coastguard Worker 
706*61046927SAndroid Build Coastguard Worker    *num_kernels = parser.kernels.size();
707*61046927SAndroid Build Coastguard Worker    *num_spec_constants = parser.specConstants.size();
708*61046927SAndroid Build Coastguard Worker    if (*num_kernels) {
709*61046927SAndroid Build Coastguard Worker       kernels = reinterpret_cast<struct clc_kernel_info *>(calloc(*num_kernels,
710*61046927SAndroid Build Coastguard Worker                                                                   sizeof(*kernels)));
711*61046927SAndroid Build Coastguard Worker       assert(kernels);
712*61046927SAndroid Build Coastguard Worker       for (unsigned i = 0; i < parser.kernels.size(); i++) {
713*61046927SAndroid Build Coastguard Worker          kernels[i].name = strdup(parser.kernels[i].name.c_str());
714*61046927SAndroid Build Coastguard Worker          kernels[i].num_args = parser.kernels[i].args.size();
715*61046927SAndroid Build Coastguard Worker          kernels[i].vec_hint_size = parser.kernels[i].vecHint >> 16;
716*61046927SAndroid Build Coastguard Worker          kernels[i].vec_hint_type = (enum clc_vec_hint_type)(parser.kernels[i].vecHint & 0xFFFF);
717*61046927SAndroid Build Coastguard Worker          memcpy(kernels[i].local_size, parser.kernels[i].localSize, sizeof(kernels[i].local_size));
718*61046927SAndroid Build Coastguard Worker          memcpy(kernels[i].local_size_hint, parser.kernels[i].localSizeHint, sizeof(kernels[i].local_size_hint));
719*61046927SAndroid Build Coastguard Worker          if (!kernels[i].num_args)
720*61046927SAndroid Build Coastguard Worker             continue;
721*61046927SAndroid Build Coastguard Worker 
722*61046927SAndroid Build Coastguard Worker          struct clc_kernel_arg *args;
723*61046927SAndroid Build Coastguard Worker 
724*61046927SAndroid Build Coastguard Worker          args = reinterpret_cast<struct clc_kernel_arg *>(calloc(kernels[i].num_args,
725*61046927SAndroid Build Coastguard Worker                                                                  sizeof(*kernels->args)));
726*61046927SAndroid Build Coastguard Worker          kernels[i].args = args;
727*61046927SAndroid Build Coastguard Worker          assert(args);
728*61046927SAndroid Build Coastguard Worker          for (unsigned j = 0; j < kernels[i].num_args; j++) {
729*61046927SAndroid Build Coastguard Worker             if (!parser.kernels[i].args[j].name.empty())
730*61046927SAndroid Build Coastguard Worker                args[j].name = strdup(parser.kernels[i].args[j].name.c_str());
731*61046927SAndroid Build Coastguard Worker             args[j].type_name = strdup(parser.kernels[i].args[j].typeName.c_str());
732*61046927SAndroid Build Coastguard Worker             args[j].address_qualifier = parser.kernels[i].args[j].addrQualifier;
733*61046927SAndroid Build Coastguard Worker             args[j].type_qualifier = parser.kernels[i].args[j].typeQualifier;
734*61046927SAndroid Build Coastguard Worker             args[j].access_qualifier = parser.kernels[i].args[j].accessQualifier;
735*61046927SAndroid Build Coastguard Worker          }
736*61046927SAndroid Build Coastguard Worker       }
737*61046927SAndroid Build Coastguard Worker    }
738*61046927SAndroid Build Coastguard Worker 
739*61046927SAndroid Build Coastguard Worker    if (*num_spec_constants) {
740*61046927SAndroid Build Coastguard Worker       spec_constants = reinterpret_cast<struct clc_parsed_spec_constant *>(calloc(*num_spec_constants,
741*61046927SAndroid Build Coastguard Worker                                                                                   sizeof(*spec_constants)));
742*61046927SAndroid Build Coastguard Worker       assert(spec_constants);
743*61046927SAndroid Build Coastguard Worker 
744*61046927SAndroid Build Coastguard Worker       for (unsigned i = 0; i < parser.specConstants.size(); ++i) {
745*61046927SAndroid Build Coastguard Worker          spec_constants[i] = parser.specConstants[i].second;
746*61046927SAndroid Build Coastguard Worker       }
747*61046927SAndroid Build Coastguard Worker    }
748*61046927SAndroid Build Coastguard Worker 
749*61046927SAndroid Build Coastguard Worker    *out_kernels = kernels;
750*61046927SAndroid Build Coastguard Worker    *out_spec_constants = spec_constants;
751*61046927SAndroid Build Coastguard Worker 
752*61046927SAndroid Build Coastguard Worker    return true;
753*61046927SAndroid Build Coastguard Worker }
754*61046927SAndroid Build Coastguard Worker 
755*61046927SAndroid Build Coastguard Worker void
756*61046927SAndroid Build Coastguard Worker clc_free_kernels_info(const struct clc_kernel_info *kernels,
757*61046927SAndroid Build Coastguard Worker                       unsigned num_kernels)
758*61046927SAndroid Build Coastguard Worker {
759*61046927SAndroid Build Coastguard Worker    if (!kernels)
760*61046927SAndroid Build Coastguard Worker       return;
761*61046927SAndroid Build Coastguard Worker 
762*61046927SAndroid Build Coastguard Worker    for (unsigned i = 0; i < num_kernels; i++) {
763*61046927SAndroid Build Coastguard Worker       if (kernels[i].args) {
764*61046927SAndroid Build Coastguard Worker          for (unsigned j = 0; j < kernels[i].num_args; j++) {
765*61046927SAndroid Build Coastguard Worker             free((void *)kernels[i].args[j].name);
766*61046927SAndroid Build Coastguard Worker             free((void *)kernels[i].args[j].type_name);
767*61046927SAndroid Build Coastguard Worker          }
768*61046927SAndroid Build Coastguard Worker          free((void *)kernels[i].args);
769*61046927SAndroid Build Coastguard Worker       }
770*61046927SAndroid Build Coastguard Worker       free((void *)kernels[i].name);
771*61046927SAndroid Build Coastguard Worker    }
772*61046927SAndroid Build Coastguard Worker 
773*61046927SAndroid Build Coastguard Worker    free((void *)kernels);
774*61046927SAndroid Build Coastguard Worker }
775*61046927SAndroid Build Coastguard Worker 
776*61046927SAndroid Build Coastguard Worker static std::unique_ptr<::llvm::Module>
777*61046927SAndroid Build Coastguard Worker clc_compile_to_llvm_module(LLVMContext &llvm_ctx,
778*61046927SAndroid Build Coastguard Worker                            const struct clc_compile_args *args,
779*61046927SAndroid Build Coastguard Worker                            const struct clc_logger *logger)
780*61046927SAndroid Build Coastguard Worker {
781*61046927SAndroid Build Coastguard Worker    static_assert(std::has_unique_object_representations<clc_optional_features>(),
782*61046927SAndroid Build Coastguard Worker                  "no padding allowed inside clc_optional_features");
783*61046927SAndroid Build Coastguard Worker 
784*61046927SAndroid Build Coastguard Worker    std::string diag_log_str;
785*61046927SAndroid Build Coastguard Worker    raw_string_ostream diag_log_stream { diag_log_str };
786*61046927SAndroid Build Coastguard Worker 
787*61046927SAndroid Build Coastguard Worker    std::unique_ptr<clang::CompilerInstance> c { new clang::CompilerInstance };
788*61046927SAndroid Build Coastguard Worker 
789*61046927SAndroid Build Coastguard Worker    clang::DiagnosticsEngine diag {
790*61046927SAndroid Build Coastguard Worker       new clang::DiagnosticIDs,
791*61046927SAndroid Build Coastguard Worker       new clang::DiagnosticOptions,
792*61046927SAndroid Build Coastguard Worker       new clang::TextDiagnosticPrinter(diag_log_stream,
793*61046927SAndroid Build Coastguard Worker                                        &c->getDiagnosticOpts())
794*61046927SAndroid Build Coastguard Worker    };
795*61046927SAndroid Build Coastguard Worker 
796*61046927SAndroid Build Coastguard Worker #if LLVM_VERSION_MAJOR >= 17
797*61046927SAndroid Build Coastguard Worker    const char *triple = args->address_bits == 32 ? "spir-unknown-unknown" : "spirv64-unknown-unknown";
798*61046927SAndroid Build Coastguard Worker #else
799*61046927SAndroid Build Coastguard Worker    const char *triple = args->address_bits == 32 ? "spir-unknown-unknown" : "spir64-unknown-unknown";
800*61046927SAndroid Build Coastguard Worker #endif
801*61046927SAndroid Build Coastguard Worker 
802*61046927SAndroid Build Coastguard Worker    std::vector<const char *> clang_opts = {
803*61046927SAndroid Build Coastguard Worker       args->source.name,
804*61046927SAndroid Build Coastguard Worker       "-triple", triple,
805*61046927SAndroid Build Coastguard Worker       // By default, clang prefers to use modules to pull in the default headers,
806*61046927SAndroid Build Coastguard Worker       // which doesn't work with our technique of embedding the headers in our binary
807*61046927SAndroid Build Coastguard Worker       "-fdeclare-opencl-builtins",
808*61046927SAndroid Build Coastguard Worker #if LLVM_VERSION_MAJOR < 17
809*61046927SAndroid Build Coastguard Worker       "-no-opaque-pointers",
810*61046927SAndroid Build Coastguard Worker #endif
811*61046927SAndroid Build Coastguard Worker       // Add a default CL compiler version. Clang will pick the last one specified
812*61046927SAndroid Build Coastguard Worker       // on the command line, so the app can override this one.
813*61046927SAndroid Build Coastguard Worker       "-cl-std=cl1.2",
814*61046927SAndroid Build Coastguard Worker       // The LLVM-SPIRV-Translator doesn't support memset with variable size
815*61046927SAndroid Build Coastguard Worker       "-fno-builtin-memset",
816*61046927SAndroid Build Coastguard Worker       // LLVM's optimizations can produce code that the translator can't translate
817*61046927SAndroid Build Coastguard Worker       "-O0",
818*61046927SAndroid Build Coastguard Worker       // Ensure inline functions are actually emitted
819*61046927SAndroid Build Coastguard Worker       "-fgnu89-inline",
820*61046927SAndroid Build Coastguard Worker    };
821*61046927SAndroid Build Coastguard Worker 
822*61046927SAndroid Build Coastguard Worker    // We assume there's appropriate defines for __OPENCL_VERSION__ and __IMAGE_SUPPORT__
823*61046927SAndroid Build Coastguard Worker    // being provided by the caller here.
824*61046927SAndroid Build Coastguard Worker    clang_opts.insert(clang_opts.end(), args->args, args->args + args->num_args);
825*61046927SAndroid Build Coastguard Worker 
826*61046927SAndroid Build Coastguard Worker    if (!clang::CompilerInvocation::CreateFromArgs(c->getInvocation(),
827*61046927SAndroid Build Coastguard Worker                                                   clang_opts,
828*61046927SAndroid Build Coastguard Worker                                                   diag)) {
829*61046927SAndroid Build Coastguard Worker       clc_error(logger, "Couldn't create Clang invocation.\n");
830*61046927SAndroid Build Coastguard Worker       return {};
831*61046927SAndroid Build Coastguard Worker    }
832*61046927SAndroid Build Coastguard Worker 
833*61046927SAndroid Build Coastguard Worker    if (diag.hasErrorOccurred()) {
834*61046927SAndroid Build Coastguard Worker       clc_error(logger, "%sErrors occurred during Clang invocation.\n",
835*61046927SAndroid Build Coastguard Worker                 diag_log_str.c_str());
836*61046927SAndroid Build Coastguard Worker       return {};
837*61046927SAndroid Build Coastguard Worker    }
838*61046927SAndroid Build Coastguard Worker 
839*61046927SAndroid Build Coastguard Worker    // This is a workaround for a Clang bug which causes the number
840*61046927SAndroid Build Coastguard Worker    // of warnings and errors to be printed to stderr.
841*61046927SAndroid Build Coastguard Worker    // http://www.llvm.org/bugs/show_bug.cgi?id=19735
842*61046927SAndroid Build Coastguard Worker    c->getDiagnosticOpts().ShowCarets = false;
843*61046927SAndroid Build Coastguard Worker 
844*61046927SAndroid Build Coastguard Worker    c->createDiagnostics(new clang::TextDiagnosticPrinter(
845*61046927SAndroid Build Coastguard Worker                            diag_log_stream,
846*61046927SAndroid Build Coastguard Worker                            &c->getDiagnosticOpts()));
847*61046927SAndroid Build Coastguard Worker 
848*61046927SAndroid Build Coastguard Worker    c->setTarget(clang::TargetInfo::CreateTargetInfo(
849*61046927SAndroid Build Coastguard Worker                    c->getDiagnostics(), c->getInvocation().TargetOpts));
850*61046927SAndroid Build Coastguard Worker 
851*61046927SAndroid Build Coastguard Worker    c->getFrontendOpts().ProgramAction = clang::frontend::EmitLLVMOnly;
852*61046927SAndroid Build Coastguard Worker 
853*61046927SAndroid Build Coastguard Worker #ifdef USE_STATIC_OPENCL_C_H
854*61046927SAndroid Build Coastguard Worker    c->getHeaderSearchOpts().UseBuiltinIncludes = false;
855*61046927SAndroid Build Coastguard Worker    c->getHeaderSearchOpts().UseStandardSystemIncludes = false;
856*61046927SAndroid Build Coastguard Worker 
857*61046927SAndroid Build Coastguard Worker    // Add opencl-c generic search path
858*61046927SAndroid Build Coastguard Worker    {
859*61046927SAndroid Build Coastguard Worker       ::llvm::SmallString<128> system_header_path;
860*61046927SAndroid Build Coastguard Worker       ::llvm::sys::path::system_temp_directory(true, system_header_path);
861*61046927SAndroid Build Coastguard Worker       ::llvm::sys::path::append(system_header_path, "openclon12");
862*61046927SAndroid Build Coastguard Worker       c->getHeaderSearchOpts().AddPath(system_header_path.str(),
863*61046927SAndroid Build Coastguard Worker                                        clang::frontend::Angled,
864*61046927SAndroid Build Coastguard Worker                                        false, false);
865*61046927SAndroid Build Coastguard Worker 
866*61046927SAndroid Build Coastguard Worker       ::llvm::sys::path::append(system_header_path, "opencl-c-base.h");
867*61046927SAndroid Build Coastguard Worker       c->getPreprocessorOpts().addRemappedFile(system_header_path.str(),
868*61046927SAndroid Build Coastguard Worker          ::llvm::MemoryBuffer::getMemBuffer(llvm::StringRef(opencl_c_base_source, ARRAY_SIZE(opencl_c_base_source) - 1)).release());
869*61046927SAndroid Build Coastguard Worker       // this line is actually important to make it include `opencl-c.h`
870*61046927SAndroid Build Coastguard Worker       ::llvm::sys::path::remove_filename(system_header_path);
871*61046927SAndroid Build Coastguard Worker       ::llvm::sys::path::append(system_header_path, "opencl-c.h");
872*61046927SAndroid Build Coastguard Worker       c->getPreprocessorOpts().addRemappedFile(system_header_path.str(),
873*61046927SAndroid Build Coastguard Worker          ::llvm::MemoryBuffer::getMemBuffer(llvm::StringRef(opencl_c_source, ARRAY_SIZE(opencl_c_source) - 1)).release());
874*61046927SAndroid Build Coastguard Worker    }
875*61046927SAndroid Build Coastguard Worker #else
876*61046927SAndroid Build Coastguard Worker 
877*61046927SAndroid Build Coastguard Worker    Dl_info info;
878*61046927SAndroid Build Coastguard Worker    if (dladdr((void *)clang::CompilerInvocation::CreateFromArgs, &info) == 0) {
879*61046927SAndroid Build Coastguard Worker       clc_error(logger, "Couldn't find libclang path.\n");
880*61046927SAndroid Build Coastguard Worker       return {};
881*61046927SAndroid Build Coastguard Worker    }
882*61046927SAndroid Build Coastguard Worker 
883*61046927SAndroid Build Coastguard Worker    char *clang_path = realpath(info.dli_fname, NULL);
884*61046927SAndroid Build Coastguard Worker    if (clang_path == nullptr) {
885*61046927SAndroid Build Coastguard Worker       clc_error(logger, "Couldn't find libclang path.\n");
886*61046927SAndroid Build Coastguard Worker       return {};
887*61046927SAndroid Build Coastguard Worker    }
888*61046927SAndroid Build Coastguard Worker 
889*61046927SAndroid Build Coastguard Worker    // GetResourcePath is a way to retrieve the actual libclang resource dir based on a given binary
890*61046927SAndroid Build Coastguard Worker    // or library.
891*61046927SAndroid Build Coastguard Worker    auto tmp_res_path =
892*61046927SAndroid Build Coastguard Worker #if LLVM_VERSION_MAJOR >= 20
893*61046927SAndroid Build Coastguard Worker       Driver::GetResourcesPath(std::string(clang_path));
894*61046927SAndroid Build Coastguard Worker #else
895*61046927SAndroid Build Coastguard Worker       Driver::GetResourcesPath(std::string(clang_path), CLANG_RESOURCE_DIR);
896*61046927SAndroid Build Coastguard Worker #endif
897*61046927SAndroid Build Coastguard Worker    auto clang_res_path = fs::path(tmp_res_path) / "include";
898*61046927SAndroid Build Coastguard Worker 
899*61046927SAndroid Build Coastguard Worker    free(clang_path);
900*61046927SAndroid Build Coastguard Worker 
901*61046927SAndroid Build Coastguard Worker    c->getHeaderSearchOpts().UseBuiltinIncludes = true;
902*61046927SAndroid Build Coastguard Worker    c->getHeaderSearchOpts().UseStandardSystemIncludes = true;
903*61046927SAndroid Build Coastguard Worker    c->getHeaderSearchOpts().ResourceDir = clang_res_path.string();
904*61046927SAndroid Build Coastguard Worker 
905*61046927SAndroid Build Coastguard Worker    // Add opencl-c generic search path
906*61046927SAndroid Build Coastguard Worker    c->getHeaderSearchOpts().AddPath(clang_res_path.string(),
907*61046927SAndroid Build Coastguard Worker                                     clang::frontend::Angled,
908*61046927SAndroid Build Coastguard Worker                                     false, false);
909*61046927SAndroid Build Coastguard Worker 
910*61046927SAndroid Build Coastguard Worker    auto clang_install_res_path =
911*61046927SAndroid Build Coastguard Worker       fs::path(LLVM_LIB_DIR) / "clang" / std::to_string(LLVM_VERSION_MAJOR) / "include";
912*61046927SAndroid Build Coastguard Worker    c->getHeaderSearchOpts().AddPath(clang_install_res_path.string(),
913*61046927SAndroid Build Coastguard Worker                                     clang::frontend::Angled,
914*61046927SAndroid Build Coastguard Worker                                     false, false);
915*61046927SAndroid Build Coastguard Worker #endif
916*61046927SAndroid Build Coastguard Worker 
917*61046927SAndroid Build Coastguard Worker    // Enable/Disable optional OpenCL C features. Some can be toggled via `OpenCLExtensionsAsWritten`
918*61046927SAndroid Build Coastguard Worker    // others we have to (un)define via macros ourselves.
919*61046927SAndroid Build Coastguard Worker 
920*61046927SAndroid Build Coastguard Worker    // Undefine clang added SPIR(V) defines so we don't magically enable extensions
921*61046927SAndroid Build Coastguard Worker    c->getPreprocessorOpts().addMacroUndef("__SPIR__");
922*61046927SAndroid Build Coastguard Worker    c->getPreprocessorOpts().addMacroUndef("__SPIRV__");
923*61046927SAndroid Build Coastguard Worker 
924*61046927SAndroid Build Coastguard Worker    c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("-all");
925*61046927SAndroid Build Coastguard Worker    c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+cl_khr_byte_addressable_store");
926*61046927SAndroid Build Coastguard Worker    c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+cl_khr_global_int32_base_atomics");
927*61046927SAndroid Build Coastguard Worker    c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+cl_khr_global_int32_extended_atomics");
928*61046927SAndroid Build Coastguard Worker    c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+cl_khr_local_int32_base_atomics");
929*61046927SAndroid Build Coastguard Worker    c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+cl_khr_local_int32_extended_atomics");
930*61046927SAndroid Build Coastguard Worker    c->getPreprocessorOpts().addMacroDef("cl_khr_expect_assume=1");
931*61046927SAndroid Build Coastguard Worker 
932*61046927SAndroid Build Coastguard Worker    bool needs_opencl_c_h = false;
933*61046927SAndroid Build Coastguard Worker    if (args->features.fp16) {
934*61046927SAndroid Build Coastguard Worker       c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+cl_khr_fp16");
935*61046927SAndroid Build Coastguard Worker    }
936*61046927SAndroid Build Coastguard Worker    if (args->features.fp64) {
937*61046927SAndroid Build Coastguard Worker       c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+cl_khr_fp64");
938*61046927SAndroid Build Coastguard Worker       c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+__opencl_c_fp64");
939*61046927SAndroid Build Coastguard Worker    }
940*61046927SAndroid Build Coastguard Worker    if (args->features.int64) {
941*61046927SAndroid Build Coastguard Worker       c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+cles_khr_int64");
942*61046927SAndroid Build Coastguard Worker       c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+__opencl_c_int64");
943*61046927SAndroid Build Coastguard Worker    } else {
944*61046927SAndroid Build Coastguard Worker       // clang defines this unconditionally, we need to fix that.
945*61046927SAndroid Build Coastguard Worker       c->getPreprocessorOpts().addMacroUndef("__opencl_c_int64");
946*61046927SAndroid Build Coastguard Worker    }
947*61046927SAndroid Build Coastguard Worker    if (args->features.images) {
948*61046927SAndroid Build Coastguard Worker       c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+__opencl_c_images");
949*61046927SAndroid Build Coastguard Worker    } else {
950*61046927SAndroid Build Coastguard Worker       // clang defines this unconditionally, we need to fix that.
951*61046927SAndroid Build Coastguard Worker       c->getPreprocessorOpts().addMacroUndef("__IMAGE_SUPPORT__");
952*61046927SAndroid Build Coastguard Worker    }
953*61046927SAndroid Build Coastguard Worker    if (args->features.images_read_write) {
954*61046927SAndroid Build Coastguard Worker       c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+__opencl_c_read_write_images");
955*61046927SAndroid Build Coastguard Worker    }
956*61046927SAndroid Build Coastguard Worker    if (args->features.images_write_3d) {
957*61046927SAndroid Build Coastguard Worker       c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+cl_khr_3d_image_writes");
958*61046927SAndroid Build Coastguard Worker       c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+__opencl_c_3d_image_writes");
959*61046927SAndroid Build Coastguard Worker    }
960*61046927SAndroid Build Coastguard Worker    if (args->features.images_depth) {
961*61046927SAndroid Build Coastguard Worker       c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+cl_khr_depth_images");
962*61046927SAndroid Build Coastguard Worker    }
963*61046927SAndroid Build Coastguard Worker    if (args->features.images_gl_depth) {
964*61046927SAndroid Build Coastguard Worker       c->getPreprocessorOpts().addMacroDef("cl_khr_gl_depth_images=1");
965*61046927SAndroid Build Coastguard Worker    }
966*61046927SAndroid Build Coastguard Worker    if (args->features.images_mipmap) {
967*61046927SAndroid Build Coastguard Worker       c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+cl_khr_mipmap_image");
968*61046927SAndroid Build Coastguard Worker    }
969*61046927SAndroid Build Coastguard Worker    if (args->features.images_mipmap_writes) {
970*61046927SAndroid Build Coastguard Worker       c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+cl_khr_mipmap_image_writes");
971*61046927SAndroid Build Coastguard Worker    }
972*61046927SAndroid Build Coastguard Worker    if (args->features.images_gl_msaa) {
973*61046927SAndroid Build Coastguard Worker       c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+cl_khr_gl_msaa_sharing");
974*61046927SAndroid Build Coastguard Worker    }
975*61046927SAndroid Build Coastguard Worker    if (args->features.intel_subgroups) {
976*61046927SAndroid Build Coastguard Worker       c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+cl_intel_subgroups");
977*61046927SAndroid Build Coastguard Worker       needs_opencl_c_h = true;
978*61046927SAndroid Build Coastguard Worker    }
979*61046927SAndroid Build Coastguard Worker    if (args->features.subgroups) {
980*61046927SAndroid Build Coastguard Worker       c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+__opencl_c_subgroups");
981*61046927SAndroid Build Coastguard Worker       if (args->features.subgroups_shuffle) {
982*61046927SAndroid Build Coastguard Worker          c->getPreprocessorOpts().addMacroDef("cl_khr_subgroup_shuffle=1");
983*61046927SAndroid Build Coastguard Worker       }
984*61046927SAndroid Build Coastguard Worker       if (args->features.subgroups_shuffle_relative) {
985*61046927SAndroid Build Coastguard Worker          c->getPreprocessorOpts().addMacroDef("cl_khr_subgroup_shuffle_relative=1");
986*61046927SAndroid Build Coastguard Worker       }
987*61046927SAndroid Build Coastguard Worker    }
988*61046927SAndroid Build Coastguard Worker    if (args->features.subgroups_ifp) {
989*61046927SAndroid Build Coastguard Worker       assert(args->features.subgroups);
990*61046927SAndroid Build Coastguard Worker       c->getTargetOpts().OpenCLExtensionsAsWritten.push_back("+cl_khr_subgroups");
991*61046927SAndroid Build Coastguard Worker    }
992*61046927SAndroid Build Coastguard Worker    if (args->features.integer_dot_product) {
993*61046927SAndroid Build Coastguard Worker       c->getPreprocessorOpts().addMacroDef("cl_khr_integer_dot_product=1");
994*61046927SAndroid Build Coastguard Worker       c->getPreprocessorOpts().addMacroDef("__opencl_c_integer_dot_product_input_4x8bit_packed=1");
995*61046927SAndroid Build Coastguard Worker       c->getPreprocessorOpts().addMacroDef("__opencl_c_integer_dot_product_input_4x8bit=1");
996*61046927SAndroid Build Coastguard Worker    }
997*61046927SAndroid Build Coastguard Worker 
998*61046927SAndroid Build Coastguard Worker    // Add opencl include
999*61046927SAndroid Build Coastguard Worker    c->getPreprocessorOpts().Includes.push_back("opencl-c-base.h");
1000*61046927SAndroid Build Coastguard Worker    if (needs_opencl_c_h) {
1001*61046927SAndroid Build Coastguard Worker       c->getPreprocessorOpts().Includes.push_back("opencl-c.h");
1002*61046927SAndroid Build Coastguard Worker    }
1003*61046927SAndroid Build Coastguard Worker 
1004*61046927SAndroid Build Coastguard Worker    if (args->num_headers) {
1005*61046927SAndroid Build Coastguard Worker       ::llvm::SmallString<128> tmp_header_path;
1006*61046927SAndroid Build Coastguard Worker       ::llvm::sys::path::system_temp_directory(true, tmp_header_path);
1007*61046927SAndroid Build Coastguard Worker       ::llvm::sys::path::append(tmp_header_path, "openclon12");
1008*61046927SAndroid Build Coastguard Worker 
1009*61046927SAndroid Build Coastguard Worker       c->getHeaderSearchOpts().AddPath(tmp_header_path.str(),
1010*61046927SAndroid Build Coastguard Worker                                        clang::frontend::Quoted,
1011*61046927SAndroid Build Coastguard Worker                                        false, false);
1012*61046927SAndroid Build Coastguard Worker 
1013*61046927SAndroid Build Coastguard Worker       for (size_t i = 0; i < args->num_headers; i++) {
1014*61046927SAndroid Build Coastguard Worker          auto path_copy = tmp_header_path;
1015*61046927SAndroid Build Coastguard Worker          ::llvm::sys::path::append(path_copy, ::llvm::sys::path::convert_to_slash(args->headers[i].name));
1016*61046927SAndroid Build Coastguard Worker          c->getPreprocessorOpts().addRemappedFile(path_copy.str(),
1017*61046927SAndroid Build Coastguard Worker             ::llvm::MemoryBuffer::getMemBufferCopy(args->headers[i].value).release());
1018*61046927SAndroid Build Coastguard Worker       }
1019*61046927SAndroid Build Coastguard Worker    }
1020*61046927SAndroid Build Coastguard Worker 
1021*61046927SAndroid Build Coastguard Worker    c->getPreprocessorOpts().addRemappedFile(
1022*61046927SAndroid Build Coastguard Worker            args->source.name,
1023*61046927SAndroid Build Coastguard Worker            ::llvm::MemoryBuffer::getMemBufferCopy(std::string(args->source.value)).release());
1024*61046927SAndroid Build Coastguard Worker 
1025*61046927SAndroid Build Coastguard Worker    // Compile the code
1026*61046927SAndroid Build Coastguard Worker    clang::EmitLLVMOnlyAction act(&llvm_ctx);
1027*61046927SAndroid Build Coastguard Worker    if (!c->ExecuteAction(act)) {
1028*61046927SAndroid Build Coastguard Worker       clc_error(logger, "%sError executing LLVM compilation action.\n",
1029*61046927SAndroid Build Coastguard Worker                 diag_log_str.c_str());
1030*61046927SAndroid Build Coastguard Worker       return {};
1031*61046927SAndroid Build Coastguard Worker    }
1032*61046927SAndroid Build Coastguard Worker 
1033*61046927SAndroid Build Coastguard Worker    auto mod = act.takeModule();
1034*61046927SAndroid Build Coastguard Worker 
1035*61046927SAndroid Build Coastguard Worker    if (clc_debug_flags() & CLC_DEBUG_DUMP_LLVM)
1036*61046927SAndroid Build Coastguard Worker       clc_dump_llvm(mod.get(), stdout);
1037*61046927SAndroid Build Coastguard Worker 
1038*61046927SAndroid Build Coastguard Worker    return mod;
1039*61046927SAndroid Build Coastguard Worker }
1040*61046927SAndroid Build Coastguard Worker 
1041*61046927SAndroid Build Coastguard Worker static SPIRV::VersionNumber
1042*61046927SAndroid Build Coastguard Worker spirv_version_to_llvm_spirv_translator_version(enum clc_spirv_version version)
1043*61046927SAndroid Build Coastguard Worker {
1044*61046927SAndroid Build Coastguard Worker    switch (version) {
1045*61046927SAndroid Build Coastguard Worker    case CLC_SPIRV_VERSION_MAX: return SPIRV::VersionNumber::MaximumVersion;
1046*61046927SAndroid Build Coastguard Worker    case CLC_SPIRV_VERSION_1_0: return SPIRV::VersionNumber::SPIRV_1_0;
1047*61046927SAndroid Build Coastguard Worker    case CLC_SPIRV_VERSION_1_1: return SPIRV::VersionNumber::SPIRV_1_1;
1048*61046927SAndroid Build Coastguard Worker    case CLC_SPIRV_VERSION_1_2: return SPIRV::VersionNumber::SPIRV_1_2;
1049*61046927SAndroid Build Coastguard Worker    case CLC_SPIRV_VERSION_1_3: return SPIRV::VersionNumber::SPIRV_1_3;
1050*61046927SAndroid Build Coastguard Worker    case CLC_SPIRV_VERSION_1_4: return SPIRV::VersionNumber::SPIRV_1_4;
1051*61046927SAndroid Build Coastguard Worker    default:      return invalid_spirv_trans_version;
1052*61046927SAndroid Build Coastguard Worker    }
1053*61046927SAndroid Build Coastguard Worker }
1054*61046927SAndroid Build Coastguard Worker 
1055*61046927SAndroid Build Coastguard Worker static int
1056*61046927SAndroid Build Coastguard Worker llvm_mod_to_spirv(std::unique_ptr<::llvm::Module> mod,
1057*61046927SAndroid Build Coastguard Worker                   LLVMContext &context,
1058*61046927SAndroid Build Coastguard Worker                   const struct clc_compile_args *args,
1059*61046927SAndroid Build Coastguard Worker                   const struct clc_logger *logger,
1060*61046927SAndroid Build Coastguard Worker                   struct clc_binary *out_spirv)
1061*61046927SAndroid Build Coastguard Worker {
1062*61046927SAndroid Build Coastguard Worker    std::string log;
1063*61046927SAndroid Build Coastguard Worker 
1064*61046927SAndroid Build Coastguard Worker    SPIRV::VersionNumber version =
1065*61046927SAndroid Build Coastguard Worker       spirv_version_to_llvm_spirv_translator_version(args->spirv_version);
1066*61046927SAndroid Build Coastguard Worker    if (version == invalid_spirv_trans_version) {
1067*61046927SAndroid Build Coastguard Worker       clc_error(logger, "Invalid/unsupported SPIRV specified.\n");
1068*61046927SAndroid Build Coastguard Worker       return -1;
1069*61046927SAndroid Build Coastguard Worker    }
1070*61046927SAndroid Build Coastguard Worker 
1071*61046927SAndroid Build Coastguard Worker    const char *const *extensions = args->allowed_spirv_extensions;
1072*61046927SAndroid Build Coastguard Worker    if (!extensions) {
1073*61046927SAndroid Build Coastguard Worker       /* The SPIR-V parser doesn't handle all extensions */
1074*61046927SAndroid Build Coastguard Worker       static const char *default_extensions[] = {
1075*61046927SAndroid Build Coastguard Worker          "SPV_EXT_shader_atomic_float_add",
1076*61046927SAndroid Build Coastguard Worker          "SPV_EXT_shader_atomic_float_min_max",
1077*61046927SAndroid Build Coastguard Worker          "SPV_KHR_float_controls",
1078*61046927SAndroid Build Coastguard Worker          NULL,
1079*61046927SAndroid Build Coastguard Worker       };
1080*61046927SAndroid Build Coastguard Worker       extensions = default_extensions;
1081*61046927SAndroid Build Coastguard Worker    }
1082*61046927SAndroid Build Coastguard Worker 
1083*61046927SAndroid Build Coastguard Worker    SPIRV::TranslatorOpts::ExtensionsStatusMap ext_map;
1084*61046927SAndroid Build Coastguard Worker    for (int i = 0; extensions[i]; i++) {
1085*61046927SAndroid Build Coastguard Worker #define EXT(X) \
1086*61046927SAndroid Build Coastguard Worker       if (strcmp(#X, extensions[i]) == 0) \
1087*61046927SAndroid Build Coastguard Worker          ext_map.insert(std::make_pair(SPIRV::ExtensionID::X, true));
1088*61046927SAndroid Build Coastguard Worker #include "LLVMSPIRVLib/LLVMSPIRVExtensions.inc"
1089*61046927SAndroid Build Coastguard Worker #undef EXT
1090*61046927SAndroid Build Coastguard Worker    }
1091*61046927SAndroid Build Coastguard Worker    SPIRV::TranslatorOpts spirv_opts = SPIRV::TranslatorOpts(version, ext_map);
1092*61046927SAndroid Build Coastguard Worker 
1093*61046927SAndroid Build Coastguard Worker    /* This was the default in 12.0 and older, but currently we'll fail to parse without this */
1094*61046927SAndroid Build Coastguard Worker    spirv_opts.setPreserveOCLKernelArgTypeMetadataThroughString(true);
1095*61046927SAndroid Build Coastguard Worker 
1096*61046927SAndroid Build Coastguard Worker #if LLVM_VERSION_MAJOR >= 17
1097*61046927SAndroid Build Coastguard Worker    if (args->use_llvm_spirv_target) {
1098*61046927SAndroid Build Coastguard Worker       const char *triple = args->address_bits == 32 ? "spirv-unknown-unknown" : "spirv64-unknown-unknown";
1099*61046927SAndroid Build Coastguard Worker       std::string error_msg("");
1100*61046927SAndroid Build Coastguard Worker       auto target = TargetRegistry::lookupTarget(triple, error_msg);
1101*61046927SAndroid Build Coastguard Worker       if (target) {
1102*61046927SAndroid Build Coastguard Worker          auto TM = target->createTargetMachine(
1103*61046927SAndroid Build Coastguard Worker             triple, "", "", {}, std::nullopt, std::nullopt,
1104*61046927SAndroid Build Coastguard Worker #if LLVM_VERSION_MAJOR >= 18
1105*61046927SAndroid Build Coastguard Worker             ::llvm::CodeGenOptLevel::None
1106*61046927SAndroid Build Coastguard Worker #else
1107*61046927SAndroid Build Coastguard Worker             ::llvm::CodeGenOpt::None
1108*61046927SAndroid Build Coastguard Worker #endif
1109*61046927SAndroid Build Coastguard Worker          );
1110*61046927SAndroid Build Coastguard Worker 
1111*61046927SAndroid Build Coastguard Worker          auto PM = PassManager();
1112*61046927SAndroid Build Coastguard Worker          ::llvm::SmallVector<char> buf;
1113*61046927SAndroid Build Coastguard Worker          auto OS = ::llvm::raw_svector_ostream(buf);
1114*61046927SAndroid Build Coastguard Worker          TM->addPassesToEmitFile(
1115*61046927SAndroid Build Coastguard Worker             PM, OS, nullptr,
1116*61046927SAndroid Build Coastguard Worker #if LLVM_VERSION_MAJOR >= 18
1117*61046927SAndroid Build Coastguard Worker             ::llvm::CodeGenFileType::ObjectFile
1118*61046927SAndroid Build Coastguard Worker #else
1119*61046927SAndroid Build Coastguard Worker             ::llvm::CGFT_ObjectFile
1120*61046927SAndroid Build Coastguard Worker #endif
1121*61046927SAndroid Build Coastguard Worker          );
1122*61046927SAndroid Build Coastguard Worker 
1123*61046927SAndroid Build Coastguard Worker          PM.run(*mod);
1124*61046927SAndroid Build Coastguard Worker 
1125*61046927SAndroid Build Coastguard Worker          out_spirv->size = buf.size_in_bytes();
1126*61046927SAndroid Build Coastguard Worker          out_spirv->data = malloc(out_spirv->size);
1127*61046927SAndroid Build Coastguard Worker          memcpy(out_spirv->data, buf.data(), out_spirv->size);
1128*61046927SAndroid Build Coastguard Worker          return 0;
1129*61046927SAndroid Build Coastguard Worker       } else {
1130*61046927SAndroid Build Coastguard Worker          clc_error(logger, "LLVM SPIR-V target not found.\n");
1131*61046927SAndroid Build Coastguard Worker          return -1;
1132*61046927SAndroid Build Coastguard Worker       }
1133*61046927SAndroid Build Coastguard Worker    }
1134*61046927SAndroid Build Coastguard Worker #endif
1135*61046927SAndroid Build Coastguard Worker 
1136*61046927SAndroid Build Coastguard Worker    std::ostringstream spv_stream;
1137*61046927SAndroid Build Coastguard Worker    if (!::llvm::writeSpirv(mod.get(), spirv_opts, spv_stream, log)) {
1138*61046927SAndroid Build Coastguard Worker       clc_error(logger, "%sTranslation from LLVM IR to SPIR-V failed.\n",
1139*61046927SAndroid Build Coastguard Worker                 log.c_str());
1140*61046927SAndroid Build Coastguard Worker       return -1;
1141*61046927SAndroid Build Coastguard Worker    }
1142*61046927SAndroid Build Coastguard Worker 
1143*61046927SAndroid Build Coastguard Worker    const std::string spv_out = spv_stream.str();
1144*61046927SAndroid Build Coastguard Worker    out_spirv->size = spv_out.size();
1145*61046927SAndroid Build Coastguard Worker    out_spirv->data = malloc(out_spirv->size);
1146*61046927SAndroid Build Coastguard Worker    memcpy(out_spirv->data, spv_out.data(), out_spirv->size);
1147*61046927SAndroid Build Coastguard Worker 
1148*61046927SAndroid Build Coastguard Worker    return 0;
1149*61046927SAndroid Build Coastguard Worker }
1150*61046927SAndroid Build Coastguard Worker 
1151*61046927SAndroid Build Coastguard Worker int
1152*61046927SAndroid Build Coastguard Worker clc_c_to_spir(const struct clc_compile_args *args,
1153*61046927SAndroid Build Coastguard Worker               const struct clc_logger *logger,
1154*61046927SAndroid Build Coastguard Worker               struct clc_binary *out_spir)
1155*61046927SAndroid Build Coastguard Worker {
1156*61046927SAndroid Build Coastguard Worker    clc_initialize_llvm();
1157*61046927SAndroid Build Coastguard Worker 
1158*61046927SAndroid Build Coastguard Worker    LLVMContext llvm_ctx;
1159*61046927SAndroid Build Coastguard Worker    llvm_ctx.setDiagnosticHandlerCallBack(llvm_log_handler,
1160*61046927SAndroid Build Coastguard Worker                                          const_cast<clc_logger *>(logger));
1161*61046927SAndroid Build Coastguard Worker 
1162*61046927SAndroid Build Coastguard Worker    auto mod = clc_compile_to_llvm_module(llvm_ctx, args, logger);
1163*61046927SAndroid Build Coastguard Worker    if (!mod)
1164*61046927SAndroid Build Coastguard Worker       return -1;
1165*61046927SAndroid Build Coastguard Worker 
1166*61046927SAndroid Build Coastguard Worker    ::llvm::SmallVector<char, 0> buffer;
1167*61046927SAndroid Build Coastguard Worker    ::llvm::BitcodeWriter writer(buffer);
1168*61046927SAndroid Build Coastguard Worker    writer.writeModule(*mod);
1169*61046927SAndroid Build Coastguard Worker 
1170*61046927SAndroid Build Coastguard Worker    out_spir->size = buffer.size_in_bytes();
1171*61046927SAndroid Build Coastguard Worker    out_spir->data = malloc(out_spir->size);
1172*61046927SAndroid Build Coastguard Worker    memcpy(out_spir->data, buffer.data(), out_spir->size);
1173*61046927SAndroid Build Coastguard Worker 
1174*61046927SAndroid Build Coastguard Worker    return 0;
1175*61046927SAndroid Build Coastguard Worker }
1176*61046927SAndroid Build Coastguard Worker 
1177*61046927SAndroid Build Coastguard Worker int
1178*61046927SAndroid Build Coastguard Worker clc_c_to_spirv(const struct clc_compile_args *args,
1179*61046927SAndroid Build Coastguard Worker                const struct clc_logger *logger,
1180*61046927SAndroid Build Coastguard Worker                struct clc_binary *out_spirv)
1181*61046927SAndroid Build Coastguard Worker {
1182*61046927SAndroid Build Coastguard Worker    clc_initialize_llvm();
1183*61046927SAndroid Build Coastguard Worker 
1184*61046927SAndroid Build Coastguard Worker    LLVMContext llvm_ctx;
1185*61046927SAndroid Build Coastguard Worker    llvm_ctx.setDiagnosticHandlerCallBack(llvm_log_handler,
1186*61046927SAndroid Build Coastguard Worker                                          const_cast<clc_logger *>(logger));
1187*61046927SAndroid Build Coastguard Worker 
1188*61046927SAndroid Build Coastguard Worker    auto mod = clc_compile_to_llvm_module(llvm_ctx, args, logger);
1189*61046927SAndroid Build Coastguard Worker    if (!mod)
1190*61046927SAndroid Build Coastguard Worker       return -1;
1191*61046927SAndroid Build Coastguard Worker    return llvm_mod_to_spirv(std::move(mod), llvm_ctx, args, logger, out_spirv);
1192*61046927SAndroid Build Coastguard Worker }
1193*61046927SAndroid Build Coastguard Worker 
1194*61046927SAndroid Build Coastguard Worker int
1195*61046927SAndroid Build Coastguard Worker clc_spir_to_spirv(const struct clc_binary *in_spir,
1196*61046927SAndroid Build Coastguard Worker                   const struct clc_logger *logger,
1197*61046927SAndroid Build Coastguard Worker                   struct clc_binary *out_spirv)
1198*61046927SAndroid Build Coastguard Worker {
1199*61046927SAndroid Build Coastguard Worker    clc_initialize_llvm();
1200*61046927SAndroid Build Coastguard Worker 
1201*61046927SAndroid Build Coastguard Worker    LLVMContext llvm_ctx;
1202*61046927SAndroid Build Coastguard Worker    llvm_ctx.setDiagnosticHandlerCallBack(llvm_log_handler,
1203*61046927SAndroid Build Coastguard Worker                                          const_cast<clc_logger *>(logger));
1204*61046927SAndroid Build Coastguard Worker 
1205*61046927SAndroid Build Coastguard Worker    ::llvm::StringRef spir_ref(static_cast<const char*>(in_spir->data), in_spir->size);
1206*61046927SAndroid Build Coastguard Worker    auto mod = ::llvm::parseBitcodeFile(::llvm::MemoryBufferRef(spir_ref, "<spir>"), llvm_ctx);
1207*61046927SAndroid Build Coastguard Worker    if (!mod)
1208*61046927SAndroid Build Coastguard Worker       return -1;
1209*61046927SAndroid Build Coastguard Worker 
1210*61046927SAndroid Build Coastguard Worker    return llvm_mod_to_spirv(std::move(mod.get()), llvm_ctx, NULL, logger, out_spirv);
1211*61046927SAndroid Build Coastguard Worker }
1212*61046927SAndroid Build Coastguard Worker 
1213*61046927SAndroid Build Coastguard Worker class SPIRVMessageConsumer {
1214*61046927SAndroid Build Coastguard Worker public:
1215*61046927SAndroid Build Coastguard Worker    SPIRVMessageConsumer(const struct clc_logger *logger): logger(logger) {}
1216*61046927SAndroid Build Coastguard Worker 
1217*61046927SAndroid Build Coastguard Worker    void operator()(spv_message_level_t level, const char *src,
1218*61046927SAndroid Build Coastguard Worker                    const spv_position_t &pos, const char *msg)
1219*61046927SAndroid Build Coastguard Worker    {
1220*61046927SAndroid Build Coastguard Worker       if (level == SPV_MSG_INFO || level == SPV_MSG_DEBUG)
1221*61046927SAndroid Build Coastguard Worker          return;
1222*61046927SAndroid Build Coastguard Worker 
1223*61046927SAndroid Build Coastguard Worker       std::ostringstream message;
1224*61046927SAndroid Build Coastguard Worker       message << "(file=" << (src ? src : "input")
1225*61046927SAndroid Build Coastguard Worker               << ",line=" << pos.line
1226*61046927SAndroid Build Coastguard Worker               << ",column=" << pos.column
1227*61046927SAndroid Build Coastguard Worker               << ",index=" << pos.index
1228*61046927SAndroid Build Coastguard Worker               << "): " << msg << "\n";
1229*61046927SAndroid Build Coastguard Worker 
1230*61046927SAndroid Build Coastguard Worker       if (level == SPV_MSG_WARNING)
1231*61046927SAndroid Build Coastguard Worker          clc_warning(logger, "%s", message.str().c_str());
1232*61046927SAndroid Build Coastguard Worker       else
1233*61046927SAndroid Build Coastguard Worker          clc_error(logger, "%s", message.str().c_str());
1234*61046927SAndroid Build Coastguard Worker    }
1235*61046927SAndroid Build Coastguard Worker 
1236*61046927SAndroid Build Coastguard Worker private:
1237*61046927SAndroid Build Coastguard Worker    const struct clc_logger *logger;
1238*61046927SAndroid Build Coastguard Worker };
1239*61046927SAndroid Build Coastguard Worker 
1240*61046927SAndroid Build Coastguard Worker int
1241*61046927SAndroid Build Coastguard Worker clc_link_spirv_binaries(const struct clc_linker_args *args,
1242*61046927SAndroid Build Coastguard Worker                         const struct clc_logger *logger,
1243*61046927SAndroid Build Coastguard Worker                         struct clc_binary *out_spirv)
1244*61046927SAndroid Build Coastguard Worker {
1245*61046927SAndroid Build Coastguard Worker    std::vector<std::vector<uint32_t>> binaries;
1246*61046927SAndroid Build Coastguard Worker 
1247*61046927SAndroid Build Coastguard Worker    for (unsigned i = 0; i < args->num_in_objs; i++) {
1248*61046927SAndroid Build Coastguard Worker       const uint32_t *data = static_cast<const uint32_t *>(args->in_objs[i]->data);
1249*61046927SAndroid Build Coastguard Worker       std::vector<uint32_t> bin(data, data + (args->in_objs[i]->size / 4));
1250*61046927SAndroid Build Coastguard Worker       binaries.push_back(bin);
1251*61046927SAndroid Build Coastguard Worker    }
1252*61046927SAndroid Build Coastguard Worker 
1253*61046927SAndroid Build Coastguard Worker    SPIRVMessageConsumer msgconsumer(logger);
1254*61046927SAndroid Build Coastguard Worker    spvtools::Context context(spirv_target);
1255*61046927SAndroid Build Coastguard Worker    context.SetMessageConsumer(msgconsumer);
1256*61046927SAndroid Build Coastguard Worker    spvtools::LinkerOptions options;
1257*61046927SAndroid Build Coastguard Worker    options.SetAllowPartialLinkage(args->create_library);
1258*61046927SAndroid Build Coastguard Worker    #if defined(HAS_SPIRV_LINK_LLVM_WORKAROUND) && LLVM_VERSION_MAJOR >= 17
1259*61046927SAndroid Build Coastguard Worker       options.SetAllowPtrTypeMismatch(true);
1260*61046927SAndroid Build Coastguard Worker    #endif
1261*61046927SAndroid Build Coastguard Worker    options.SetCreateLibrary(args->create_library);
1262*61046927SAndroid Build Coastguard Worker    std::vector<uint32_t> linkingResult;
1263*61046927SAndroid Build Coastguard Worker    spv_result_t status = spvtools::Link(context, binaries, &linkingResult, options);
1264*61046927SAndroid Build Coastguard Worker    if (status != SPV_SUCCESS) {
1265*61046927SAndroid Build Coastguard Worker       #if !defined(HAS_SPIRV_LINK_LLVM_WORKAROUND) && LLVM_VERSION_MAJOR >= 17
1266*61046927SAndroid Build Coastguard Worker         clc_warning(logger, "SPIRV-Tools doesn't contain https://github.com/KhronosGroup/SPIRV-Tools/pull/5534\n");
1267*61046927SAndroid Build Coastguard Worker         clc_warning(logger, "Please update in order to prevent spurious linking failures\n");
1268*61046927SAndroid Build Coastguard Worker       #endif
1269*61046927SAndroid Build Coastguard Worker       return -1;
1270*61046927SAndroid Build Coastguard Worker    }
1271*61046927SAndroid Build Coastguard Worker 
1272*61046927SAndroid Build Coastguard Worker    out_spirv->size = linkingResult.size() * 4;
1273*61046927SAndroid Build Coastguard Worker    out_spirv->data = static_cast<uint32_t *>(malloc(out_spirv->size));
1274*61046927SAndroid Build Coastguard Worker    memcpy(out_spirv->data, linkingResult.data(), out_spirv->size);
1275*61046927SAndroid Build Coastguard Worker 
1276*61046927SAndroid Build Coastguard Worker    return 0;
1277*61046927SAndroid Build Coastguard Worker }
1278*61046927SAndroid Build Coastguard Worker 
1279*61046927SAndroid Build Coastguard Worker bool
1280*61046927SAndroid Build Coastguard Worker clc_validate_spirv(const struct clc_binary *spirv,
1281*61046927SAndroid Build Coastguard Worker                    const struct clc_logger *logger,
1282*61046927SAndroid Build Coastguard Worker                    const struct clc_validator_options *options)
1283*61046927SAndroid Build Coastguard Worker {
1284*61046927SAndroid Build Coastguard Worker    SPIRVMessageConsumer msgconsumer(logger);
1285*61046927SAndroid Build Coastguard Worker    spvtools::SpirvTools tools(spirv_target);
1286*61046927SAndroid Build Coastguard Worker    tools.SetMessageConsumer(msgconsumer);
1287*61046927SAndroid Build Coastguard Worker    spvtools::ValidatorOptions spirv_options;
1288*61046927SAndroid Build Coastguard Worker    const uint32_t *data = static_cast<const uint32_t *>(spirv->data);
1289*61046927SAndroid Build Coastguard Worker 
1290*61046927SAndroid Build Coastguard Worker    if (options) {
1291*61046927SAndroid Build Coastguard Worker       spirv_options.SetUniversalLimit(
1292*61046927SAndroid Build Coastguard Worker          spv_validator_limit_max_function_args,
1293*61046927SAndroid Build Coastguard Worker          options->limit_max_function_arg);
1294*61046927SAndroid Build Coastguard Worker    }
1295*61046927SAndroid Build Coastguard Worker 
1296*61046927SAndroid Build Coastguard Worker    return tools.Validate(data, spirv->size / 4, spirv_options);
1297*61046927SAndroid Build Coastguard Worker }
1298*61046927SAndroid Build Coastguard Worker 
1299*61046927SAndroid Build Coastguard Worker int
1300*61046927SAndroid Build Coastguard Worker clc_spirv_specialize(const struct clc_binary *in_spirv,
1301*61046927SAndroid Build Coastguard Worker                      const struct clc_parsed_spirv *parsed_data,
1302*61046927SAndroid Build Coastguard Worker                      const struct clc_spirv_specialization_consts *consts,
1303*61046927SAndroid Build Coastguard Worker                      struct clc_binary *out_spirv)
1304*61046927SAndroid Build Coastguard Worker {
1305*61046927SAndroid Build Coastguard Worker    std::unordered_map<uint32_t, std::vector<uint32_t>> spec_const_map;
1306*61046927SAndroid Build Coastguard Worker    for (unsigned i = 0; i < consts->num_specializations; ++i) {
1307*61046927SAndroid Build Coastguard Worker       unsigned id = consts->specializations[i].id;
1308*61046927SAndroid Build Coastguard Worker       auto parsed_spec_const = std::find_if(parsed_data->spec_constants,
1309*61046927SAndroid Build Coastguard Worker          parsed_data->spec_constants + parsed_data->num_spec_constants,
1310*61046927SAndroid Build Coastguard Worker          [id](const clc_parsed_spec_constant &c) { return c.id == id; });
1311*61046927SAndroid Build Coastguard Worker       assert(parsed_spec_const != parsed_data->spec_constants + parsed_data->num_spec_constants);
1312*61046927SAndroid Build Coastguard Worker 
1313*61046927SAndroid Build Coastguard Worker       std::vector<uint32_t> words;
1314*61046927SAndroid Build Coastguard Worker       switch (parsed_spec_const->type) {
1315*61046927SAndroid Build Coastguard Worker       case CLC_SPEC_CONSTANT_BOOL:
1316*61046927SAndroid Build Coastguard Worker          words.push_back(consts->specializations[i].value.b);
1317*61046927SAndroid Build Coastguard Worker          break;
1318*61046927SAndroid Build Coastguard Worker       case CLC_SPEC_CONSTANT_INT32:
1319*61046927SAndroid Build Coastguard Worker       case CLC_SPEC_CONSTANT_UINT32:
1320*61046927SAndroid Build Coastguard Worker       case CLC_SPEC_CONSTANT_FLOAT:
1321*61046927SAndroid Build Coastguard Worker          words.push_back(consts->specializations[i].value.u32);
1322*61046927SAndroid Build Coastguard Worker          break;
1323*61046927SAndroid Build Coastguard Worker       case CLC_SPEC_CONSTANT_INT16:
1324*61046927SAndroid Build Coastguard Worker          words.push_back((uint32_t)(int32_t)consts->specializations[i].value.i16);
1325*61046927SAndroid Build Coastguard Worker          break;
1326*61046927SAndroid Build Coastguard Worker       case CLC_SPEC_CONSTANT_INT8:
1327*61046927SAndroid Build Coastguard Worker          words.push_back((uint32_t)(int32_t)consts->specializations[i].value.i8);
1328*61046927SAndroid Build Coastguard Worker          break;
1329*61046927SAndroid Build Coastguard Worker       case CLC_SPEC_CONSTANT_UINT16:
1330*61046927SAndroid Build Coastguard Worker          words.push_back((uint32_t)consts->specializations[i].value.u16);
1331*61046927SAndroid Build Coastguard Worker          break;
1332*61046927SAndroid Build Coastguard Worker       case CLC_SPEC_CONSTANT_UINT8:
1333*61046927SAndroid Build Coastguard Worker          words.push_back((uint32_t)consts->specializations[i].value.u8);
1334*61046927SAndroid Build Coastguard Worker          break;
1335*61046927SAndroid Build Coastguard Worker       case CLC_SPEC_CONSTANT_DOUBLE:
1336*61046927SAndroid Build Coastguard Worker       case CLC_SPEC_CONSTANT_INT64:
1337*61046927SAndroid Build Coastguard Worker       case CLC_SPEC_CONSTANT_UINT64:
1338*61046927SAndroid Build Coastguard Worker          words.resize(2);
1339*61046927SAndroid Build Coastguard Worker          memcpy(words.data(), &consts->specializations[i].value.u64, 8);
1340*61046927SAndroid Build Coastguard Worker          break;
1341*61046927SAndroid Build Coastguard Worker       case CLC_SPEC_CONSTANT_UNKNOWN:
1342*61046927SAndroid Build Coastguard Worker          assert(0);
1343*61046927SAndroid Build Coastguard Worker          break;
1344*61046927SAndroid Build Coastguard Worker       }
1345*61046927SAndroid Build Coastguard Worker 
1346*61046927SAndroid Build Coastguard Worker       ASSERTED auto ret = spec_const_map.emplace(id, std::move(words));
1347*61046927SAndroid Build Coastguard Worker       assert(ret.second);
1348*61046927SAndroid Build Coastguard Worker    }
1349*61046927SAndroid Build Coastguard Worker 
1350*61046927SAndroid Build Coastguard Worker    spvtools::Optimizer opt(spirv_target);
1351*61046927SAndroid Build Coastguard Worker    opt.RegisterPass(spvtools::CreateSetSpecConstantDefaultValuePass(std::move(spec_const_map)));
1352*61046927SAndroid Build Coastguard Worker 
1353*61046927SAndroid Build Coastguard Worker    std::vector<uint32_t> result;
1354*61046927SAndroid Build Coastguard Worker    if (!opt.Run(static_cast<const uint32_t*>(in_spirv->data), in_spirv->size / 4, &result))
1355*61046927SAndroid Build Coastguard Worker       return false;
1356*61046927SAndroid Build Coastguard Worker 
1357*61046927SAndroid Build Coastguard Worker    out_spirv->size = result.size() * 4;
1358*61046927SAndroid Build Coastguard Worker    out_spirv->data = malloc(out_spirv->size);
1359*61046927SAndroid Build Coastguard Worker    memcpy(out_spirv->data, result.data(), out_spirv->size);
1360*61046927SAndroid Build Coastguard Worker    return true;
1361*61046927SAndroid Build Coastguard Worker }
1362*61046927SAndroid Build Coastguard Worker 
1363*61046927SAndroid Build Coastguard Worker static void
1364*61046927SAndroid Build Coastguard Worker clc_dump_llvm(const llvm::Module *mod, FILE *f)
1365*61046927SAndroid Build Coastguard Worker {
1366*61046927SAndroid Build Coastguard Worker    std::string out;
1367*61046927SAndroid Build Coastguard Worker    raw_string_ostream os(out);
1368*61046927SAndroid Build Coastguard Worker 
1369*61046927SAndroid Build Coastguard Worker    mod->print(os, nullptr);
1370*61046927SAndroid Build Coastguard Worker    os.flush();
1371*61046927SAndroid Build Coastguard Worker 
1372*61046927SAndroid Build Coastguard Worker    fwrite(out.c_str(), out.size(), 1, f);
1373*61046927SAndroid Build Coastguard Worker }
1374*61046927SAndroid Build Coastguard Worker 
1375*61046927SAndroid Build Coastguard Worker void
1376*61046927SAndroid Build Coastguard Worker clc_dump_spirv(const struct clc_binary *spvbin, FILE *f)
1377*61046927SAndroid Build Coastguard Worker {
1378*61046927SAndroid Build Coastguard Worker    spvtools::SpirvTools tools(spirv_target);
1379*61046927SAndroid Build Coastguard Worker    const uint32_t *data = static_cast<const uint32_t *>(spvbin->data);
1380*61046927SAndroid Build Coastguard Worker    std::vector<uint32_t> bin(data, data + (spvbin->size / 4));
1381*61046927SAndroid Build Coastguard Worker    std::string out;
1382*61046927SAndroid Build Coastguard Worker    tools.Disassemble(bin, &out,
1383*61046927SAndroid Build Coastguard Worker                      SPV_BINARY_TO_TEXT_OPTION_INDENT |
1384*61046927SAndroid Build Coastguard Worker                      SPV_BINARY_TO_TEXT_OPTION_FRIENDLY_NAMES);
1385*61046927SAndroid Build Coastguard Worker    fwrite(out.c_str(), out.size(), 1, f);
1386*61046927SAndroid Build Coastguard Worker }
1387*61046927SAndroid Build Coastguard Worker 
1388*61046927SAndroid Build Coastguard Worker void
1389*61046927SAndroid Build Coastguard Worker clc_free_spir_binary(struct clc_binary *spir)
1390*61046927SAndroid Build Coastguard Worker {
1391*61046927SAndroid Build Coastguard Worker    free(spir->data);
1392*61046927SAndroid Build Coastguard Worker }
1393*61046927SAndroid Build Coastguard Worker 
1394*61046927SAndroid Build Coastguard Worker void
1395*61046927SAndroid Build Coastguard Worker clc_free_spirv_binary(struct clc_binary *spvbin)
1396*61046927SAndroid Build Coastguard Worker {
1397*61046927SAndroid Build Coastguard Worker    free(spvbin->data);
1398*61046927SAndroid Build Coastguard Worker }
1399*61046927SAndroid Build Coastguard Worker 
1400*61046927SAndroid Build Coastguard Worker void
1401*61046927SAndroid Build Coastguard Worker initialize_llvm_once(void)
1402*61046927SAndroid Build Coastguard Worker {
1403*61046927SAndroid Build Coastguard Worker    LLVMInitializeAllTargets();
1404*61046927SAndroid Build Coastguard Worker    LLVMInitializeAllTargetInfos();
1405*61046927SAndroid Build Coastguard Worker    LLVMInitializeAllTargetMCs();
1406*61046927SAndroid Build Coastguard Worker    LLVMInitializeAllAsmParsers();
1407*61046927SAndroid Build Coastguard Worker    LLVMInitializeAllAsmPrinters();
1408*61046927SAndroid Build Coastguard Worker }
1409*61046927SAndroid Build Coastguard Worker 
1410*61046927SAndroid Build Coastguard Worker std::once_flag initialize_llvm_once_flag;
1411*61046927SAndroid Build Coastguard Worker 
1412*61046927SAndroid Build Coastguard Worker void
1413*61046927SAndroid Build Coastguard Worker clc_initialize_llvm(void)
1414*61046927SAndroid Build Coastguard Worker {
1415*61046927SAndroid Build Coastguard Worker    std::call_once(initialize_llvm_once_flag,
1416*61046927SAndroid Build Coastguard Worker                   []() { initialize_llvm_once(); });
1417*61046927SAndroid Build Coastguard Worker }
1418