blob: 0d1161a630c0b75142dd34aa3b5d57ee08f99741 [file] [log] [blame]
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +01001// Copyright (c) 2015 The Khronos Group Inc.
2//
3// Permission is hereby granted, free of charge, to any person obtaining a
4// copy of this software and/or associated documentation files (the
5// "Materials"), to deal in the Materials without restriction, including
6// without limitation the rights to use, copy, modify, merge, publish,
7// distribute, sublicense, and/or sell copies of the Materials, and to
8// permit persons to whom the Materials are furnished to do so, subject to
9// the following conditions:
10//
11// The above copyright notice and this permission notice shall be included
12// in all copies or substantial portions of the Materials.
13//
14// MODIFICATIONS TO THIS FILE MAY MEAN IT NO LONGER ACCURATELY REFLECTS
15// KHRONOS STANDARDS. THE UNMODIFIED, NORMATIVE VERSIONS OF KHRONOS
16// SPECIFICATIONS AND HEADER INFORMATION ARE LOCATED AT
17// https://www.khronos.org/registry/
18//
19// THE MATERIALS ARE PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
20// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
21// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
22// IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
23// CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
24// TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
25// MATERIALS OR THE USE OR OTHER DEALINGS IN THE MATERIALS.
26
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010027#include "opcode.h"
28
29#include <assert.h>
30#include <string.h>
31
Lei Zhang972788b2015-11-12 13:48:30 -050032#include <cstdlib>
33
Lei Zhang923f6c12015-11-11 12:45:23 -050034#include "endian.h"
35#include "instruction.h"
36#include "libspirv/libspirv.h"
Lei Zhangaa056cd2015-11-11 14:24:04 -050037#include "spirv_constant.h"
Lei Zhang4ac601a2015-11-11 17:29:23 -050038#include "spirv_operands.h"
Lei Zhang923f6c12015-11-11 12:45:23 -050039
David Neto78c3b432015-08-27 13:03:52 -040040namespace {
41
42// Descriptions of each opcode. Each entry describes the format of the
43// instruction that follows a particular opcode.
44//
45// Most fields are initialized statically by including an automatically
46// generated file.
47// The operandTypes fields are initialized during spvOpcodeInitialize().
48//
49// TODO(dneto): Some of the macros are quite unreadable. We could make
50// good use of constexpr functions, but some compilers don't support that yet.
Lei Zhang972788b2015-11-12 13:48:30 -050051const spv_opcode_desc_t opcodeTableEntries[] = {
Lei Zhang1a0334e2015-11-02 09:41:20 -050052#define EmptyList \
53 {}
54#define List(...) \
55 { __VA_ARGS__ }
Lei Zhangb36e7042015-10-28 13:40:52 -040056#define Capability(X) SPV_CAPABILITY_AS_MASK(SpvCapability##X)
Lei Zhang1a0334e2015-11-02 09:41:20 -050057#define Capability2(X, Y) Capability(X) | Capability(Y)
58#define SpvCapabilityNone \
59 0 // Needed so Capability(None) still expands to valid syntax.
David Neto201caf72015-11-04 17:38:17 -050060#define Instruction(Name, HasResult, HasType, NumLogicalOperands, \
61 NumCapabilities, CapabilityRequired, IsVariable, \
62 LogicalArgsList) \
63 {#Name, SpvOp##Name, (NumCapabilities) ? (CapabilityRequired) : 0, \
64 0, {}, /* Filled in later. Operand list, including \
65 result id and type id, if needed */ \
66 HasResult, HasType, LogicalArgsList},
David Neto78c3b432015-08-27 13:03:52 -040067#include "opcode.inc"
68#undef EmptyList
69#undef List
70#undef Capability
Dejan Mircevski205408b2015-09-30 16:42:34 -040071#undef Capability2
David Neto78c3b432015-08-27 13:03:52 -040072#undef CapabilityNone
73#undef Instruction
74};
75
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010076// Opcode API
77
David Neto78c3b432015-08-27 13:03:52 -040078// Converts the given operand class enum (from the SPIR-V document generation
David Neto0f166be2015-11-11 01:56:49 -050079// logic) to the operand type required by the parser. The SPV_OPERAND_TYPE_NONE
80// value indicates there is no current operand and no further operands.
David Neto78c3b432015-08-27 13:03:52 -040081// This only applies to logical operands.
Lei Zhangb36e7042015-10-28 13:40:52 -040082spv_operand_type_t convertOperandClassToType(SpvOp opcode,
83 OperandClass operandClass) {
David Neto78c3b432015-08-27 13:03:52 -040084 // The spec document generator uses OptionalOperandLiteral for several kinds
85 // of repeating values. Our parser needs more specific information about
86 // what is being repeated.
87 if (operandClass == OperandOptionalLiteral) {
Lei Zhang40056702015-09-11 14:31:27 -040088 switch (opcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -040089 case SpvOpLoad:
90 case SpvOpStore:
91 case SpvOpCopyMemory:
92 case SpvOpCopyMemorySized:
David Neto3fca4cd2015-09-17 17:39:45 -040093 // Expect an optional mask. When the Aligned bit is set in the mask,
94 // we will later add the expectation of a literal number operand.
95 return SPV_OPERAND_TYPE_OPTIONAL_MEMORY_ACCESS;
Lei Zhangb36e7042015-10-28 13:40:52 -040096 case SpvOpExecutionMode:
David Neto78c3b432015-08-27 13:03:52 -040097 return SPV_OPERAND_TYPE_VARIABLE_EXECUTION_MODE;
98 default:
99 break;
100 }
Lei Zhangb41d1502015-09-14 15:22:23 -0400101 } else if (operandClass == OperandVariableLiterals) {
David Neto201caf72015-11-04 17:38:17 -0500102 if (opcode == SpvOpConstant || opcode == SpvOpSpecConstant) {
103 // The number type is determined by the type Id operand.
104 return SPV_OPERAND_TYPE_TYPED_LITERAL_NUMBER;
105 }
David Neto78c3b432015-08-27 13:03:52 -0400106 }
107
Lei Zhang1a0334e2015-11-02 09:41:20 -0500108 switch (operandClass) {
109 case OperandNone:
110 return SPV_OPERAND_TYPE_NONE;
111 case OperandId:
112 return SPV_OPERAND_TYPE_ID;
113 case OperandOptionalId:
114 return SPV_OPERAND_TYPE_OPTIONAL_ID;
115 case OperandOptionalImage:
116 return SPV_OPERAND_TYPE_OPTIONAL_IMAGE;
117 case OperandVariableIds:
David Neto0f166be2015-11-11 01:56:49 -0500118 if (opcode == SpvOpSpecConstantOp) {
119 // These are the operands to the specialization constant opcode.
120 // The assembler and binary parser set up the extra Id and literal
121 // arguments when processing the opcode operand. So don't add
122 // an operand type for them here.
123 return SPV_OPERAND_TYPE_NONE;
124 }
Lei Zhang1a0334e2015-11-02 09:41:20 -0500125 return SPV_OPERAND_TYPE_VARIABLE_ID;
David Neto561dc4e2015-09-25 14:23:29 -0400126 // The spec only uses OptionalLiteral for an optional literal number.
Lei Zhang1a0334e2015-11-02 09:41:20 -0500127 case OperandOptionalLiteral:
128 return SPV_OPERAND_TYPE_OPTIONAL_LITERAL_INTEGER;
129 case OperandOptionalLiteralString:
130 return SPV_OPERAND_TYPE_OPTIONAL_LITERAL_STRING;
David Neto561dc4e2015-09-25 14:23:29 -0400131 // This is only used for sequences of literal numbers.
Lei Zhang1a0334e2015-11-02 09:41:20 -0500132 case OperandVariableLiterals:
133 return SPV_OPERAND_TYPE_VARIABLE_LITERAL_INTEGER;
David Neto445ce442015-10-15 15:22:06 -0400134 case OperandLiteralNumber:
Lei Zhangb36e7042015-10-28 13:40:52 -0400135 if (opcode == SpvOpExtInst) {
David Neto445ce442015-10-15 15:22:06 -0400136 // We use a special operand type for the extension instruction number.
Lei Zhang1a0334e2015-11-02 09:41:20 -0500137 // For now, we assume there is only one LiteraNumber argument to
138 // OpExtInst, and it is the extension instruction argument.
David Neto445ce442015-10-15 15:22:06 -0400139 // See the ExtInst entry in opcode.inc
140 // TODO(dneto): Use a function to confirm the assumption, and to verify
141 // that the index into the operandClass is 1, as expected.
142 return SPV_OPERAND_TYPE_EXTENSION_INSTRUCTION_NUMBER;
David Neto0f166be2015-11-11 01:56:49 -0500143 } else if (opcode == SpvOpSpecConstantOp) {
144 // Use a special operand type for the opcode operand, so we can
145 // use mnemonic names instead of the numbers. For example, the
146 // assembler should accept "IAdd" instead of the numeric value of
147 // SpvOpIAdd.
148 return SPV_OPERAND_TYPE_SPEC_CONSTANT_OP_NUMBER;
David Neto445ce442015-10-15 15:22:06 -0400149 }
150 return SPV_OPERAND_TYPE_LITERAL_INTEGER;
Lei Zhang1a0334e2015-11-02 09:41:20 -0500151 case OperandLiteralString:
152 return SPV_OPERAND_TYPE_LITERAL_STRING;
153 case OperandSource:
154 return SPV_OPERAND_TYPE_SOURCE_LANGUAGE;
155 case OperandExecutionModel:
156 return SPV_OPERAND_TYPE_EXECUTION_MODEL;
157 case OperandAddressing:
158 return SPV_OPERAND_TYPE_ADDRESSING_MODEL;
159 case OperandMemory:
160 return SPV_OPERAND_TYPE_MEMORY_MODEL;
161 case OperandExecutionMode:
162 return SPV_OPERAND_TYPE_EXECUTION_MODE;
163 case OperandStorage:
164 return SPV_OPERAND_TYPE_STORAGE_CLASS;
165 case OperandDimensionality:
166 return SPV_OPERAND_TYPE_DIMENSIONALITY;
167 case OperandSamplerAddressingMode:
168 return SPV_OPERAND_TYPE_SAMPLER_ADDRESSING_MODE;
169 case OperandSamplerFilterMode:
170 return SPV_OPERAND_TYPE_SAMPLER_FILTER_MODE;
171 case OperandSamplerImageFormat:
172 return SPV_OPERAND_TYPE_SAMPLER_IMAGE_FORMAT;
David Neto7cefb232015-09-28 15:33:49 -0400173 case OperandImageChannelOrder:
174 // This is only used to describe the value generated by OpImageQueryOrder.
175 // It is not used as an operand.
176 break;
177 case OperandImageChannelDataType:
Lei Zhang1a0334e2015-11-02 09:41:20 -0500178 // This is only used to describe the value generated by
179 // OpImageQueryFormat. It is not used as an operand.
David Neto7cefb232015-09-28 15:33:49 -0400180 break;
181 case OperandImageOperands:
Dejan Mircevskif8e091a2015-10-13 14:41:31 -0400182 // This is not used in opcode.inc. It only exists to generate the
183 // corresponding spec section. In parsing, image operands meld into the
184 // OperandOptionalImage case.
David Neto7cefb232015-09-28 15:33:49 -0400185 break;
Lei Zhang1a0334e2015-11-02 09:41:20 -0500186 case OperandFPFastMath:
187 return SPV_OPERAND_TYPE_FP_FAST_MATH_MODE;
188 case OperandFPRoundingMode:
189 return SPV_OPERAND_TYPE_FP_ROUNDING_MODE;
190 case OperandLinkageType:
191 return SPV_OPERAND_TYPE_LINKAGE_TYPE;
192 case OperandAccessQualifier:
193 return SPV_OPERAND_TYPE_ACCESS_QUALIFIER;
194 case OperandFuncParamAttr:
195 return SPV_OPERAND_TYPE_FUNCTION_PARAMETER_ATTRIBUTE;
196 case OperandDecoration:
197 return SPV_OPERAND_TYPE_DECORATION;
198 case OperandBuiltIn:
199 return SPV_OPERAND_TYPE_BUILT_IN;
200 case OperandSelect:
201 return SPV_OPERAND_TYPE_SELECTION_CONTROL;
202 case OperandLoop:
203 return SPV_OPERAND_TYPE_LOOP_CONTROL;
204 case OperandFunction:
205 return SPV_OPERAND_TYPE_FUNCTION_CONTROL;
206 case OperandMemorySemantics:
207 return SPV_OPERAND_TYPE_MEMORY_SEMANTICS;
David Neto78c3b432015-08-27 13:03:52 -0400208 case OperandMemoryAccess:
David Neto7cefb232015-09-28 15:33:49 -0400209 // This case does not occur in the table for SPIR-V 0.99 Rev 32.
David Neto3fca4cd2015-09-17 17:39:45 -0400210 // We expect that it will become SPV_OPERAND_TYPE_OPTIONAL_MEMORY_ACCESS,
David Neto78c3b432015-08-27 13:03:52 -0400211 // and we can remove the special casing above for memory operation
212 // instructions.
213 break;
Lei Zhang1a0334e2015-11-02 09:41:20 -0500214 case OperandScope:
215 return SPV_OPERAND_TYPE_EXECUTION_SCOPE;
216 case OperandGroupOperation:
217 return SPV_OPERAND_TYPE_GROUP_OPERATION;
218 case OperandKernelEnqueueFlags:
219 return SPV_OPERAND_TYPE_KERNEL_ENQ_FLAGS;
220 case OperandKernelProfilingInfo:
221 return SPV_OPERAND_TYPE_KERNEL_PROFILING_INFO;
222 case OperandCapability:
223 return SPV_OPERAND_TYPE_CAPABILITY;
David Neto78c3b432015-08-27 13:03:52 -0400224
225 // Used by GroupMemberDecorate
Lei Zhang1a0334e2015-11-02 09:41:20 -0500226 case OperandVariableIdLiteral:
227 return SPV_OPERAND_TYPE_VARIABLE_ID_LITERAL_INTEGER;
David Neto78c3b432015-08-27 13:03:52 -0400228
229 // Used by Switch
Lei Zhang1a0334e2015-11-02 09:41:20 -0500230 case OperandVariableLiteralId:
231 return SPV_OPERAND_TYPE_VARIABLE_LITERAL_INTEGER_ID;
David Neto78c3b432015-08-27 13:03:52 -0400232
233 // These exceptional cases shouldn't occur.
234 case OperandCount:
235 default:
236 break;
237 }
238 assert(0 && "Unexpected operand class");
239 return SPV_OPERAND_TYPE_NONE;
240}
241
Lei Zhanga94701d2015-09-14 10:05:37 -0400242} // anonymous namespace
David Neto78c3b432015-08-27 13:03:52 -0400243
244// Finish populating the opcodeTableEntries array.
Lei Zhang972788b2015-11-12 13:48:30 -0500245void spvOpcodeTableInitialize(spv_opcode_desc_t* entries,
246 uint32_t num_entries) {
David Neto78c3b432015-08-27 13:03:52 -0400247 // Compute the operandTypes field for each entry.
Lei Zhang972788b2015-11-12 13:48:30 -0500248 for (uint32_t i = 0; i < num_entries; ++i) {
249 spv_opcode_desc_t& opcode = entries[i];
David Neto78c3b432015-08-27 13:03:52 -0400250 opcode.numTypes = 0;
251 // Type ID always comes first, if present.
252 if (opcode.hasType)
Andrew Woloszyn537e7762015-09-29 11:28:34 -0400253 opcode.operandTypes[opcode.numTypes++] = SPV_OPERAND_TYPE_TYPE_ID;
David Neto78c3b432015-08-27 13:03:52 -0400254 // Result ID always comes next, if present
255 if (opcode.hasResult)
256 opcode.operandTypes[opcode.numTypes++] = SPV_OPERAND_TYPE_RESULT_ID;
Lei Zhanga94701d2015-09-14 10:05:37 -0400257 const uint16_t maxNumOperands =
258 sizeof(opcode.operandTypes) / sizeof(opcode.operandTypes[0]);
259 const uint16_t maxNumClasses =
260 sizeof(opcode.operandClass) / sizeof(opcode.operandClass[0]);
261 for (uint16_t classIndex = 0;
262 opcode.numTypes < maxNumOperands && classIndex < maxNumClasses;
263 classIndex++) {
David Neto78c3b432015-08-27 13:03:52 -0400264 const OperandClass operandClass = opcode.operandClass[classIndex];
David Neto0f166be2015-11-11 01:56:49 -0500265 const auto operandType =
Lei Zhanga94701d2015-09-14 10:05:37 -0400266 convertOperandClassToType(opcode.opcode, operandClass);
David Neto0f166be2015-11-11 01:56:49 -0500267 opcode.operandTypes[opcode.numTypes++] = operandType;
David Neto78c3b432015-08-27 13:03:52 -0400268 // The OperandNone value is not explicitly represented in the .inc file.
269 // However, it is the zero value, and is created via implicit value
David Neto0f166be2015-11-11 01:56:49 -0500270 // initialization. It converts to SPV_OPERAND_TYPE_NONE.
271 // The SPV_OPERAND_TYPE_NONE operand type indicates no current or futher
272 // operands.
273 if (operandType == SPV_OPERAND_TYPE_NONE) {
David Neto78c3b432015-08-27 13:03:52 -0400274 opcode.numTypes--;
275 break;
276 }
277 }
David Neto0f166be2015-11-11 01:56:49 -0500278
David Neto78c3b432015-08-27 13:03:52 -0400279 // We should have written the terminating SPV_OPERAND_TYPE_NONE entry, but
280 // also without overflowing.
Lei Zhanga94701d2015-09-14 10:05:37 -0400281 assert((opcode.numTypes < maxNumOperands) &&
282 "Operand class list is too long. Expand "
283 "spv_opcode_desc_t.operandClass");
David Neto78c3b432015-08-27 13:03:52 -0400284 }
David Neto78c3b432015-08-27 13:03:52 -0400285}
286
Lei Zhang1a0334e2015-11-02 09:41:20 -0500287const char* spvGeneratorStr(uint32_t generator) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100288 switch (generator) {
289 case SPV_GENERATOR_KHRONOS:
290 return "Khronos";
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100291 case SPV_GENERATOR_LUNARG:
292 return "LunarG";
David Neto1780fc42015-10-26 15:43:12 -0400293 case SPV_GENERATOR_VALVE:
294 return "Valve";
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100295 case SPV_GENERATOR_CODEPLAY:
296 return "Codeplay Software Ltd.";
David Neto1780fc42015-10-26 15:43:12 -0400297 case SPV_GENERATOR_NVIDIA:
298 return "NVIDIA";
299 case SPV_GENERATOR_ARM:
300 return "ARM";
David Neto14b93e42015-11-12 18:33:47 -0500301 case SPV_GENERATOR_KHRONOS_LLVM_TRANSLATOR:
302 return "Khronos LLVM/SPIR-V Translator";
303 case SPV_GENERATOR_KHRONOS_ASSEMBLER:
304 return "Khronos SPIR-V Tools Assembler";
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100305 default:
306 return "Unknown";
307 }
308}
309
Lei Zhangb36e7042015-10-28 13:40:52 -0400310uint32_t spvOpcodeMake(uint16_t wordCount, SpvOp opcode) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100311 return ((uint32_t)opcode) | (((uint32_t)wordCount) << 16);
312}
313
Lei Zhang1a0334e2015-11-02 09:41:20 -0500314void spvOpcodeSplit(const uint32_t word, uint16_t* pWordCount, SpvOp* pOpcode) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100315 if (pWordCount) {
316 *pWordCount = (uint16_t)((0xffff0000 & word) >> 16);
317 }
318 if (pOpcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400319 *pOpcode = (SpvOp)(0x0000ffff & word);
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100320 }
321}
322
Lei Zhang1a0334e2015-11-02 09:41:20 -0500323spv_result_t spvOpcodeTableGet(spv_opcode_table* pInstTable) {
Lei Zhang40056702015-09-11 14:31:27 -0400324 if (!pInstTable) return SPV_ERROR_INVALID_POINTER;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100325
Lei Zhang972788b2015-11-12 13:48:30 -0500326 const uint32_t size = sizeof(opcodeTableEntries);
327 spv_opcode_desc_t* copied_entries =
328 static_cast<spv_opcode_desc_t*>(::malloc(size));
329 if (!copied_entries) return SPV_ERROR_OUT_OF_MEMORY;
330 ::memcpy(copied_entries, opcodeTableEntries, size);
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100331
Lei Zhang972788b2015-11-12 13:48:30 -0500332 const uint32_t count = sizeof(opcodeTableEntries) / sizeof(spv_opcode_desc_t);
333 spv_opcode_table_t* table = new spv_opcode_table_t{count, copied_entries};
David Neto78c3b432015-08-27 13:03:52 -0400334
Lei Zhang972788b2015-11-12 13:48:30 -0500335 spvOpcodeTableInitialize(copied_entries, count);
336
337 *pInstTable = table;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100338
339 return SPV_SUCCESS;
340}
341
342spv_result_t spvOpcodeTableNameLookup(const spv_opcode_table table,
Lei Zhang1a0334e2015-11-02 09:41:20 -0500343 const char* name,
344 spv_opcode_desc* pEntry) {
Lei Zhang40056702015-09-11 14:31:27 -0400345 if (!name || !pEntry) return SPV_ERROR_INVALID_POINTER;
346 if (!table) return SPV_ERROR_INVALID_TABLE;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100347
348 // TODO: This lookup of the Opcode table is suboptimal! Binary sort would be
349 // preferable but the table requires sorting on the Opcode name, but it's
350 // static
351 // const initialized and matches the order of the spec.
352 const size_t nameLength = strlen(name);
353 for (uint64_t opcodeIndex = 0; opcodeIndex < table->count; ++opcodeIndex) {
354 if (nameLength == strlen(table->entries[opcodeIndex].name) &&
355 !strncmp(name, table->entries[opcodeIndex].name, nameLength)) {
356 // NOTE: Found out Opcode!
357 *pEntry = &table->entries[opcodeIndex];
358 return SPV_SUCCESS;
359 }
360 }
361
362 return SPV_ERROR_INVALID_LOOKUP;
363}
364
365spv_result_t spvOpcodeTableValueLookup(const spv_opcode_table table,
Lei Zhangb36e7042015-10-28 13:40:52 -0400366 const SpvOp opcode,
Lei Zhang1a0334e2015-11-02 09:41:20 -0500367 spv_opcode_desc* pEntry) {
Lei Zhang40056702015-09-11 14:31:27 -0400368 if (!table) return SPV_ERROR_INVALID_TABLE;
369 if (!pEntry) return SPV_ERROR_INVALID_POINTER;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100370
371 // TODO: As above this lookup is not optimal.
372 for (uint64_t opcodeIndex = 0; opcodeIndex < table->count; ++opcodeIndex) {
373 if (opcode == table->entries[opcodeIndex].opcode) {
374 // NOTE: Found the Opcode!
375 *pEntry = &table->entries[opcodeIndex];
376 return SPV_SUCCESS;
377 }
378 }
379
380 return SPV_ERROR_INVALID_LOOKUP;
381}
382
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100383int32_t spvOpcodeRequiresCapabilities(spv_opcode_desc entry) {
David Neto9db3a532015-10-07 16:58:38 -0400384 return entry->capabilities != 0;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100385}
386
Lei Zhang1a0334e2015-11-02 09:41:20 -0500387void spvInstructionCopy(const uint32_t* words, const SpvOp opcode,
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100388 const uint16_t wordCount, const spv_endianness_t endian,
Lei Zhang1a0334e2015-11-02 09:41:20 -0500389 spv_instruction_t* pInst) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100390 pInst->opcode = opcode;
David Netob5dc8fc2015-10-06 16:22:00 -0400391 pInst->words.resize(wordCount);
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100392 for (uint16_t wordIndex = 0; wordIndex < wordCount; ++wordIndex) {
393 pInst->words[wordIndex] = spvFixWord(words[wordIndex], endian);
394 if (!wordIndex) {
395 uint16_t thisWordCount;
Lei Zhangb36e7042015-10-28 13:40:52 -0400396 SpvOp thisOpcode;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100397 spvOpcodeSplit(pInst->words[wordIndex], &thisWordCount, &thisOpcode);
398 assert(opcode == thisOpcode && wordCount == thisWordCount &&
399 "Endianness failed!");
400 }
401 }
402}
403
Lei Zhang1a0334e2015-11-02 09:41:20 -0500404const char* spvOpcodeString(const SpvOp opcode) {
David Neto201caf72015-11-04 17:38:17 -0500405// Use the syntax table so it's sure to be complete.
David Neto1bcd3d12015-11-02 16:03:12 -0500406#define Instruction(Name, ...) \
407 case SpvOp##Name: \
408 return #Name;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100409 switch (opcode) {
David Neto1bcd3d12015-11-02 16:03:12 -0500410#include "opcode.inc"
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100411 default:
412 assert(0 && "Unreachable!");
413 }
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100414 return "unknown";
David Neto1bcd3d12015-11-02 16:03:12 -0500415#undef Instruction
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100416}
417
Lei Zhangb36e7042015-10-28 13:40:52 -0400418int32_t spvOpcodeIsScalarType(const SpvOp opcode) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100419 switch (opcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400420 case SpvOpTypeInt:
421 case SpvOpTypeFloat:
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100422 return true;
423 default:
424 return false;
425 }
426}
427
Lei Zhangb36e7042015-10-28 13:40:52 -0400428int32_t spvOpcodeIsConstant(const SpvOp opcode) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100429 switch (opcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400430 case SpvOpConstantTrue:
431 case SpvOpConstantFalse:
432 case SpvOpConstant:
433 case SpvOpConstantComposite:
434 case SpvOpConstantSampler:
435 // case SpvOpConstantNull:
436 case SpvOpConstantNull:
437 case SpvOpSpecConstantTrue:
438 case SpvOpSpecConstantFalse:
439 case SpvOpSpecConstant:
440 case SpvOpSpecConstantComposite:
441 // case SpvOpSpecConstantOp:
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100442 return true;
443 default:
444 return false;
445 }
446}
447
Lei Zhangb36e7042015-10-28 13:40:52 -0400448int32_t spvOpcodeIsComposite(const SpvOp opcode) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100449 switch (opcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400450 case SpvOpTypeVector:
451 case SpvOpTypeMatrix:
452 case SpvOpTypeArray:
453 case SpvOpTypeStruct:
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100454 return true;
455 default:
456 return false;
457 }
458}
459
Lei Zhang1a0334e2015-11-02 09:41:20 -0500460int32_t spvOpcodeAreTypesEqual(const spv_instruction_t* pTypeInst0,
461 const spv_instruction_t* pTypeInst1) {
Lei Zhang40056702015-09-11 14:31:27 -0400462 if (pTypeInst0->opcode != pTypeInst1->opcode) return false;
463 if (pTypeInst0->words[1] != pTypeInst1->words[1]) return false;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100464 return true;
465}
466
Lei Zhangb36e7042015-10-28 13:40:52 -0400467int32_t spvOpcodeIsPointer(const SpvOp opcode) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100468 switch (opcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400469 case SpvOpVariable:
470 case SpvOpAccessChain:
471 case SpvOpInBoundsAccessChain:
472 case SpvOpFunctionParameter:
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100473 return true;
474 default:
475 return false;
476 }
477}
478
Lei Zhangb36e7042015-10-28 13:40:52 -0400479int32_t spvOpcodeIsObject(const SpvOp opcode) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100480 switch (opcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400481 case SpvOpConstantTrue:
482 case SpvOpConstantFalse:
483 case SpvOpConstant:
484 case SpvOpConstantComposite:
485 // TODO: case SpvOpConstantSampler:
486 case SpvOpConstantNull:
487 case SpvOpSpecConstantTrue:
488 case SpvOpSpecConstantFalse:
489 case SpvOpSpecConstant:
490 case SpvOpSpecConstantComposite:
491 // TODO: case SpvOpSpecConstantOp:
492 case SpvOpVariable:
493 case SpvOpAccessChain:
494 case SpvOpInBoundsAccessChain:
495 case SpvOpConvertFToU:
496 case SpvOpConvertFToS:
497 case SpvOpConvertSToF:
498 case SpvOpConvertUToF:
499 case SpvOpUConvert:
500 case SpvOpSConvert:
501 case SpvOpFConvert:
502 case SpvOpConvertPtrToU:
503 // TODO: case SpvOpConvertUToPtr:
504 case SpvOpPtrCastToGeneric:
505 // TODO: case SpvOpGenericCastToPtr:
506 case SpvOpBitcast:
507 // TODO: case SpvOpGenericCastToPtrExplicit:
508 case SpvOpSatConvertSToU:
509 case SpvOpSatConvertUToS:
510 case SpvOpVectorExtractDynamic:
511 case SpvOpCompositeConstruct:
512 case SpvOpCompositeExtract:
513 case SpvOpCopyObject:
514 case SpvOpTranspose:
515 case SpvOpSNegate:
516 case SpvOpFNegate:
517 case SpvOpNot:
518 case SpvOpIAdd:
519 case SpvOpFAdd:
520 case SpvOpISub:
521 case SpvOpFSub:
522 case SpvOpIMul:
523 case SpvOpFMul:
524 case SpvOpUDiv:
525 case SpvOpSDiv:
526 case SpvOpFDiv:
527 case SpvOpUMod:
528 case SpvOpSRem:
529 case SpvOpSMod:
530 case SpvOpVectorTimesScalar:
531 case SpvOpMatrixTimesScalar:
532 case SpvOpVectorTimesMatrix:
533 case SpvOpMatrixTimesVector:
534 case SpvOpMatrixTimesMatrix:
535 case SpvOpOuterProduct:
536 case SpvOpDot:
537 case SpvOpShiftRightLogical:
538 case SpvOpShiftRightArithmetic:
539 case SpvOpShiftLeftLogical:
540 case SpvOpBitwiseOr:
541 case SpvOpBitwiseXor:
542 case SpvOpBitwiseAnd:
543 case SpvOpAny:
544 case SpvOpAll:
545 case SpvOpIsNan:
546 case SpvOpIsInf:
547 case SpvOpIsFinite:
548 case SpvOpIsNormal:
549 case SpvOpSignBitSet:
550 case SpvOpLessOrGreater:
551 case SpvOpOrdered:
552 case SpvOpUnordered:
553 case SpvOpLogicalOr:
554 case SpvOpLogicalAnd:
555 case SpvOpSelect:
556 case SpvOpIEqual:
557 case SpvOpFOrdEqual:
558 case SpvOpFUnordEqual:
559 case SpvOpINotEqual:
560 case SpvOpFOrdNotEqual:
561 case SpvOpFUnordNotEqual:
562 case SpvOpULessThan:
563 case SpvOpSLessThan:
564 case SpvOpFOrdLessThan:
565 case SpvOpFUnordLessThan:
566 case SpvOpUGreaterThan:
567 case SpvOpSGreaterThan:
568 case SpvOpFOrdGreaterThan:
569 case SpvOpFUnordGreaterThan:
570 case SpvOpULessThanEqual:
571 case SpvOpSLessThanEqual:
572 case SpvOpFOrdLessThanEqual:
573 case SpvOpFUnordLessThanEqual:
574 case SpvOpUGreaterThanEqual:
575 case SpvOpSGreaterThanEqual:
576 case SpvOpFOrdGreaterThanEqual:
577 case SpvOpFUnordGreaterThanEqual:
578 case SpvOpDPdx:
579 case SpvOpDPdy:
580 case SpvOpFwidth:
581 case SpvOpDPdxFine:
582 case SpvOpDPdyFine:
583 case SpvOpFwidthFine:
584 case SpvOpDPdxCoarse:
585 case SpvOpDPdyCoarse:
586 case SpvOpFwidthCoarse:
587 case SpvOpReturnValue:
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100588 return true;
589 default:
590 return false;
591 }
592}
593
Lei Zhangb36e7042015-10-28 13:40:52 -0400594int32_t spvOpcodeIsBasicTypeNullable(SpvOp opcode) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100595 switch (opcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400596 case SpvOpTypeBool:
597 case SpvOpTypeInt:
598 case SpvOpTypeFloat:
599 case SpvOpTypePointer:
600 case SpvOpTypeEvent:
601 case SpvOpTypeDeviceEvent:
602 case SpvOpTypeReserveId:
603 case SpvOpTypeQueue:
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100604 return true;
605 default:
606 return false;
607 }
608}
609
Lei Zhang1a0334e2015-11-02 09:41:20 -0500610int32_t spvInstructionIsInBasicBlock(const spv_instruction_t* pFirstInst,
611 const spv_instruction_t* pInst) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100612 while (pFirstInst != pInst) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400613 if (SpvOpFunction == pInst->opcode) break;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100614 pInst--;
615 }
Lei Zhangb36e7042015-10-28 13:40:52 -0400616 if (SpvOpFunction != pInst->opcode) return false;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100617 return true;
618}
619
Lei Zhangb36e7042015-10-28 13:40:52 -0400620int32_t spvOpcodeIsValue(SpvOp opcode) {
Lei Zhang40056702015-09-11 14:31:27 -0400621 if (spvOpcodeIsPointer(opcode)) return true;
622 if (spvOpcodeIsConstant(opcode)) return true;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100623 switch (opcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400624 case SpvOpLoad:
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100625 // TODO: Other Opcode's resulting in a value
626 return true;
627 default:
628 return false;
629 }
630}
Andrew Woloszyn537e7762015-09-29 11:28:34 -0400631
Lei Zhangb36e7042015-10-28 13:40:52 -0400632int32_t spvOpcodeGeneratesType(SpvOp op) {
Lei Zhang1a0334e2015-11-02 09:41:20 -0500633 switch (op) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400634 case SpvOpTypeVoid:
635 case SpvOpTypeBool:
636 case SpvOpTypeInt:
637 case SpvOpTypeFloat:
638 case SpvOpTypeVector:
639 case SpvOpTypeMatrix:
640 case SpvOpTypeImage:
641 case SpvOpTypeSampler:
642 case SpvOpTypeSampledImage:
643 case SpvOpTypeArray:
644 case SpvOpTypeRuntimeArray:
645 case SpvOpTypeStruct:
646 case SpvOpTypeOpaque:
647 case SpvOpTypePointer:
648 case SpvOpTypeFunction:
649 case SpvOpTypeEvent:
650 case SpvOpTypeDeviceEvent:
651 case SpvOpTypeReserveId:
652 case SpvOpTypeQueue:
653 case SpvOpTypePipe:
Andrew Woloszyn537e7762015-09-29 11:28:34 -0400654 return true;
David Netoaef608c2015-11-02 14:59:02 -0500655 default:
656 // In particular, OpTypeForwardPointer does not generate a type,
657 // but declares a storage class for a pointer type generated
658 // by a different instruction.
659 break;
Andrew Woloszyn537e7762015-09-29 11:28:34 -0400660 }
661 return 0;
662}