blob: 526933d1c9f084da7e3c5c5565599796ece17ab2 [file] [log] [blame]
Dejan Mircevskib6fe02f2016-01-07 13:44:22 -05001// Copyright (c) 2015-2016 The Khronos Group Inc.
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +01002//
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 "instruction.h"
35#include "libspirv/libspirv.h"
Lei Zhangaa056cd2015-11-11 14:24:04 -050036#include "spirv_constant.h"
David Neto4c215712015-12-22 15:08:41 -050037#include "spirv_endian.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 Neto39fa1482015-11-30 14:39:31 -0500102 switch (opcode) {
103 case SpvOpConstant:
104 case SpvOpSpecConstant:
105 // The number type is determined by the type Id operand.
106 return SPV_OPERAND_TYPE_TYPED_LITERAL_NUMBER;
107 case SpvOpDecorate:
108 case SpvOpMemberDecorate:
109 // The operand types at the end of the instruction are
110 // determined instead by the decoration kind.
111 return SPV_OPERAND_TYPE_NONE;
112 default:
113 break;
David Neto201caf72015-11-04 17:38:17 -0500114 }
David Neto78c3b432015-08-27 13:03:52 -0400115 }
116
Lei Zhang1a0334e2015-11-02 09:41:20 -0500117 switch (operandClass) {
118 case OperandNone:
119 return SPV_OPERAND_TYPE_NONE;
120 case OperandId:
121 return SPV_OPERAND_TYPE_ID;
122 case OperandOptionalId:
123 return SPV_OPERAND_TYPE_OPTIONAL_ID;
124 case OperandOptionalImage:
125 return SPV_OPERAND_TYPE_OPTIONAL_IMAGE;
126 case OperandVariableIds:
David Neto0f166be2015-11-11 01:56:49 -0500127 if (opcode == SpvOpSpecConstantOp) {
128 // These are the operands to the specialization constant opcode.
129 // The assembler and binary parser set up the extra Id and literal
130 // arguments when processing the opcode operand. So don't add
131 // an operand type for them here.
132 return SPV_OPERAND_TYPE_NONE;
133 }
Lei Zhang1a0334e2015-11-02 09:41:20 -0500134 return SPV_OPERAND_TYPE_VARIABLE_ID;
David Neto561dc4e2015-09-25 14:23:29 -0400135 // The spec only uses OptionalLiteral for an optional literal number.
Lei Zhang1a0334e2015-11-02 09:41:20 -0500136 case OperandOptionalLiteral:
137 return SPV_OPERAND_TYPE_OPTIONAL_LITERAL_INTEGER;
138 case OperandOptionalLiteralString:
139 return SPV_OPERAND_TYPE_OPTIONAL_LITERAL_STRING;
David Neto561dc4e2015-09-25 14:23:29 -0400140 // This is only used for sequences of literal numbers.
Lei Zhang1a0334e2015-11-02 09:41:20 -0500141 case OperandVariableLiterals:
142 return SPV_OPERAND_TYPE_VARIABLE_LITERAL_INTEGER;
David Neto445ce442015-10-15 15:22:06 -0400143 case OperandLiteralNumber:
Lei Zhangb36e7042015-10-28 13:40:52 -0400144 if (opcode == SpvOpExtInst) {
David Neto445ce442015-10-15 15:22:06 -0400145 // We use a special operand type for the extension instruction number.
Lei Zhang1a0334e2015-11-02 09:41:20 -0500146 // For now, we assume there is only one LiteraNumber argument to
147 // OpExtInst, and it is the extension instruction argument.
David Neto445ce442015-10-15 15:22:06 -0400148 // See the ExtInst entry in opcode.inc
149 // TODO(dneto): Use a function to confirm the assumption, and to verify
150 // that the index into the operandClass is 1, as expected.
151 return SPV_OPERAND_TYPE_EXTENSION_INSTRUCTION_NUMBER;
David Neto0f166be2015-11-11 01:56:49 -0500152 } else if (opcode == SpvOpSpecConstantOp) {
153 // Use a special operand type for the opcode operand, so we can
154 // use mnemonic names instead of the numbers. For example, the
155 // assembler should accept "IAdd" instead of the numeric value of
156 // SpvOpIAdd.
157 return SPV_OPERAND_TYPE_SPEC_CONSTANT_OP_NUMBER;
David Neto445ce442015-10-15 15:22:06 -0400158 }
159 return SPV_OPERAND_TYPE_LITERAL_INTEGER;
Lei Zhang1a0334e2015-11-02 09:41:20 -0500160 case OperandLiteralString:
161 return SPV_OPERAND_TYPE_LITERAL_STRING;
162 case OperandSource:
163 return SPV_OPERAND_TYPE_SOURCE_LANGUAGE;
164 case OperandExecutionModel:
165 return SPV_OPERAND_TYPE_EXECUTION_MODEL;
166 case OperandAddressing:
167 return SPV_OPERAND_TYPE_ADDRESSING_MODEL;
168 case OperandMemory:
169 return SPV_OPERAND_TYPE_MEMORY_MODEL;
170 case OperandExecutionMode:
171 return SPV_OPERAND_TYPE_EXECUTION_MODE;
172 case OperandStorage:
173 return SPV_OPERAND_TYPE_STORAGE_CLASS;
174 case OperandDimensionality:
175 return SPV_OPERAND_TYPE_DIMENSIONALITY;
176 case OperandSamplerAddressingMode:
177 return SPV_OPERAND_TYPE_SAMPLER_ADDRESSING_MODE;
178 case OperandSamplerFilterMode:
179 return SPV_OPERAND_TYPE_SAMPLER_FILTER_MODE;
180 case OperandSamplerImageFormat:
181 return SPV_OPERAND_TYPE_SAMPLER_IMAGE_FORMAT;
David Neto7cefb232015-09-28 15:33:49 -0400182 case OperandImageChannelOrder:
183 // This is only used to describe the value generated by OpImageQueryOrder.
184 // It is not used as an operand.
185 break;
186 case OperandImageChannelDataType:
Lei Zhang1a0334e2015-11-02 09:41:20 -0500187 // This is only used to describe the value generated by
188 // OpImageQueryFormat. It is not used as an operand.
David Neto7cefb232015-09-28 15:33:49 -0400189 break;
190 case OperandImageOperands:
Dejan Mircevskif8e091a2015-10-13 14:41:31 -0400191 // This is not used in opcode.inc. It only exists to generate the
192 // corresponding spec section. In parsing, image operands meld into the
193 // OperandOptionalImage case.
David Neto7cefb232015-09-28 15:33:49 -0400194 break;
Lei Zhang1a0334e2015-11-02 09:41:20 -0500195 case OperandFPFastMath:
196 return SPV_OPERAND_TYPE_FP_FAST_MATH_MODE;
197 case OperandFPRoundingMode:
198 return SPV_OPERAND_TYPE_FP_ROUNDING_MODE;
199 case OperandLinkageType:
200 return SPV_OPERAND_TYPE_LINKAGE_TYPE;
201 case OperandAccessQualifier:
202 return SPV_OPERAND_TYPE_ACCESS_QUALIFIER;
203 case OperandFuncParamAttr:
204 return SPV_OPERAND_TYPE_FUNCTION_PARAMETER_ATTRIBUTE;
205 case OperandDecoration:
206 return SPV_OPERAND_TYPE_DECORATION;
207 case OperandBuiltIn:
208 return SPV_OPERAND_TYPE_BUILT_IN;
209 case OperandSelect:
210 return SPV_OPERAND_TYPE_SELECTION_CONTROL;
211 case OperandLoop:
212 return SPV_OPERAND_TYPE_LOOP_CONTROL;
213 case OperandFunction:
214 return SPV_OPERAND_TYPE_FUNCTION_CONTROL;
215 case OperandMemorySemantics:
David Neto64a9be92015-11-18 15:48:32 -0500216 return SPV_OPERAND_TYPE_MEMORY_SEMANTICS_ID;
David Neto78c3b432015-08-27 13:03:52 -0400217 case OperandMemoryAccess:
David Neto7cefb232015-09-28 15:33:49 -0400218 // This case does not occur in the table for SPIR-V 0.99 Rev 32.
David Neto3fca4cd2015-09-17 17:39:45 -0400219 // We expect that it will become SPV_OPERAND_TYPE_OPTIONAL_MEMORY_ACCESS,
David Neto78c3b432015-08-27 13:03:52 -0400220 // and we can remove the special casing above for memory operation
221 // instructions.
222 break;
Lei Zhang1a0334e2015-11-02 09:41:20 -0500223 case OperandScope:
David Neto64a9be92015-11-18 15:48:32 -0500224 return SPV_OPERAND_TYPE_SCOPE_ID;
Lei Zhang1a0334e2015-11-02 09:41:20 -0500225 case OperandGroupOperation:
226 return SPV_OPERAND_TYPE_GROUP_OPERATION;
227 case OperandKernelEnqueueFlags:
228 return SPV_OPERAND_TYPE_KERNEL_ENQ_FLAGS;
229 case OperandKernelProfilingInfo:
230 return SPV_OPERAND_TYPE_KERNEL_PROFILING_INFO;
231 case OperandCapability:
232 return SPV_OPERAND_TYPE_CAPABILITY;
David Neto78c3b432015-08-27 13:03:52 -0400233
234 // Used by GroupMemberDecorate
Lei Zhang1a0334e2015-11-02 09:41:20 -0500235 case OperandVariableIdLiteral:
236 return SPV_OPERAND_TYPE_VARIABLE_ID_LITERAL_INTEGER;
David Neto78c3b432015-08-27 13:03:52 -0400237
238 // Used by Switch
Lei Zhang1a0334e2015-11-02 09:41:20 -0500239 case OperandVariableLiteralId:
240 return SPV_OPERAND_TYPE_VARIABLE_LITERAL_INTEGER_ID;
David Neto78c3b432015-08-27 13:03:52 -0400241
242 // These exceptional cases shouldn't occur.
243 case OperandCount:
244 default:
245 break;
246 }
247 assert(0 && "Unexpected operand class");
248 return SPV_OPERAND_TYPE_NONE;
249}
250
Lei Zhanga94701d2015-09-14 10:05:37 -0400251} // anonymous namespace
David Neto78c3b432015-08-27 13:03:52 -0400252
253// Finish populating the opcodeTableEntries array.
Lei Zhang972788b2015-11-12 13:48:30 -0500254void spvOpcodeTableInitialize(spv_opcode_desc_t* entries,
255 uint32_t num_entries) {
David Neto78c3b432015-08-27 13:03:52 -0400256 // Compute the operandTypes field for each entry.
Lei Zhang972788b2015-11-12 13:48:30 -0500257 for (uint32_t i = 0; i < num_entries; ++i) {
258 spv_opcode_desc_t& opcode = entries[i];
David Neto78c3b432015-08-27 13:03:52 -0400259 opcode.numTypes = 0;
260 // Type ID always comes first, if present.
261 if (opcode.hasType)
Andrew Woloszyn537e7762015-09-29 11:28:34 -0400262 opcode.operandTypes[opcode.numTypes++] = SPV_OPERAND_TYPE_TYPE_ID;
David Neto78c3b432015-08-27 13:03:52 -0400263 // Result ID always comes next, if present
264 if (opcode.hasResult)
265 opcode.operandTypes[opcode.numTypes++] = SPV_OPERAND_TYPE_RESULT_ID;
David Netoba73a7c2016-01-06 13:08:39 -0500266 const uint16_t maxNumOperands = static_cast<uint16_t>(
267 sizeof(opcode.operandTypes) / sizeof(opcode.operandTypes[0]));
268 const uint16_t maxNumClasses = static_cast<uint16_t>(
269 sizeof(opcode.operandClass) / sizeof(opcode.operandClass[0]));
Lei Zhanga94701d2015-09-14 10:05:37 -0400270 for (uint16_t classIndex = 0;
271 opcode.numTypes < maxNumOperands && classIndex < maxNumClasses;
272 classIndex++) {
David Neto78c3b432015-08-27 13:03:52 -0400273 const OperandClass operandClass = opcode.operandClass[classIndex];
David Neto0f166be2015-11-11 01:56:49 -0500274 const auto operandType =
Lei Zhanga94701d2015-09-14 10:05:37 -0400275 convertOperandClassToType(opcode.opcode, operandClass);
David Neto0f166be2015-11-11 01:56:49 -0500276 opcode.operandTypes[opcode.numTypes++] = operandType;
David Neto78c3b432015-08-27 13:03:52 -0400277 // The OperandNone value is not explicitly represented in the .inc file.
278 // However, it is the zero value, and is created via implicit value
David Neto0f166be2015-11-11 01:56:49 -0500279 // initialization. It converts to SPV_OPERAND_TYPE_NONE.
280 // The SPV_OPERAND_TYPE_NONE operand type indicates no current or futher
281 // operands.
282 if (operandType == SPV_OPERAND_TYPE_NONE) {
David Neto78c3b432015-08-27 13:03:52 -0400283 opcode.numTypes--;
284 break;
285 }
286 }
David Neto0f166be2015-11-11 01:56:49 -0500287
David Neto78c3b432015-08-27 13:03:52 -0400288 // We should have written the terminating SPV_OPERAND_TYPE_NONE entry, but
289 // also without overflowing.
Lei Zhanga94701d2015-09-14 10:05:37 -0400290 assert((opcode.numTypes < maxNumOperands) &&
291 "Operand class list is too long. Expand "
292 "spv_opcode_desc_t.operandClass");
David Neto78c3b432015-08-27 13:03:52 -0400293 }
David Neto78c3b432015-08-27 13:03:52 -0400294}
295
Lei Zhang1a0334e2015-11-02 09:41:20 -0500296const char* spvGeneratorStr(uint32_t generator) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100297 switch (generator) {
298 case SPV_GENERATOR_KHRONOS:
299 return "Khronos";
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100300 case SPV_GENERATOR_LUNARG:
301 return "LunarG";
David Neto1780fc42015-10-26 15:43:12 -0400302 case SPV_GENERATOR_VALVE:
303 return "Valve";
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100304 case SPV_GENERATOR_CODEPLAY:
305 return "Codeplay Software Ltd.";
David Neto1780fc42015-10-26 15:43:12 -0400306 case SPV_GENERATOR_NVIDIA:
307 return "NVIDIA";
308 case SPV_GENERATOR_ARM:
309 return "ARM";
David Neto14b93e42015-11-12 18:33:47 -0500310 case SPV_GENERATOR_KHRONOS_LLVM_TRANSLATOR:
311 return "Khronos LLVM/SPIR-V Translator";
312 case SPV_GENERATOR_KHRONOS_ASSEMBLER:
313 return "Khronos SPIR-V Tools Assembler";
David Neto2266ba12015-11-13 12:03:28 -0600314 case SPV_GENERATOR_KHRONOS_GLSLANG:
315 return "Khronos Glslang Reference Front End";
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100316 default:
317 return "Unknown";
318 }
319}
320
Lei Zhangb36e7042015-10-28 13:40:52 -0400321uint32_t spvOpcodeMake(uint16_t wordCount, SpvOp opcode) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100322 return ((uint32_t)opcode) | (((uint32_t)wordCount) << 16);
323}
324
Lei Zhang1a0334e2015-11-02 09:41:20 -0500325void spvOpcodeSplit(const uint32_t word, uint16_t* pWordCount, SpvOp* pOpcode) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100326 if (pWordCount) {
327 *pWordCount = (uint16_t)((0xffff0000 & word) >> 16);
328 }
329 if (pOpcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400330 *pOpcode = (SpvOp)(0x0000ffff & word);
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100331 }
332}
333
Lei Zhang1a0334e2015-11-02 09:41:20 -0500334spv_result_t spvOpcodeTableGet(spv_opcode_table* pInstTable) {
Lei Zhang40056702015-09-11 14:31:27 -0400335 if (!pInstTable) return SPV_ERROR_INVALID_POINTER;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100336
Lei Zhang972788b2015-11-12 13:48:30 -0500337 const uint32_t size = sizeof(opcodeTableEntries);
338 spv_opcode_desc_t* copied_entries =
339 static_cast<spv_opcode_desc_t*>(::malloc(size));
340 if (!copied_entries) return SPV_ERROR_OUT_OF_MEMORY;
341 ::memcpy(copied_entries, opcodeTableEntries, size);
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100342
David Netoba73a7c2016-01-06 13:08:39 -0500343 const uint32_t count = static_cast<uint32_t>(sizeof(opcodeTableEntries) /
344 sizeof(spv_opcode_desc_t));
Lei Zhang972788b2015-11-12 13:48:30 -0500345 spv_opcode_table_t* table = new spv_opcode_table_t{count, copied_entries};
David Neto78c3b432015-08-27 13:03:52 -0400346
Lei Zhang972788b2015-11-12 13:48:30 -0500347 spvOpcodeTableInitialize(copied_entries, count);
348
349 *pInstTable = table;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100350
351 return SPV_SUCCESS;
352}
353
354spv_result_t spvOpcodeTableNameLookup(const spv_opcode_table table,
Lei Zhang1a0334e2015-11-02 09:41:20 -0500355 const char* name,
356 spv_opcode_desc* pEntry) {
Lei Zhang40056702015-09-11 14:31:27 -0400357 if (!name || !pEntry) return SPV_ERROR_INVALID_POINTER;
358 if (!table) return SPV_ERROR_INVALID_TABLE;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100359
360 // TODO: This lookup of the Opcode table is suboptimal! Binary sort would be
361 // preferable but the table requires sorting on the Opcode name, but it's
362 // static
363 // const initialized and matches the order of the spec.
364 const size_t nameLength = strlen(name);
365 for (uint64_t opcodeIndex = 0; opcodeIndex < table->count; ++opcodeIndex) {
366 if (nameLength == strlen(table->entries[opcodeIndex].name) &&
367 !strncmp(name, table->entries[opcodeIndex].name, nameLength)) {
368 // NOTE: Found out Opcode!
369 *pEntry = &table->entries[opcodeIndex];
370 return SPV_SUCCESS;
371 }
372 }
373
374 return SPV_ERROR_INVALID_LOOKUP;
375}
376
377spv_result_t spvOpcodeTableValueLookup(const spv_opcode_table table,
Lei Zhangb36e7042015-10-28 13:40:52 -0400378 const SpvOp opcode,
Lei Zhang1a0334e2015-11-02 09:41:20 -0500379 spv_opcode_desc* pEntry) {
Lei Zhang40056702015-09-11 14:31:27 -0400380 if (!table) return SPV_ERROR_INVALID_TABLE;
381 if (!pEntry) return SPV_ERROR_INVALID_POINTER;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100382
383 // TODO: As above this lookup is not optimal.
384 for (uint64_t opcodeIndex = 0; opcodeIndex < table->count; ++opcodeIndex) {
385 if (opcode == table->entries[opcodeIndex].opcode) {
386 // NOTE: Found the Opcode!
387 *pEntry = &table->entries[opcodeIndex];
388 return SPV_SUCCESS;
389 }
390 }
391
392 return SPV_ERROR_INVALID_LOOKUP;
393}
394
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100395int32_t spvOpcodeRequiresCapabilities(spv_opcode_desc entry) {
David Neto9db3a532015-10-07 16:58:38 -0400396 return entry->capabilities != 0;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100397}
398
Lei Zhang1a0334e2015-11-02 09:41:20 -0500399void spvInstructionCopy(const uint32_t* words, const SpvOp opcode,
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100400 const uint16_t wordCount, const spv_endianness_t endian,
Lei Zhang1a0334e2015-11-02 09:41:20 -0500401 spv_instruction_t* pInst) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100402 pInst->opcode = opcode;
David Netob5dc8fc2015-10-06 16:22:00 -0400403 pInst->words.resize(wordCount);
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100404 for (uint16_t wordIndex = 0; wordIndex < wordCount; ++wordIndex) {
405 pInst->words[wordIndex] = spvFixWord(words[wordIndex], endian);
406 if (!wordIndex) {
407 uint16_t thisWordCount;
Lei Zhangb36e7042015-10-28 13:40:52 -0400408 SpvOp thisOpcode;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100409 spvOpcodeSplit(pInst->words[wordIndex], &thisWordCount, &thisOpcode);
410 assert(opcode == thisOpcode && wordCount == thisWordCount &&
411 "Endianness failed!");
412 }
413 }
414}
415
Lei Zhang1a0334e2015-11-02 09:41:20 -0500416const char* spvOpcodeString(const SpvOp opcode) {
David Neto201caf72015-11-04 17:38:17 -0500417// Use the syntax table so it's sure to be complete.
David Neto1bcd3d12015-11-02 16:03:12 -0500418#define Instruction(Name, ...) \
419 case SpvOp##Name: \
420 return #Name;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100421 switch (opcode) {
David Neto1bcd3d12015-11-02 16:03:12 -0500422#include "opcode.inc"
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100423 default:
424 assert(0 && "Unreachable!");
425 }
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100426 return "unknown";
David Neto1bcd3d12015-11-02 16:03:12 -0500427#undef Instruction
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100428}
429
Lei Zhangb36e7042015-10-28 13:40:52 -0400430int32_t spvOpcodeIsScalarType(const SpvOp opcode) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100431 switch (opcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400432 case SpvOpTypeInt:
433 case SpvOpTypeFloat:
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100434 return true;
435 default:
436 return false;
437 }
438}
439
Lei Zhangb36e7042015-10-28 13:40:52 -0400440int32_t spvOpcodeIsConstant(const SpvOp opcode) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100441 switch (opcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400442 case SpvOpConstantTrue:
443 case SpvOpConstantFalse:
444 case SpvOpConstant:
445 case SpvOpConstantComposite:
446 case SpvOpConstantSampler:
447 // case SpvOpConstantNull:
448 case SpvOpConstantNull:
449 case SpvOpSpecConstantTrue:
450 case SpvOpSpecConstantFalse:
451 case SpvOpSpecConstant:
452 case SpvOpSpecConstantComposite:
453 // case SpvOpSpecConstantOp:
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100454 return true;
455 default:
456 return false;
457 }
458}
459
Lei Zhangb36e7042015-10-28 13:40:52 -0400460int32_t spvOpcodeIsComposite(const SpvOp opcode) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100461 switch (opcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400462 case SpvOpTypeVector:
463 case SpvOpTypeMatrix:
464 case SpvOpTypeArray:
465 case SpvOpTypeStruct:
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100466 return true;
467 default:
468 return false;
469 }
470}
471
Lei Zhang1a0334e2015-11-02 09:41:20 -0500472int32_t spvOpcodeAreTypesEqual(const spv_instruction_t* pTypeInst0,
473 const spv_instruction_t* pTypeInst1) {
Lei Zhang40056702015-09-11 14:31:27 -0400474 if (pTypeInst0->opcode != pTypeInst1->opcode) return false;
475 if (pTypeInst0->words[1] != pTypeInst1->words[1]) return false;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100476 return true;
477}
478
Lei Zhangb36e7042015-10-28 13:40:52 -0400479int32_t spvOpcodeIsPointer(const SpvOp opcode) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100480 switch (opcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400481 case SpvOpVariable:
482 case SpvOpAccessChain:
483 case SpvOpInBoundsAccessChain:
484 case SpvOpFunctionParameter:
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100485 return true;
486 default:
487 return false;
488 }
489}
490
Lei Zhangb36e7042015-10-28 13:40:52 -0400491int32_t spvOpcodeIsObject(const SpvOp opcode) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100492 switch (opcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400493 case SpvOpConstantTrue:
494 case SpvOpConstantFalse:
495 case SpvOpConstant:
496 case SpvOpConstantComposite:
497 // TODO: case SpvOpConstantSampler:
498 case SpvOpConstantNull:
499 case SpvOpSpecConstantTrue:
500 case SpvOpSpecConstantFalse:
501 case SpvOpSpecConstant:
502 case SpvOpSpecConstantComposite:
503 // TODO: case SpvOpSpecConstantOp:
504 case SpvOpVariable:
505 case SpvOpAccessChain:
506 case SpvOpInBoundsAccessChain:
507 case SpvOpConvertFToU:
508 case SpvOpConvertFToS:
509 case SpvOpConvertSToF:
510 case SpvOpConvertUToF:
511 case SpvOpUConvert:
512 case SpvOpSConvert:
513 case SpvOpFConvert:
514 case SpvOpConvertPtrToU:
515 // TODO: case SpvOpConvertUToPtr:
516 case SpvOpPtrCastToGeneric:
517 // TODO: case SpvOpGenericCastToPtr:
518 case SpvOpBitcast:
519 // TODO: case SpvOpGenericCastToPtrExplicit:
520 case SpvOpSatConvertSToU:
521 case SpvOpSatConvertUToS:
522 case SpvOpVectorExtractDynamic:
523 case SpvOpCompositeConstruct:
524 case SpvOpCompositeExtract:
525 case SpvOpCopyObject:
526 case SpvOpTranspose:
527 case SpvOpSNegate:
528 case SpvOpFNegate:
529 case SpvOpNot:
530 case SpvOpIAdd:
531 case SpvOpFAdd:
532 case SpvOpISub:
533 case SpvOpFSub:
534 case SpvOpIMul:
535 case SpvOpFMul:
536 case SpvOpUDiv:
537 case SpvOpSDiv:
538 case SpvOpFDiv:
539 case SpvOpUMod:
540 case SpvOpSRem:
541 case SpvOpSMod:
542 case SpvOpVectorTimesScalar:
543 case SpvOpMatrixTimesScalar:
544 case SpvOpVectorTimesMatrix:
545 case SpvOpMatrixTimesVector:
546 case SpvOpMatrixTimesMatrix:
547 case SpvOpOuterProduct:
548 case SpvOpDot:
549 case SpvOpShiftRightLogical:
550 case SpvOpShiftRightArithmetic:
551 case SpvOpShiftLeftLogical:
552 case SpvOpBitwiseOr:
553 case SpvOpBitwiseXor:
554 case SpvOpBitwiseAnd:
555 case SpvOpAny:
556 case SpvOpAll:
557 case SpvOpIsNan:
558 case SpvOpIsInf:
559 case SpvOpIsFinite:
560 case SpvOpIsNormal:
561 case SpvOpSignBitSet:
562 case SpvOpLessOrGreater:
563 case SpvOpOrdered:
564 case SpvOpUnordered:
565 case SpvOpLogicalOr:
566 case SpvOpLogicalAnd:
567 case SpvOpSelect:
568 case SpvOpIEqual:
569 case SpvOpFOrdEqual:
570 case SpvOpFUnordEqual:
571 case SpvOpINotEqual:
572 case SpvOpFOrdNotEqual:
573 case SpvOpFUnordNotEqual:
574 case SpvOpULessThan:
575 case SpvOpSLessThan:
576 case SpvOpFOrdLessThan:
577 case SpvOpFUnordLessThan:
578 case SpvOpUGreaterThan:
579 case SpvOpSGreaterThan:
580 case SpvOpFOrdGreaterThan:
581 case SpvOpFUnordGreaterThan:
582 case SpvOpULessThanEqual:
583 case SpvOpSLessThanEqual:
584 case SpvOpFOrdLessThanEqual:
585 case SpvOpFUnordLessThanEqual:
586 case SpvOpUGreaterThanEqual:
587 case SpvOpSGreaterThanEqual:
588 case SpvOpFOrdGreaterThanEqual:
589 case SpvOpFUnordGreaterThanEqual:
590 case SpvOpDPdx:
591 case SpvOpDPdy:
592 case SpvOpFwidth:
593 case SpvOpDPdxFine:
594 case SpvOpDPdyFine:
595 case SpvOpFwidthFine:
596 case SpvOpDPdxCoarse:
597 case SpvOpDPdyCoarse:
598 case SpvOpFwidthCoarse:
599 case SpvOpReturnValue:
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100600 return true;
601 default:
602 return false;
603 }
604}
605
Lei Zhangb36e7042015-10-28 13:40:52 -0400606int32_t spvOpcodeIsBasicTypeNullable(SpvOp opcode) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100607 switch (opcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400608 case SpvOpTypeBool:
609 case SpvOpTypeInt:
610 case SpvOpTypeFloat:
611 case SpvOpTypePointer:
612 case SpvOpTypeEvent:
613 case SpvOpTypeDeviceEvent:
614 case SpvOpTypeReserveId:
615 case SpvOpTypeQueue:
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100616 return true;
617 default:
618 return false;
619 }
620}
621
Lei Zhang1a0334e2015-11-02 09:41:20 -0500622int32_t spvInstructionIsInBasicBlock(const spv_instruction_t* pFirstInst,
623 const spv_instruction_t* pInst) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100624 while (pFirstInst != pInst) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400625 if (SpvOpFunction == pInst->opcode) break;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100626 pInst--;
627 }
Lei Zhangb36e7042015-10-28 13:40:52 -0400628 if (SpvOpFunction != pInst->opcode) return false;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100629 return true;
630}
631
Lei Zhangb36e7042015-10-28 13:40:52 -0400632int32_t spvOpcodeIsValue(SpvOp opcode) {
Lei Zhang40056702015-09-11 14:31:27 -0400633 if (spvOpcodeIsPointer(opcode)) return true;
634 if (spvOpcodeIsConstant(opcode)) return true;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100635 switch (opcode) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400636 case SpvOpLoad:
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +0100637 // TODO: Other Opcode's resulting in a value
638 return true;
639 default:
640 return false;
641 }
642}
Andrew Woloszyn537e7762015-09-29 11:28:34 -0400643
Lei Zhangb36e7042015-10-28 13:40:52 -0400644int32_t spvOpcodeGeneratesType(SpvOp op) {
Lei Zhang1a0334e2015-11-02 09:41:20 -0500645 switch (op) {
Lei Zhangb36e7042015-10-28 13:40:52 -0400646 case SpvOpTypeVoid:
647 case SpvOpTypeBool:
648 case SpvOpTypeInt:
649 case SpvOpTypeFloat:
650 case SpvOpTypeVector:
651 case SpvOpTypeMatrix:
652 case SpvOpTypeImage:
653 case SpvOpTypeSampler:
654 case SpvOpTypeSampledImage:
655 case SpvOpTypeArray:
656 case SpvOpTypeRuntimeArray:
657 case SpvOpTypeStruct:
658 case SpvOpTypeOpaque:
659 case SpvOpTypePointer:
660 case SpvOpTypeFunction:
661 case SpvOpTypeEvent:
662 case SpvOpTypeDeviceEvent:
663 case SpvOpTypeReserveId:
664 case SpvOpTypeQueue:
665 case SpvOpTypePipe:
Andrew Woloszyn537e7762015-09-29 11:28:34 -0400666 return true;
David Netoaef608c2015-11-02 14:59:02 -0500667 default:
668 // In particular, OpTypeForwardPointer does not generate a type,
669 // but declares a storage class for a pointer type generated
670 // by a different instruction.
671 break;
Andrew Woloszyn537e7762015-09-29 11:28:34 -0400672 }
673 return 0;
674}