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