• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2015-2020 The Khronos Group Inc.
2 // Modifications Copyright (C) 2020 Advanced Micro Devices, Inc. All rights
3 // reserved.
4 //
5 // Licensed under the Apache License, Version 2.0 (the "License");
6 // you may not use this file except in compliance with the License.
7 // You may obtain a copy of the License at
8 //
9 //     http://www.apache.org/licenses/LICENSE-2.0
10 //
11 // Unless required by applicable law or agreed to in writing, software
12 // distributed under the License is distributed on an "AS IS" BASIS,
13 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 // See the License for the specific language governing permissions and
15 // limitations under the License.
16 
17 #include "source/opcode.h"
18 
19 #include <assert.h>
20 #include <string.h>
21 
22 #include <algorithm>
23 #include <cstdlib>
24 
25 #include "source/instruction.h"
26 #include "source/macro.h"
27 #include "source/spirv_constant.h"
28 #include "source/spirv_endian.h"
29 #include "source/spirv_target_env.h"
30 #include "spirv-tools/libspirv.h"
31 
32 namespace {
33 struct OpcodeDescPtrLen {
34   const spv_opcode_desc_t* ptr;
35   uint32_t len;
36 };
37 
38 #include "core.insts-unified1.inc"
39 
40 static const spv_opcode_table_t kOpcodeTable = {ARRAY_SIZE(kOpcodeTableEntries),
41                                                 kOpcodeTableEntries};
42 
43 // Represents a vendor tool entry in the SPIR-V XML Regsitry.
44 struct VendorTool {
45   uint32_t value;
46   const char* vendor;
47   const char* tool;         // Might be empty string.
48   const char* vendor_tool;  // Combiantion of vendor and tool.
49 };
50 
51 const VendorTool vendor_tools[] = {
52 #include "generators.inc"
53 };
54 
55 }  // anonymous namespace
56 
57 // TODO(dneto): Move this to another file.  It doesn't belong with opcode
58 // processing.
spvGeneratorStr(uint32_t generator)59 const char* spvGeneratorStr(uint32_t generator) {
60   auto where = std::find_if(
61       std::begin(vendor_tools), std::end(vendor_tools),
62       [generator](const VendorTool& vt) { return generator == vt.value; });
63   if (where != std::end(vendor_tools)) return where->vendor_tool;
64   return "Unknown";
65 }
66 
spvOpcodeMake(uint16_t wordCount,SpvOp opcode)67 uint32_t spvOpcodeMake(uint16_t wordCount, SpvOp opcode) {
68   return ((uint32_t)opcode) | (((uint32_t)wordCount) << 16);
69 }
70 
spvOpcodeSplit(const uint32_t word,uint16_t * pWordCount,uint16_t * pOpcode)71 void spvOpcodeSplit(const uint32_t word, uint16_t* pWordCount,
72                     uint16_t* pOpcode) {
73   if (pWordCount) {
74     *pWordCount = (uint16_t)((0xffff0000 & word) >> 16);
75   }
76   if (pOpcode) {
77     *pOpcode = 0x0000ffff & word;
78   }
79 }
80 
spvOpcodeTableGet(spv_opcode_table * pInstTable,spv_target_env)81 spv_result_t spvOpcodeTableGet(spv_opcode_table* pInstTable, spv_target_env) {
82   if (!pInstTable) return SPV_ERROR_INVALID_POINTER;
83 
84   // Descriptions of each opcode.  Each entry describes the format of the
85   // instruction that follows a particular opcode.
86 
87   *pInstTable = &kOpcodeTable;
88   return SPV_SUCCESS;
89 }
90 
spvOpcodeTableNameLookup(spv_target_env env,const spv_opcode_table table,const char * name,spv_opcode_desc * pEntry)91 spv_result_t spvOpcodeTableNameLookup(spv_target_env env,
92                                       const spv_opcode_table table,
93                                       const char* name,
94                                       spv_opcode_desc* pEntry) {
95   if (!name || !pEntry) return SPV_ERROR_INVALID_POINTER;
96   if (!table) return SPV_ERROR_INVALID_TABLE;
97 
98   // TODO: This lookup of the Opcode table is suboptimal! Binary sort would be
99   // preferable but the table requires sorting on the Opcode name, but it's
100   // static const initialized and matches the order of the spec.
101   const size_t nameLength = strlen(name);
102   const auto version = spvVersionForTargetEnv(env);
103   for (uint64_t opcodeIndex = 0; opcodeIndex < table->count; ++opcodeIndex) {
104     const spv_opcode_desc_t& entry = table->entries[opcodeIndex];
105     // We considers the current opcode as available as long as
106     // 1. The target environment satisfies the minimal requirement of the
107     //    opcode; or
108     // 2. There is at least one extension enabling this opcode.
109     //
110     // Note that the second rule assumes the extension enabling this instruction
111     // is indeed requested in the SPIR-V code; checking that should be
112     // validator's work.
113     if (((version >= entry.minVersion && version <= entry.lastVersion) ||
114          entry.numExtensions > 0u || entry.numCapabilities > 0u) &&
115         nameLength == strlen(entry.name) &&
116         !strncmp(name, entry.name, nameLength)) {
117       // NOTE: Found out Opcode!
118       *pEntry = &entry;
119       return SPV_SUCCESS;
120     }
121   }
122 
123   return SPV_ERROR_INVALID_LOOKUP;
124 }
125 
spvOpcodeTableValueLookup(spv_target_env env,const spv_opcode_table table,const SpvOp opcode,spv_opcode_desc * pEntry)126 spv_result_t spvOpcodeTableValueLookup(spv_target_env env,
127                                        const spv_opcode_table table,
128                                        const SpvOp opcode,
129                                        spv_opcode_desc* pEntry) {
130   if (!table) return SPV_ERROR_INVALID_TABLE;
131   if (!pEntry) return SPV_ERROR_INVALID_POINTER;
132 
133   const auto beg = table->entries;
134   const auto end = table->entries + table->count;
135 
136   spv_opcode_desc_t needle = {"",    opcode, 0, nullptr, 0,   {},
137                               false, false,  0, nullptr, ~0u, ~0u};
138 
139   auto comp = [](const spv_opcode_desc_t& lhs, const spv_opcode_desc_t& rhs) {
140     return lhs.opcode < rhs.opcode;
141   };
142 
143   // We need to loop here because there can exist multiple symbols for the same
144   // opcode value, and they can be introduced in different target environments,
145   // which means they can have different minimal version requirements.
146   // Assumes the underlying table is already sorted ascendingly according to
147   // opcode value.
148   const auto version = spvVersionForTargetEnv(env);
149   for (auto it = std::lower_bound(beg, end, needle, comp);
150        it != end && it->opcode == opcode; ++it) {
151     // We considers the current opcode as available as long as
152     // 1. The target environment satisfies the minimal requirement of the
153     //    opcode; or
154     // 2. There is at least one extension enabling this opcode.
155     //
156     // Note that the second rule assumes the extension enabling this instruction
157     // is indeed requested in the SPIR-V code; checking that should be
158     // validator's work.
159     if ((version >= it->minVersion && version <= it->lastVersion) ||
160         it->numExtensions > 0u || it->numCapabilities > 0u) {
161       *pEntry = it;
162       return SPV_SUCCESS;
163     }
164   }
165 
166   return SPV_ERROR_INVALID_LOOKUP;
167 }
168 
spvInstructionCopy(const uint32_t * words,const SpvOp opcode,const uint16_t wordCount,const spv_endianness_t endian,spv_instruction_t * pInst)169 void spvInstructionCopy(const uint32_t* words, const SpvOp opcode,
170                         const uint16_t wordCount, const spv_endianness_t endian,
171                         spv_instruction_t* pInst) {
172   pInst->opcode = opcode;
173   pInst->words.resize(wordCount);
174   for (uint16_t wordIndex = 0; wordIndex < wordCount; ++wordIndex) {
175     pInst->words[wordIndex] = spvFixWord(words[wordIndex], endian);
176     if (!wordIndex) {
177       uint16_t thisWordCount;
178       uint16_t thisOpcode;
179       spvOpcodeSplit(pInst->words[wordIndex], &thisWordCount, &thisOpcode);
180       assert(opcode == static_cast<SpvOp>(thisOpcode) &&
181              wordCount == thisWordCount && "Endianness failed!");
182     }
183   }
184 }
185 
spvOpcodeString(const uint32_t opcode)186 const char* spvOpcodeString(const uint32_t opcode) {
187   const auto beg = kOpcodeTableEntries;
188   const auto end = kOpcodeTableEntries + ARRAY_SIZE(kOpcodeTableEntries);
189   spv_opcode_desc_t needle = {"",    static_cast<SpvOp>(opcode),
190                               0,     nullptr,
191                               0,     {},
192                               false, false,
193                               0,     nullptr,
194                               ~0u,   ~0u};
195   auto comp = [](const spv_opcode_desc_t& lhs, const spv_opcode_desc_t& rhs) {
196     return lhs.opcode < rhs.opcode;
197   };
198   auto it = std::lower_bound(beg, end, needle, comp);
199   if (it != end && it->opcode == opcode) {
200     return it->name;
201   }
202 
203   assert(0 && "Unreachable!");
204   return "unknown";
205 }
206 
spvOpcodeIsScalarType(const SpvOp opcode)207 int32_t spvOpcodeIsScalarType(const SpvOp opcode) {
208   switch (opcode) {
209     case SpvOpTypeInt:
210     case SpvOpTypeFloat:
211     case SpvOpTypeBool:
212       return true;
213     default:
214       return false;
215   }
216 }
217 
spvOpcodeIsSpecConstant(const SpvOp opcode)218 int32_t spvOpcodeIsSpecConstant(const SpvOp opcode) {
219   switch (opcode) {
220     case SpvOpSpecConstantTrue:
221     case SpvOpSpecConstantFalse:
222     case SpvOpSpecConstant:
223     case SpvOpSpecConstantComposite:
224     case SpvOpSpecConstantOp:
225       return true;
226     default:
227       return false;
228   }
229 }
230 
spvOpcodeIsConstant(const SpvOp opcode)231 int32_t spvOpcodeIsConstant(const SpvOp opcode) {
232   switch (opcode) {
233     case SpvOpConstantTrue:
234     case SpvOpConstantFalse:
235     case SpvOpConstant:
236     case SpvOpConstantComposite:
237     case SpvOpConstantSampler:
238     case SpvOpConstantNull:
239     case SpvOpSpecConstantTrue:
240     case SpvOpSpecConstantFalse:
241     case SpvOpSpecConstant:
242     case SpvOpSpecConstantComposite:
243     case SpvOpSpecConstantOp:
244       return true;
245     default:
246       return false;
247   }
248 }
249 
spvOpcodeIsConstantOrUndef(const SpvOp opcode)250 bool spvOpcodeIsConstantOrUndef(const SpvOp opcode) {
251   return opcode == SpvOpUndef || spvOpcodeIsConstant(opcode);
252 }
253 
spvOpcodeIsScalarSpecConstant(const SpvOp opcode)254 bool spvOpcodeIsScalarSpecConstant(const SpvOp opcode) {
255   switch (opcode) {
256     case SpvOpSpecConstantTrue:
257     case SpvOpSpecConstantFalse:
258     case SpvOpSpecConstant:
259       return true;
260     default:
261       return false;
262   }
263 }
264 
spvOpcodeIsComposite(const SpvOp opcode)265 int32_t spvOpcodeIsComposite(const SpvOp opcode) {
266   switch (opcode) {
267     case SpvOpTypeVector:
268     case SpvOpTypeMatrix:
269     case SpvOpTypeArray:
270     case SpvOpTypeStruct:
271     case SpvOpTypeCooperativeMatrixNV:
272       return true;
273     default:
274       return false;
275   }
276 }
277 
spvOpcodeReturnsLogicalVariablePointer(const SpvOp opcode)278 bool spvOpcodeReturnsLogicalVariablePointer(const SpvOp opcode) {
279   switch (opcode) {
280     case SpvOpVariable:
281     case SpvOpAccessChain:
282     case SpvOpInBoundsAccessChain:
283     case SpvOpFunctionParameter:
284     case SpvOpImageTexelPointer:
285     case SpvOpCopyObject:
286     case SpvOpSelect:
287     case SpvOpPhi:
288     case SpvOpFunctionCall:
289     case SpvOpPtrAccessChain:
290     case SpvOpLoad:
291     case SpvOpConstantNull:
292       return true;
293     default:
294       return false;
295   }
296 }
297 
spvOpcodeReturnsLogicalPointer(const SpvOp opcode)298 int32_t spvOpcodeReturnsLogicalPointer(const SpvOp opcode) {
299   switch (opcode) {
300     case SpvOpVariable:
301     case SpvOpAccessChain:
302     case SpvOpInBoundsAccessChain:
303     case SpvOpFunctionParameter:
304     case SpvOpImageTexelPointer:
305     case SpvOpCopyObject:
306       return true;
307     default:
308       return false;
309   }
310 }
311 
spvOpcodeGeneratesType(SpvOp op)312 int32_t spvOpcodeGeneratesType(SpvOp op) {
313   switch (op) {
314     case SpvOpTypeVoid:
315     case SpvOpTypeBool:
316     case SpvOpTypeInt:
317     case SpvOpTypeFloat:
318     case SpvOpTypeVector:
319     case SpvOpTypeMatrix:
320     case SpvOpTypeImage:
321     case SpvOpTypeSampler:
322     case SpvOpTypeSampledImage:
323     case SpvOpTypeArray:
324     case SpvOpTypeRuntimeArray:
325     case SpvOpTypeStruct:
326     case SpvOpTypeOpaque:
327     case SpvOpTypePointer:
328     case SpvOpTypeFunction:
329     case SpvOpTypeEvent:
330     case SpvOpTypeDeviceEvent:
331     case SpvOpTypeReserveId:
332     case SpvOpTypeQueue:
333     case SpvOpTypePipe:
334     case SpvOpTypePipeStorage:
335     case SpvOpTypeNamedBarrier:
336     case SpvOpTypeAccelerationStructureNV:
337     case SpvOpTypeCooperativeMatrixNV:
338     // case SpvOpTypeAccelerationStructureKHR: covered by
339     // SpvOpTypeAccelerationStructureNV
340     case SpvOpTypeRayQueryKHR:
341       return true;
342     default:
343       // In particular, OpTypeForwardPointer does not generate a type,
344       // but declares a storage class for a pointer type generated
345       // by a different instruction.
346       break;
347   }
348   return 0;
349 }
350 
spvOpcodeIsDecoration(const SpvOp opcode)351 bool spvOpcodeIsDecoration(const SpvOp opcode) {
352   switch (opcode) {
353     case SpvOpDecorate:
354     case SpvOpDecorateId:
355     case SpvOpMemberDecorate:
356     case SpvOpGroupDecorate:
357     case SpvOpGroupMemberDecorate:
358     case SpvOpDecorateStringGOOGLE:
359     case SpvOpMemberDecorateStringGOOGLE:
360       return true;
361     default:
362       break;
363   }
364   return false;
365 }
366 
spvOpcodeIsLoad(const SpvOp opcode)367 bool spvOpcodeIsLoad(const SpvOp opcode) {
368   switch (opcode) {
369     case SpvOpLoad:
370     case SpvOpImageSampleExplicitLod:
371     case SpvOpImageSampleImplicitLod:
372     case SpvOpImageSampleDrefImplicitLod:
373     case SpvOpImageSampleDrefExplicitLod:
374     case SpvOpImageSampleProjImplicitLod:
375     case SpvOpImageSampleProjExplicitLod:
376     case SpvOpImageSampleProjDrefImplicitLod:
377     case SpvOpImageSampleProjDrefExplicitLod:
378     case SpvOpImageFetch:
379     case SpvOpImageGather:
380     case SpvOpImageDrefGather:
381     case SpvOpImageRead:
382     case SpvOpImageSparseSampleImplicitLod:
383     case SpvOpImageSparseSampleExplicitLod:
384     case SpvOpImageSparseSampleDrefExplicitLod:
385     case SpvOpImageSparseSampleDrefImplicitLod:
386     case SpvOpImageSparseFetch:
387     case SpvOpImageSparseGather:
388     case SpvOpImageSparseDrefGather:
389     case SpvOpImageSparseRead:
390       return true;
391     default:
392       return false;
393   }
394 }
395 
spvOpcodeIsBranch(SpvOp opcode)396 bool spvOpcodeIsBranch(SpvOp opcode) {
397   switch (opcode) {
398     case SpvOpBranch:
399     case SpvOpBranchConditional:
400     case SpvOpSwitch:
401       return true;
402     default:
403       return false;
404   }
405 }
406 
spvOpcodeIsAtomicWithLoad(const SpvOp opcode)407 bool spvOpcodeIsAtomicWithLoad(const SpvOp opcode) {
408   switch (opcode) {
409     case SpvOpAtomicLoad:
410     case SpvOpAtomicExchange:
411     case SpvOpAtomicCompareExchange:
412     case SpvOpAtomicCompareExchangeWeak:
413     case SpvOpAtomicIIncrement:
414     case SpvOpAtomicIDecrement:
415     case SpvOpAtomicIAdd:
416     case SpvOpAtomicFAddEXT:
417     case SpvOpAtomicISub:
418     case SpvOpAtomicSMin:
419     case SpvOpAtomicUMin:
420     case SpvOpAtomicFMinEXT:
421     case SpvOpAtomicSMax:
422     case SpvOpAtomicUMax:
423     case SpvOpAtomicFMaxEXT:
424     case SpvOpAtomicAnd:
425     case SpvOpAtomicOr:
426     case SpvOpAtomicXor:
427     case SpvOpAtomicFlagTestAndSet:
428       return true;
429     default:
430       return false;
431   }
432 }
433 
spvOpcodeIsAtomicOp(const SpvOp opcode)434 bool spvOpcodeIsAtomicOp(const SpvOp opcode) {
435   return (spvOpcodeIsAtomicWithLoad(opcode) || opcode == SpvOpAtomicStore ||
436           opcode == SpvOpAtomicFlagClear);
437 }
438 
spvOpcodeIsReturn(SpvOp opcode)439 bool spvOpcodeIsReturn(SpvOp opcode) {
440   switch (opcode) {
441     case SpvOpReturn:
442     case SpvOpReturnValue:
443       return true;
444     default:
445       return false;
446   }
447 }
448 
spvOpcodeIsAbort(SpvOp opcode)449 bool spvOpcodeIsAbort(SpvOp opcode) {
450   switch (opcode) {
451     case SpvOpKill:
452     case SpvOpUnreachable:
453     case SpvOpTerminateInvocation:
454     case SpvOpTerminateRayKHR:
455     case SpvOpIgnoreIntersectionKHR:
456       return true;
457     default:
458       return false;
459   }
460 }
461 
spvOpcodeIsReturnOrAbort(SpvOp opcode)462 bool spvOpcodeIsReturnOrAbort(SpvOp opcode) {
463   return spvOpcodeIsReturn(opcode) || spvOpcodeIsAbort(opcode);
464 }
465 
spvOpcodeIsBlockTerminator(SpvOp opcode)466 bool spvOpcodeIsBlockTerminator(SpvOp opcode) {
467   return spvOpcodeIsBranch(opcode) || spvOpcodeIsReturnOrAbort(opcode);
468 }
469 
spvOpcodeTerminatesExecution(SpvOp opcode)470 bool spvOpcodeTerminatesExecution(SpvOp opcode) {
471   return opcode == SpvOpKill || opcode == SpvOpTerminateInvocation ||
472          opcode == SpvOpTerminateRayKHR || opcode == SpvOpIgnoreIntersectionKHR;
473 }
474 
spvOpcodeIsBaseOpaqueType(SpvOp opcode)475 bool spvOpcodeIsBaseOpaqueType(SpvOp opcode) {
476   switch (opcode) {
477     case SpvOpTypeImage:
478     case SpvOpTypeSampler:
479     case SpvOpTypeSampledImage:
480     case SpvOpTypeOpaque:
481     case SpvOpTypeEvent:
482     case SpvOpTypeDeviceEvent:
483     case SpvOpTypeReserveId:
484     case SpvOpTypeQueue:
485     case SpvOpTypePipe:
486     case SpvOpTypeForwardPointer:
487     case SpvOpTypePipeStorage:
488     case SpvOpTypeNamedBarrier:
489       return true;
490     default:
491       return false;
492   }
493 }
494 
spvOpcodeIsNonUniformGroupOperation(SpvOp opcode)495 bool spvOpcodeIsNonUniformGroupOperation(SpvOp opcode) {
496   switch (opcode) {
497     case SpvOpGroupNonUniformElect:
498     case SpvOpGroupNonUniformAll:
499     case SpvOpGroupNonUniformAny:
500     case SpvOpGroupNonUniformAllEqual:
501     case SpvOpGroupNonUniformBroadcast:
502     case SpvOpGroupNonUniformBroadcastFirst:
503     case SpvOpGroupNonUniformBallot:
504     case SpvOpGroupNonUniformInverseBallot:
505     case SpvOpGroupNonUniformBallotBitExtract:
506     case SpvOpGroupNonUniformBallotBitCount:
507     case SpvOpGroupNonUniformBallotFindLSB:
508     case SpvOpGroupNonUniformBallotFindMSB:
509     case SpvOpGroupNonUniformShuffle:
510     case SpvOpGroupNonUniformShuffleXor:
511     case SpvOpGroupNonUniformShuffleUp:
512     case SpvOpGroupNonUniformShuffleDown:
513     case SpvOpGroupNonUniformIAdd:
514     case SpvOpGroupNonUniformFAdd:
515     case SpvOpGroupNonUniformIMul:
516     case SpvOpGroupNonUniformFMul:
517     case SpvOpGroupNonUniformSMin:
518     case SpvOpGroupNonUniformUMin:
519     case SpvOpGroupNonUniformFMin:
520     case SpvOpGroupNonUniformSMax:
521     case SpvOpGroupNonUniformUMax:
522     case SpvOpGroupNonUniformFMax:
523     case SpvOpGroupNonUniformBitwiseAnd:
524     case SpvOpGroupNonUniformBitwiseOr:
525     case SpvOpGroupNonUniformBitwiseXor:
526     case SpvOpGroupNonUniformLogicalAnd:
527     case SpvOpGroupNonUniformLogicalOr:
528     case SpvOpGroupNonUniformLogicalXor:
529     case SpvOpGroupNonUniformQuadBroadcast:
530     case SpvOpGroupNonUniformQuadSwap:
531       return true;
532     default:
533       return false;
534   }
535 }
536 
spvOpcodeIsScalarizable(SpvOp opcode)537 bool spvOpcodeIsScalarizable(SpvOp opcode) {
538   switch (opcode) {
539     case SpvOpPhi:
540     case SpvOpCopyObject:
541     case SpvOpConvertFToU:
542     case SpvOpConvertFToS:
543     case SpvOpConvertSToF:
544     case SpvOpConvertUToF:
545     case SpvOpUConvert:
546     case SpvOpSConvert:
547     case SpvOpFConvert:
548     case SpvOpQuantizeToF16:
549     case SpvOpVectorInsertDynamic:
550     case SpvOpSNegate:
551     case SpvOpFNegate:
552     case SpvOpIAdd:
553     case SpvOpFAdd:
554     case SpvOpISub:
555     case SpvOpFSub:
556     case SpvOpIMul:
557     case SpvOpFMul:
558     case SpvOpUDiv:
559     case SpvOpSDiv:
560     case SpvOpFDiv:
561     case SpvOpUMod:
562     case SpvOpSRem:
563     case SpvOpSMod:
564     case SpvOpFRem:
565     case SpvOpFMod:
566     case SpvOpVectorTimesScalar:
567     case SpvOpIAddCarry:
568     case SpvOpISubBorrow:
569     case SpvOpUMulExtended:
570     case SpvOpSMulExtended:
571     case SpvOpShiftRightLogical:
572     case SpvOpShiftRightArithmetic:
573     case SpvOpShiftLeftLogical:
574     case SpvOpBitwiseOr:
575     case SpvOpBitwiseAnd:
576     case SpvOpNot:
577     case SpvOpBitFieldInsert:
578     case SpvOpBitFieldSExtract:
579     case SpvOpBitFieldUExtract:
580     case SpvOpBitReverse:
581     case SpvOpBitCount:
582     case SpvOpIsNan:
583     case SpvOpIsInf:
584     case SpvOpIsFinite:
585     case SpvOpIsNormal:
586     case SpvOpSignBitSet:
587     case SpvOpLessOrGreater:
588     case SpvOpOrdered:
589     case SpvOpUnordered:
590     case SpvOpLogicalEqual:
591     case SpvOpLogicalNotEqual:
592     case SpvOpLogicalOr:
593     case SpvOpLogicalAnd:
594     case SpvOpLogicalNot:
595     case SpvOpSelect:
596     case SpvOpIEqual:
597     case SpvOpINotEqual:
598     case SpvOpUGreaterThan:
599     case SpvOpSGreaterThan:
600     case SpvOpUGreaterThanEqual:
601     case SpvOpSGreaterThanEqual:
602     case SpvOpULessThan:
603     case SpvOpSLessThan:
604     case SpvOpULessThanEqual:
605     case SpvOpSLessThanEqual:
606     case SpvOpFOrdEqual:
607     case SpvOpFUnordEqual:
608     case SpvOpFOrdNotEqual:
609     case SpvOpFUnordNotEqual:
610     case SpvOpFOrdLessThan:
611     case SpvOpFUnordLessThan:
612     case SpvOpFOrdGreaterThan:
613     case SpvOpFUnordGreaterThan:
614     case SpvOpFOrdLessThanEqual:
615     case SpvOpFUnordLessThanEqual:
616     case SpvOpFOrdGreaterThanEqual:
617     case SpvOpFUnordGreaterThanEqual:
618       return true;
619     default:
620       return false;
621   }
622 }
623 
spvOpcodeIsDebug(SpvOp opcode)624 bool spvOpcodeIsDebug(SpvOp opcode) {
625   switch (opcode) {
626     case SpvOpName:
627     case SpvOpMemberName:
628     case SpvOpSource:
629     case SpvOpSourceContinued:
630     case SpvOpSourceExtension:
631     case SpvOpString:
632     case SpvOpLine:
633     case SpvOpNoLine:
634       return true;
635     default:
636       return false;
637   }
638 }
639 
spvOpcodeIsCommutativeBinaryOperator(SpvOp opcode)640 bool spvOpcodeIsCommutativeBinaryOperator(SpvOp opcode) {
641   switch (opcode) {
642     case SpvOpPtrEqual:
643     case SpvOpPtrNotEqual:
644     case SpvOpIAdd:
645     case SpvOpFAdd:
646     case SpvOpIMul:
647     case SpvOpFMul:
648     case SpvOpDot:
649     case SpvOpIAddCarry:
650     case SpvOpUMulExtended:
651     case SpvOpSMulExtended:
652     case SpvOpBitwiseOr:
653     case SpvOpBitwiseXor:
654     case SpvOpBitwiseAnd:
655     case SpvOpOrdered:
656     case SpvOpUnordered:
657     case SpvOpLogicalEqual:
658     case SpvOpLogicalNotEqual:
659     case SpvOpLogicalOr:
660     case SpvOpLogicalAnd:
661     case SpvOpIEqual:
662     case SpvOpINotEqual:
663     case SpvOpFOrdEqual:
664     case SpvOpFUnordEqual:
665     case SpvOpFOrdNotEqual:
666     case SpvOpFUnordNotEqual:
667       return true;
668     default:
669       return false;
670   }
671 }
672 
spvOpcodeIsLinearAlgebra(SpvOp opcode)673 bool spvOpcodeIsLinearAlgebra(SpvOp opcode) {
674   switch (opcode) {
675     case SpvOpTranspose:
676     case SpvOpVectorTimesScalar:
677     case SpvOpMatrixTimesScalar:
678     case SpvOpVectorTimesMatrix:
679     case SpvOpMatrixTimesVector:
680     case SpvOpMatrixTimesMatrix:
681     case SpvOpOuterProduct:
682     case SpvOpDot:
683       return true;
684     default:
685       return false;
686   }
687 }
688 
spvOpcodeIsImageSample(const SpvOp opcode)689 bool spvOpcodeIsImageSample(const SpvOp opcode) {
690   switch (opcode) {
691     case SpvOpImageSampleImplicitLod:
692     case SpvOpImageSampleExplicitLod:
693     case SpvOpImageSampleDrefImplicitLod:
694     case SpvOpImageSampleDrefExplicitLod:
695     case SpvOpImageSampleProjImplicitLod:
696     case SpvOpImageSampleProjExplicitLod:
697     case SpvOpImageSampleProjDrefImplicitLod:
698     case SpvOpImageSampleProjDrefExplicitLod:
699     case SpvOpImageSparseSampleImplicitLod:
700     case SpvOpImageSparseSampleExplicitLod:
701     case SpvOpImageSparseSampleDrefImplicitLod:
702     case SpvOpImageSparseSampleDrefExplicitLod:
703       return true;
704     default:
705       return false;
706   }
707 }
708 
spvOpcodeMemorySemanticsOperandIndices(SpvOp opcode)709 std::vector<uint32_t> spvOpcodeMemorySemanticsOperandIndices(SpvOp opcode) {
710   switch (opcode) {
711     case SpvOpMemoryBarrier:
712       return {1};
713     case SpvOpAtomicStore:
714     case SpvOpControlBarrier:
715     case SpvOpAtomicFlagClear:
716     case SpvOpMemoryNamedBarrier:
717       return {2};
718     case SpvOpAtomicLoad:
719     case SpvOpAtomicExchange:
720     case SpvOpAtomicIIncrement:
721     case SpvOpAtomicIDecrement:
722     case SpvOpAtomicIAdd:
723     case SpvOpAtomicFAddEXT:
724     case SpvOpAtomicISub:
725     case SpvOpAtomicSMin:
726     case SpvOpAtomicUMin:
727     case SpvOpAtomicSMax:
728     case SpvOpAtomicUMax:
729     case SpvOpAtomicAnd:
730     case SpvOpAtomicOr:
731     case SpvOpAtomicXor:
732     case SpvOpAtomicFlagTestAndSet:
733       return {4};
734     case SpvOpAtomicCompareExchange:
735     case SpvOpAtomicCompareExchangeWeak:
736       return {4, 5};
737     default:
738       return {};
739   }
740 }
741 
spvOpcodeIsAccessChain(SpvOp opcode)742 bool spvOpcodeIsAccessChain(SpvOp opcode) {
743   switch (opcode) {
744     case SpvOpAccessChain:
745     case SpvOpInBoundsAccessChain:
746     case SpvOpPtrAccessChain:
747     case SpvOpInBoundsPtrAccessChain:
748       return true;
749     default:
750       return false;
751   }
752 }
753 
spvOpcodeIsBit(SpvOp opcode)754 bool spvOpcodeIsBit(SpvOp opcode) {
755   switch (opcode) {
756     case SpvOpShiftRightLogical:
757     case SpvOpShiftRightArithmetic:
758     case SpvOpShiftLeftLogical:
759     case SpvOpBitwiseOr:
760     case SpvOpBitwiseXor:
761     case SpvOpBitwiseAnd:
762     case SpvOpNot:
763     case SpvOpBitReverse:
764     case SpvOpBitCount:
765       return true;
766     default:
767       return false;
768   }
769 }
770