blob: 5269fc677ecd258d34f18515d36e675604aeb550 [file] [log] [blame]
sfricke-samsungdb3f3f82022-01-18 06:41:15 -08001/* Copyright (c) 2021-2022 The Khronos Group Inc.
sfricke-samsung962cad92021-04-13 00:46:29 -07002 *
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 *
15 * Author: Spencer Fricke <s.fricke@samsung.com>
16 */
17
18#include "shader_module.h"
19
20#include <sstream>
21#include <string>
22
23#include "vk_layer_data.h"
24#include "vk_layer_utils.h"
Jeremy Gebben5d970742021-05-31 16:04:14 -060025#include "pipeline_state.h"
26#include "descriptor_sets.h"
sfricke-samsung3c5dee22021-10-14 09:58:14 -070027#include "spirv_grammar_helper.h"
sfricke-samsung962cad92021-04-13 00:46:29 -070028
29void decoration_set::merge(decoration_set const &other) {
30 if (other.flags & location_bit) location = other.location;
31 if (other.flags & component_bit) component = other.component;
32 if (other.flags & input_attachment_index_bit) input_attachment_index = other.input_attachment_index;
33 if (other.flags & descriptor_set_bit) descriptor_set = other.descriptor_set;
34 if (other.flags & binding_bit) binding = other.binding;
35 if (other.flags & builtin_bit) builtin = other.builtin;
36 flags |= other.flags;
37}
38
39void decoration_set::add(uint32_t decoration, uint32_t value) {
40 switch (decoration) {
41 case spv::DecorationLocation:
42 flags |= location_bit;
43 location = value;
44 break;
45 case spv::DecorationPatch:
46 flags |= patch_bit;
47 break;
48 case spv::DecorationRelaxedPrecision:
49 flags |= relaxed_precision_bit;
50 break;
51 case spv::DecorationBlock:
52 flags |= block_bit;
53 break;
54 case spv::DecorationBufferBlock:
55 flags |= buffer_block_bit;
56 break;
57 case spv::DecorationComponent:
58 flags |= component_bit;
59 component = value;
60 break;
61 case spv::DecorationInputAttachmentIndex:
62 flags |= input_attachment_index_bit;
63 input_attachment_index = value;
64 break;
65 case spv::DecorationDescriptorSet:
66 flags |= descriptor_set_bit;
67 descriptor_set = value;
68 break;
69 case spv::DecorationBinding:
70 flags |= binding_bit;
71 binding = value;
72 break;
73 case spv::DecorationNonWritable:
74 flags |= nonwritable_bit;
75 break;
76 case spv::DecorationBuiltIn:
77 flags |= builtin_bit;
78 builtin = value;
79 break;
Lionel Landwerlin38d2e122021-07-21 14:21:47 +030080 case spv::DecorationNonReadable:
81 flags |= nonreadable_bit;
82 break;
ziga-lunarg9e94e112021-09-27 00:21:10 +020083 case spv::DecorationPerVertexNV:
84 flags |= per_vertex_bit;
85 break;
86 case spv::DecorationPassthroughNV:
87 flags |= passthrough_bit;
88 break;
sjfrickede734312022-07-14 19:22:43 +090089 case spv::DecorationAliased:
90 flags |= aliased_bit;
91 break;
sfricke-samsung962cad92021-04-13 00:46:29 -070092 }
93}
94
95std::string shader_struct_member::GetLocationDesc(uint32_t index_used_bytes) const {
96 std::string desc = "";
97 if (array_length_hierarchy.size() > 0) {
98 desc += " index:";
99 for (const auto block_size : array_block_size) {
100 desc += "[";
101 desc += std::to_string(index_used_bytes / (block_size * size));
102 desc += "]";
103 index_used_bytes = index_used_bytes % (block_size * size);
104 }
105 }
106 const int struct_members_size = static_cast<int>(struct_members.size());
107 if (struct_members_size > 0) {
108 desc += " member:";
109 for (int i = struct_members_size - 1; i >= 0; --i) {
110 if (index_used_bytes > struct_members[i].offset) {
111 desc += std::to_string(i);
112 desc += struct_members[i].GetLocationDesc(index_used_bytes - struct_members[i].offset);
113 break;
114 }
115 }
116 } else {
117 desc += " offset:";
118 desc += std::to_string(index_used_bytes);
119 }
120 return desc;
121}
122
sfricke-samsung7fac88a2022-01-26 11:44:22 -0800123static uint32_t ExecutionModelToShaderStageFlagBits(uint32_t mode) {
sfricke-samsung962cad92021-04-13 00:46:29 -0700124 switch (mode) {
125 case spv::ExecutionModelVertex:
126 return VK_SHADER_STAGE_VERTEX_BIT;
127 case spv::ExecutionModelTessellationControl:
128 return VK_SHADER_STAGE_TESSELLATION_CONTROL_BIT;
129 case spv::ExecutionModelTessellationEvaluation:
130 return VK_SHADER_STAGE_TESSELLATION_EVALUATION_BIT;
131 case spv::ExecutionModelGeometry:
132 return VK_SHADER_STAGE_GEOMETRY_BIT;
133 case spv::ExecutionModelFragment:
134 return VK_SHADER_STAGE_FRAGMENT_BIT;
135 case spv::ExecutionModelGLCompute:
136 return VK_SHADER_STAGE_COMPUTE_BIT;
sjfricke62366d32022-08-01 21:04:10 +0900137 case spv::ExecutionModelRayGenerationKHR:
138 return VK_SHADER_STAGE_RAYGEN_BIT_KHR;
139 case spv::ExecutionModelAnyHitKHR:
140 return VK_SHADER_STAGE_ANY_HIT_BIT_KHR;
141 case spv::ExecutionModelClosestHitKHR:
142 return VK_SHADER_STAGE_CLOSEST_HIT_BIT_KHR;
143 case spv::ExecutionModelMissKHR:
144 return VK_SHADER_STAGE_MISS_BIT_KHR;
145 case spv::ExecutionModelIntersectionKHR:
146 return VK_SHADER_STAGE_INTERSECTION_BIT_KHR;
147 case spv::ExecutionModelCallableKHR:
148 return VK_SHADER_STAGE_CALLABLE_BIT_KHR;
sfricke-samsung962cad92021-04-13 00:46:29 -0700149 case spv::ExecutionModelTaskNV:
150 return VK_SHADER_STAGE_TASK_BIT_NV;
151 case spv::ExecutionModelMeshNV:
152 return VK_SHADER_STAGE_MESH_BIT_NV;
153 default:
154 return 0;
155 }
156}
157
158// For some analyses, we need to know about all ids referenced by the static call tree of a particular entrypoint. This is
159// important for identifying the set of shader resources actually used by an entrypoint, for example.
160// Note: we only explore parts of the image which might actually contain ids we care about for the above analyses.
161// - NOT the shader input/output interfaces.
162//
163// TODO: The set of interesting opcodes here was determined by eyeballing the SPIRV spec. It might be worth
164// converting parts of this to be generated from the machine-readable spec instead.
165layer_data::unordered_set<uint32_t> SHADER_MODULE_STATE::MarkAccessibleIds(spirv_inst_iter entrypoint) const {
166 layer_data::unordered_set<uint32_t> ids;
Jeremy Gebben84b838b2021-08-23 08:41:39 -0600167 if (entrypoint == end() || !has_valid_spirv) {
168 return ids;
169 }
sfricke-samsung962cad92021-04-13 00:46:29 -0700170 layer_data::unordered_set<uint32_t> worklist;
171 worklist.insert(entrypoint.word(2));
172
173 while (!worklist.empty()) {
174 auto id_iter = worklist.begin();
175 auto id = *id_iter;
176 worklist.erase(id_iter);
177
178 auto insn = get_def(id);
179 if (insn == end()) {
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600180 // ID is something we didn't collect in SpirvStaticData. that's OK -- we'll stumble across all kinds of things here
sfricke-samsung962cad92021-04-13 00:46:29 -0700181 // that we may not care about.
182 continue;
183 }
184
185 // Try to add to the output set
186 if (!ids.insert(id).second) {
187 continue; // If we already saw this id, we don't want to walk it again.
188 }
189
190 switch (insn.opcode()) {
191 case spv::OpFunction:
192 // Scan whole body of the function, enlisting anything interesting
193 while (++insn, insn.opcode() != spv::OpFunctionEnd) {
194 switch (insn.opcode()) {
195 case spv::OpLoad:
196 worklist.insert(insn.word(3)); // ptr
197 break;
198 case spv::OpStore:
199 worklist.insert(insn.word(1)); // ptr
200 break;
201 case spv::OpAccessChain:
202 case spv::OpInBoundsAccessChain:
203 worklist.insert(insn.word(3)); // base ptr
204 break;
205 case spv::OpSampledImage:
206 case spv::OpImageSampleImplicitLod:
207 case spv::OpImageSampleExplicitLod:
208 case spv::OpImageSampleDrefImplicitLod:
209 case spv::OpImageSampleDrefExplicitLod:
210 case spv::OpImageSampleProjImplicitLod:
211 case spv::OpImageSampleProjExplicitLod:
212 case spv::OpImageSampleProjDrefImplicitLod:
213 case spv::OpImageSampleProjDrefExplicitLod:
214 case spv::OpImageFetch:
215 case spv::OpImageGather:
216 case spv::OpImageDrefGather:
217 case spv::OpImageRead:
218 case spv::OpImage:
219 case spv::OpImageQueryFormat:
220 case spv::OpImageQueryOrder:
221 case spv::OpImageQuerySizeLod:
222 case spv::OpImageQuerySize:
223 case spv::OpImageQueryLod:
224 case spv::OpImageQueryLevels:
225 case spv::OpImageQuerySamples:
226 case spv::OpImageSparseSampleImplicitLod:
227 case spv::OpImageSparseSampleExplicitLod:
228 case spv::OpImageSparseSampleDrefImplicitLod:
229 case spv::OpImageSparseSampleDrefExplicitLod:
230 case spv::OpImageSparseSampleProjImplicitLod:
231 case spv::OpImageSparseSampleProjExplicitLod:
232 case spv::OpImageSparseSampleProjDrefImplicitLod:
233 case spv::OpImageSparseSampleProjDrefExplicitLod:
234 case spv::OpImageSparseFetch:
235 case spv::OpImageSparseGather:
236 case spv::OpImageSparseDrefGather:
237 case spv::OpImageTexelPointer:
238 worklist.insert(insn.word(3)); // Image or sampled image
239 break;
240 case spv::OpImageWrite:
241 worklist.insert(insn.word(1)); // Image -- different operand order to above
242 break;
243 case spv::OpFunctionCall:
244 for (uint32_t i = 3; i < insn.len(); i++) {
245 worklist.insert(insn.word(i)); // fn itself, and all args
246 }
247 break;
248
249 case spv::OpExtInst:
250 for (uint32_t i = 5; i < insn.len(); i++) {
251 worklist.insert(insn.word(i)); // Operands to ext inst
252 }
253 break;
254
255 default: {
256 if (AtomicOperation(insn.opcode())) {
257 if (insn.opcode() == spv::OpAtomicStore) {
258 worklist.insert(insn.word(1)); // ptr
259 } else {
260 worklist.insert(insn.word(3)); // ptr
261 }
262 }
263 break;
264 }
265 }
266 }
267 break;
268 }
269 }
270
271 return ids;
272}
273
Jeremy Gebben84b838b2021-08-23 08:41:39 -0600274layer_data::optional<VkPrimitiveTopology> SHADER_MODULE_STATE::GetTopology(const spirv_inst_iter &entrypoint) const {
275 layer_data::optional<VkPrimitiveTopology> result;
276
sfricke-samsung962cad92021-04-13 00:46:29 -0700277 auto entrypoint_id = entrypoint.word(2);
278 bool is_point_mode = false;
279
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600280 auto it = static_data_.execution_mode_inst.find(entrypoint_id);
281 if (it != static_data_.execution_mode_inst.end()) {
sfricke-samsung962cad92021-04-13 00:46:29 -0700282 for (auto insn : it->second) {
283 switch (insn.word(2)) {
284 case spv::ExecutionModePointMode:
285 // In tessellation shaders, PointMode is separate and trumps the tessellation topology.
286 is_point_mode = true;
287 break;
288
289 case spv::ExecutionModeOutputPoints:
Jeremy Gebben84b838b2021-08-23 08:41:39 -0600290 result.emplace(VK_PRIMITIVE_TOPOLOGY_POINT_LIST);
sfricke-samsung962cad92021-04-13 00:46:29 -0700291 break;
292
293 case spv::ExecutionModeIsolines:
294 case spv::ExecutionModeOutputLineStrip:
Ricardo Garcia122f8f02021-09-28 16:47:19 +0200295 case spv::ExecutionModeOutputLinesNV:
Jeremy Gebben84b838b2021-08-23 08:41:39 -0600296 result.emplace(VK_PRIMITIVE_TOPOLOGY_LINE_STRIP);
sfricke-samsung962cad92021-04-13 00:46:29 -0700297 break;
298
299 case spv::ExecutionModeTriangles:
300 case spv::ExecutionModeQuads:
301 case spv::ExecutionModeOutputTriangleStrip:
302 case spv::ExecutionModeOutputTrianglesNV:
Jeremy Gebben84b838b2021-08-23 08:41:39 -0600303 result.emplace(VK_PRIMITIVE_TOPOLOGY_TRIANGLE_STRIP);
sfricke-samsung962cad92021-04-13 00:46:29 -0700304 break;
305 }
306 }
307 }
308
Jeremy Gebben84b838b2021-08-23 08:41:39 -0600309 if (is_point_mode) {
310 result.emplace(VK_PRIMITIVE_TOPOLOGY_POINT_LIST);
311 }
312
313 return result;
sfricke-samsung962cad92021-04-13 00:46:29 -0700314}
315
Nathaniel Cesario3fd4f762022-02-16 16:07:06 -0700316layer_data::optional<VkPrimitiveTopology> SHADER_MODULE_STATE::GetTopology() const {
317 if (static_data_.entry_points.size() > 0) {
318 const auto entrypoint = static_data_.entry_points.cbegin()->second;
319 return GetTopology(get_def(entrypoint.offset));
320 }
321 return {};
322}
323
sfricke-samsungef15e482022-01-26 11:32:49 -0800324SHADER_MODULE_STATE::SpirvStaticData::SpirvStaticData(const SHADER_MODULE_STATE &module_state) {
325 for (auto insn : module_state) {
sfricke-samsung5a48ed42022-02-13 17:37:13 -0800326 const uint32_t result_word = OpcodeResultWord(insn.opcode());
327 if (result_word != 0) {
328 def_index[insn.word(result_word)] = insn.offset();
329 }
330
sfricke-samsung962cad92021-04-13 00:46:29 -0700331 switch (insn.opcode()) {
sfricke-samsung962cad92021-04-13 00:46:29 -0700332 // Specialization constants
333 case spv::OpSpecConstantTrue:
334 case spv::OpSpecConstantFalse:
335 case spv::OpSpecConstant:
336 case spv::OpSpecConstantComposite:
337 case spv::OpSpecConstantOp:
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600338 has_specialization_constants = true;
sfricke-samsung962cad92021-04-13 00:46:29 -0700339 break;
340
sfricke-samsung962cad92021-04-13 00:46:29 -0700341 // Decorations
342 case spv::OpDecorate: {
343 auto target_id = insn.word(1);
344 decorations[target_id].add(insn.word(2), insn.len() > 3u ? insn.word(3) : 0u);
345 decoration_inst.push_back(insn);
346 if (insn.word(2) == spv::DecorationBuiltIn) {
347 builtin_decoration_list.emplace_back(insn.offset(), static_cast<spv::BuiltIn>(insn.word(3)));
Nathaniel Cesariocf69bda2021-06-22 13:23:42 -0600348 } else if (insn.word(2) == spv::DecorationSpecId) {
349 spec_const_map[insn.word(3)] = target_id;
sfricke-samsung962cad92021-04-13 00:46:29 -0700350 }
351
352 } break;
353 case spv::OpGroupDecorate: {
354 auto const &src = decorations[insn.word(1)];
355 for (auto i = 2u; i < insn.len(); i++) decorations[insn.word(i)].merge(src);
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600356 has_group_decoration = true;
357 } break;
358 case spv::OpDecorationGroup:
359 case spv::OpGroupMemberDecorate: {
360 has_group_decoration = true;
sfricke-samsung962cad92021-04-13 00:46:29 -0700361 } break;
362 case spv::OpMemberDecorate: {
363 member_decoration_inst.push_back(insn);
364 if (insn.word(3) == spv::DecorationBuiltIn) {
365 builtin_decoration_list.emplace_back(insn.offset(), static_cast<spv::BuiltIn>(insn.word(4)));
366 }
367 } break;
368
ziga-lunargbe003732022-04-22 00:20:13 +0200369 case spv::OpCapability:
370 capability_list.push_back(static_cast<spv::Capability>(insn.word(1)));
371 break;
372
sjfricke44d663c2022-06-01 06:42:58 +0900373 case spv::OpVariable:
374 variable_inst.push_back(insn);
375 break;
376
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600377 // Execution Mode
sfricke-samsung61d50ec2022-02-13 17:01:25 -0800378 case spv::OpExecutionMode:
379 case spv::OpExecutionModeId: {
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600380 execution_mode_inst[insn.word(1)].push_back(insn);
381 } break;
sjfricke657dfdc2022-08-25 23:40:32 +0900382 // Listed from vkspec.html#ray-tracing-repack
ziga-lunarge25f5f02022-04-16 15:07:35 +0200383 case spv::OpTraceRayKHR:
384 case spv::OpTraceRayMotionNV:
385 case spv::OpReportIntersectionKHR:
386 case spv::OpExecuteCallableKHR:
387 has_invocation_repack_instruction = true;
388 break;
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600389
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600390 default:
391 if (AtomicOperation(insn.opcode()) == true) {
392 // All atomics have a pointer referenced
393 spirv_inst_iter access;
394 if (insn.opcode() == spv::OpAtomicStore) {
sfricke-samsungef15e482022-01-26 11:32:49 -0800395 access = module_state.get_def(insn.word(1));
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600396 } else {
sfricke-samsungef15e482022-01-26 11:32:49 -0800397 access = module_state.get_def(insn.word(3));
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600398 }
399
400 atomic_instruction atomic;
401
sfricke-samsungef15e482022-01-26 11:32:49 -0800402 auto pointer = module_state.get_def(access.word(1));
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600403 // spirv-val should catch if not pointer
404 assert(pointer.opcode() == spv::OpTypePointer);
405 atomic.storage_class = pointer.word(2);
406
sfricke-samsungef15e482022-01-26 11:32:49 -0800407 auto data_type = module_state.get_def(pointer.word(3));
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600408 atomic.type = data_type.opcode();
409
410 // TODO - Should have a proper GetBitWidth like spirv-val does
411 assert(data_type.opcode() == spv::OpTypeFloat || data_type.opcode() == spv::OpTypeInt);
412 atomic.bit_width = data_type.word(2);
413
414 atomic_inst[insn.offset()] = atomic;
415 }
416 // We don't care about any other defs for now.
417 break;
418 }
419 }
420
sfricke-samsungef15e482022-01-26 11:32:49 -0800421 entry_points = SHADER_MODULE_STATE::ProcessEntryPoints(module_state);
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600422 multiple_entry_points = entry_points.size() > 1;
423}
424
425// static
426std::unordered_multimap<std::string, SHADER_MODULE_STATE::EntryPoint> SHADER_MODULE_STATE::ProcessEntryPoints(
sfricke-samsungef15e482022-01-26 11:32:49 -0800427 const SHADER_MODULE_STATE &module_state) {
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600428 std::unordered_multimap<std::string, SHADER_MODULE_STATE::EntryPoint> entry_points;
429 function_set func_set = {};
430 EntryPoint *entry_point = nullptr;
431
sfricke-samsungef15e482022-01-26 11:32:49 -0800432 for (auto insn : module_state) {
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600433 // offset is not 0, it means it's updated and the offset is in a Function.
434 if (func_set.offset) {
435 func_set.op_lists.emplace(insn.opcode(), insn.offset());
436 } else if (entry_point) {
437 entry_point->decorate_list.emplace(insn.opcode(), insn.offset());
438 }
439
440 switch (insn.opcode()) {
441 // Functions
442 case spv::OpFunction:
443 func_set.id = insn.word(2);
444 func_set.offset = insn.offset();
445 func_set.op_lists.clear();
446 break;
447
448 // Entry points ... add to the entrypoint table
449 case spv::OpEntryPoint: {
sfricke-samsung962cad92021-04-13 00:46:29 -0700450 // Entry points do not have an id (the id is the function id) and thus need their own table
451 auto entrypoint_name = reinterpret_cast<char const *>(&insn.word(3));
452 auto execution_model = insn.word(1);
453 auto entrypoint_stage = ExecutionModelToShaderStageFlagBits(execution_model);
454 entry_points.emplace(entrypoint_name,
455 EntryPoint{insn.offset(), static_cast<VkShaderStageFlagBits>(entrypoint_stage)});
456
457 auto range = entry_points.equal_range(entrypoint_name);
458 for (auto it = range.first; it != range.second; ++it) {
459 if (it->second.offset == insn.offset()) {
460 entry_point = &(it->second);
461 break;
462 }
463 }
464 assert(entry_point != nullptr);
465 break;
466 }
467 case spv::OpFunctionEnd: {
468 assert(entry_point != nullptr);
469 func_set.length = insn.offset() - func_set.offset;
470 entry_point->function_set_list.emplace_back(func_set);
471 break;
472 }
sfricke-samsung962cad92021-04-13 00:46:29 -0700473 }
474 }
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600475
sfricke-samsungef15e482022-01-26 11:32:49 -0800476 SHADER_MODULE_STATE::SetPushConstantUsedInShader(module_state, entry_points);
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600477 return entry_points;
sfricke-samsung962cad92021-04-13 00:46:29 -0700478}
479
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600480void SHADER_MODULE_STATE::PreprocessShaderBinary(const spv_target_env env) {
481 if (static_data_.has_group_decoration) {
sfricke-samsung962cad92021-04-13 00:46:29 -0700482 spvtools::Optimizer optimizer(env);
483 optimizer.RegisterPass(spvtools::CreateFlattenDecorationPass());
484 std::vector<uint32_t> optimized_binary;
485 // Run optimizer to flatten decorations only, set skip_validation so as to not re-run validator
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600486 auto result = optimizer.Run(words.data(), words.size(), &optimized_binary, spvtools::ValidatorOptions(), true);
487
sfricke-samsung962cad92021-04-13 00:46:29 -0700488 if (result) {
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600489 // NOTE: We need to update words with the result from the spirv-tools optimizer.
490 // **THIS ONLY HAPPENS ON INITIALIZATION**. words should remain const for the lifetime
491 // of the SHADER_MODULE_STATE instance.
492 *const_cast<std::vector<uint32_t> *>(&words) = std::move(optimized_binary);
sfricke-samsung962cad92021-04-13 00:46:29 -0700493 }
494 }
sfricke-samsung962cad92021-04-13 00:46:29 -0700495}
496
sfricke-samsung7fac88a2022-01-26 11:44:22 -0800497void SHADER_MODULE_STATE::DescribeTypeInner(std::ostringstream &ss, uint32_t type) const {
sfricke-samsung962cad92021-04-13 00:46:29 -0700498 auto insn = get_def(type);
499 assert(insn != end());
500
501 switch (insn.opcode()) {
502 case spv::OpTypeBool:
503 ss << "bool";
504 break;
505 case spv::OpTypeInt:
506 ss << (insn.word(3) ? 's' : 'u') << "int" << insn.word(2);
507 break;
508 case spv::OpTypeFloat:
509 ss << "float" << insn.word(2);
510 break;
511 case spv::OpTypeVector:
512 ss << "vec" << insn.word(3) << " of ";
513 DescribeTypeInner(ss, insn.word(2));
514 break;
515 case spv::OpTypeMatrix:
516 ss << "mat" << insn.word(3) << " of ";
517 DescribeTypeInner(ss, insn.word(2));
518 break;
519 case spv::OpTypeArray:
520 ss << "arr[" << GetConstantValueById(insn.word(3)) << "] of ";
521 DescribeTypeInner(ss, insn.word(2));
522 break;
523 case spv::OpTypeRuntimeArray:
524 ss << "runtime arr[] of ";
525 DescribeTypeInner(ss, insn.word(2));
526 break;
527 case spv::OpTypePointer:
sjfricke657dfdc2022-08-25 23:40:32 +0900528 ss << "ptr to " << string_SpvStorageClass(insn.word(2)) << " ";
sfricke-samsung962cad92021-04-13 00:46:29 -0700529 DescribeTypeInner(ss, insn.word(3));
530 break;
531 case spv::OpTypeStruct: {
532 ss << "struct of (";
sfricke-samsung7fac88a2022-01-26 11:44:22 -0800533 for (uint32_t i = 2; i < insn.len(); i++) {
sfricke-samsung962cad92021-04-13 00:46:29 -0700534 DescribeTypeInner(ss, insn.word(i));
535 if (i == insn.len() - 1) {
536 ss << ")";
537 } else {
538 ss << ", ";
539 }
540 }
541 break;
542 }
543 case spv::OpTypeSampler:
544 ss << "sampler";
545 break;
546 case spv::OpTypeSampledImage:
547 ss << "sampler+";
548 DescribeTypeInner(ss, insn.word(2));
549 break;
550 case spv::OpTypeImage:
551 ss << "image(dim=" << insn.word(3) << ", sampled=" << insn.word(7) << ")";
552 break;
553 case spv::OpTypeAccelerationStructureNV:
554 ss << "accelerationStruture";
555 break;
556 default:
557 ss << "oddtype";
558 break;
559 }
560}
561
sfricke-samsung7fac88a2022-01-26 11:44:22 -0800562std::string SHADER_MODULE_STATE::DescribeType(uint32_t type) const {
sfricke-samsung962cad92021-04-13 00:46:29 -0700563 std::ostringstream ss;
564 DescribeTypeInner(ss, type);
565 return ss.str();
566}
567
sfricke-samsung7a9bdca2022-01-24 14:38:03 -0800568std::string SHADER_MODULE_STATE::DescribeInstruction(const spirv_inst_iter &insn) const {
569 std::ostringstream ss;
570 const uint32_t opcode = insn.opcode();
571 uint32_t operand_offset = 1; // where to start printing operands
572 // common disassembled for SPIR-V is
573 // %result = Opcode %result_type %operands
574 if (OpcodeHasResult(opcode)) {
575 operand_offset++;
576 ss << "%" << (OpcodeHasType(opcode) ? insn.word(2) : insn.word(1)) << " = ";
577 }
578
579 ss << string_SpvOpcode(opcode);
580
581 if (OpcodeHasType(opcode)) {
582 operand_offset++;
583 ss << " %" << insn.word(1);
584 }
585
sfricke-samsunged00aa42022-01-27 19:03:01 -0800586 // TODO - For now don't list the '%' for any operands since they are only for reference IDs. Without generating a table of each
587 // instructions operand types and covering the many edge cases (such as optional, paired, or variable operands) this is the
588 // simplest way to print the instruction and give the developer something to look into when an error occurs.
589 //
590 // For now this safely should be able to assume it will never come across a LiteralString such as in OpExtInstImport or
591 // OpEntryPoint
sfricke-samsung7a9bdca2022-01-24 14:38:03 -0800592 for (uint32_t i = operand_offset; i < insn.len(); i++) {
sfricke-samsunged00aa42022-01-27 19:03:01 -0800593 ss << " " << insn.word(i);
sfricke-samsung7a9bdca2022-01-24 14:38:03 -0800594 }
595 return ss.str();
596}
597
sfricke-samsung962cad92021-04-13 00:46:29 -0700598const SHADER_MODULE_STATE::EntryPoint *SHADER_MODULE_STATE::FindEntrypointStruct(char const *name,
599 VkShaderStageFlagBits stageBits) const {
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600600 auto range = static_data_.entry_points.equal_range(name);
sfricke-samsung962cad92021-04-13 00:46:29 -0700601 for (auto it = range.first; it != range.second; ++it) {
602 if (it->second.stage == stageBits) {
603 return &(it->second);
604 }
605 }
606 return nullptr;
607}
608
609spirv_inst_iter SHADER_MODULE_STATE::FindEntrypoint(char const *name, VkShaderStageFlagBits stageBits) const {
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -0600610 auto range = static_data_.entry_points.equal_range(name);
sfricke-samsung962cad92021-04-13 00:46:29 -0700611 for (auto it = range.first; it != range.second; ++it) {
612 if (it->second.stage == stageBits) {
613 return at(it->second.offset);
614 }
615 }
616 return end();
617}
618
619// Because the following is legal, need the entry point
620// OpEntryPoint GLCompute %main "name_a"
621// OpEntryPoint GLCompute %main "name_b"
sfricke-samsung61d50ec2022-02-13 17:01:25 -0800622// Assumes shader module contains no spec constants used to set the local size values
sfricke-samsung962cad92021-04-13 00:46:29 -0700623bool SHADER_MODULE_STATE::FindLocalSize(const spirv_inst_iter &entrypoint, uint32_t &local_size_x, uint32_t &local_size_y,
624 uint32_t &local_size_z) const {
sfricke-samsung61d50ec2022-02-13 17:01:25 -0800625 // "If an object is decorated with the WorkgroupSize decoration, this takes precedence over any LocalSize or LocalSizeId
626 // execution mode."
627 for (const auto &builtin : static_data_.builtin_decoration_list) {
628 if (builtin.builtin == spv::BuiltInWorkgroupSize) {
629 const uint32_t workgroup_size_id = at(builtin.offset).word(1);
630 auto composite_def = get_def(workgroup_size_id);
631 if (composite_def.opcode() == spv::OpConstantComposite) {
632 // VUID-WorkgroupSize-WorkgroupSize-04427 makes sure this is a OpTypeVector of int32
sjfricke3b0cb102022-08-10 16:27:45 +0900633 local_size_x = GetConstantValueById(composite_def.word(3));
634 local_size_y = GetConstantValueById(composite_def.word(4));
635 local_size_z = GetConstantValueById(composite_def.word(5));
sfricke-samsung962cad92021-04-13 00:46:29 -0700636 return true;
637 }
638 }
639 }
sfricke-samsung61d50ec2022-02-13 17:01:25 -0800640
641 auto entrypoint_id = entrypoint.word(2);
642 auto it = static_data_.execution_mode_inst.find(entrypoint_id);
643 if (it != static_data_.execution_mode_inst.end()) {
644 for (auto insn : it->second) {
645 if (insn.opcode() == spv::OpExecutionMode && insn.word(2) == spv::ExecutionModeLocalSize) {
646 local_size_x = insn.word(3);
647 local_size_y = insn.word(4);
648 local_size_z = insn.word(5);
649 return true;
650 } else if (insn.opcode() == spv::OpExecutionModeId && insn.word(2) == spv::ExecutionModeLocalSizeId) {
651 local_size_x = GetConstantValueById(insn.word(3));
652 local_size_y = GetConstantValueById(insn.word(4));
653 local_size_z = GetConstantValueById(insn.word(5));
654 return true;
655 }
656 }
657 }
658 return false; // not found
sfricke-samsung962cad92021-04-13 00:46:29 -0700659}
660
661// If the instruction at id is a constant or copy of a constant, returns a valid iterator pointing to that instruction.
662// Otherwise, returns src->end().
sfricke-samsung7fac88a2022-01-26 11:44:22 -0800663spirv_inst_iter SHADER_MODULE_STATE::GetConstantDef(uint32_t id) const {
sfricke-samsung962cad92021-04-13 00:46:29 -0700664 auto value = get_def(id);
665
666 // If id is a copy, see where it was copied from
667 if ((end() != value) && ((value.opcode() == spv::OpCopyObject) || (value.opcode() == spv::OpCopyLogical))) {
668 id = value.word(3);
669 value = get_def(id);
670 }
671
672 if ((end() != value) && (value.opcode() == spv::OpConstant)) {
673 return value;
674 }
675 return end();
676}
677
sjfricke3b0cb102022-08-10 16:27:45 +0900678// While simple, function name provides a more human readable description why word(3) is used
679uint32_t SHADER_MODULE_STATE::GetConstantValue(const spirv_inst_iter &itr) const {
sjfricke284a13f2022-08-16 15:34:31 +0900680 // This should be a OpConstant (not a OpSpecConstant), if this asserts then 2 things are happening
681 // 1. This function is being used where we don't actually know it is a constant and is a bug in the validation layers
682 // 2. The CreateFoldSpecConstantOpAndCompositePass didn't fully fold everything and is a bug in spirv-opt
sjfricke3b0cb102022-08-10 16:27:45 +0900683 assert(itr.opcode() == spv::OpConstant);
684 return itr.word(3);
685}
686
sfricke-samsung962cad92021-04-13 00:46:29 -0700687// Either returns the constant value described by the instruction at id, or 1
sfricke-samsung7fac88a2022-01-26 11:44:22 -0800688uint32_t SHADER_MODULE_STATE::GetConstantValueById(uint32_t id) const {
sfricke-samsung962cad92021-04-13 00:46:29 -0700689 auto value = GetConstantDef(id);
690
691 if (end() == value) {
692 // TODO: Either ensure that the specialization transform is already performed on a module we're
693 // considering here, OR -- specialize on the fly now.
694 return 1;
695 }
sjfricke3b0cb102022-08-10 16:27:45 +0900696
sfricke-samsung962cad92021-04-13 00:46:29 -0700697 return GetConstantValue(value);
698}
699
700// Returns an int32_t corresponding to the spv::Dim of the given resource, when positive, and corresponding to an unknown type, when
701// negative.
702int32_t SHADER_MODULE_STATE::GetShaderResourceDimensionality(const interface_var &resource) const {
703 auto type = get_def(resource.type_id);
704 while (true) {
705 switch (type.opcode()) {
706 case spv::OpTypeSampledImage:
707 type = get_def(type.word(2));
708 break;
709 case spv::OpTypePointer:
710 type = get_def(type.word(3));
711 break;
712 case spv::OpTypeImage:
713 return type.word(3);
714 default:
715 return -1;
716 }
717 }
718}
719
sfricke-samsung7fac88a2022-01-26 11:44:22 -0800720uint32_t SHADER_MODULE_STATE::GetLocationsConsumedByType(uint32_t type, bool strip_array_level) const {
sfricke-samsung962cad92021-04-13 00:46:29 -0700721 auto insn = get_def(type);
722 assert(insn != end());
723
724 switch (insn.opcode()) {
725 case spv::OpTypePointer:
726 // See through the ptr -- this is only ever at the toplevel for graphics shaders we're never actually passing
727 // pointers around.
728 return GetLocationsConsumedByType(insn.word(3), strip_array_level);
729 case spv::OpTypeArray:
730 if (strip_array_level) {
731 return GetLocationsConsumedByType(insn.word(2), false);
732 } else {
733 return GetConstantValueById(insn.word(3)) * GetLocationsConsumedByType(insn.word(2), false);
734 }
735 case spv::OpTypeMatrix:
736 // Num locations is the dimension * element size
737 return insn.word(3) * GetLocationsConsumedByType(insn.word(2), false);
738 case spv::OpTypeVector: {
739 auto scalar_type = get_def(insn.word(2));
740 auto bit_width =
741 (scalar_type.opcode() == spv::OpTypeInt || scalar_type.opcode() == spv::OpTypeFloat) ? scalar_type.word(2) : 32;
742
743 // Locations are 128-bit wide; 3- and 4-component vectors of 64 bit types require two.
744 return (bit_width * insn.word(3) + 127) / 128;
745 }
746 default:
747 // Everything else is just 1.
748 return 1;
749
750 // TODO: extend to handle 64bit scalar types, whose vectors may need multiple locations.
751 }
752}
753
sfricke-samsung7fac88a2022-01-26 11:44:22 -0800754uint32_t SHADER_MODULE_STATE::GetComponentsConsumedByType(uint32_t type, bool strip_array_level) const {
sfricke-samsung962cad92021-04-13 00:46:29 -0700755 auto insn = get_def(type);
756 assert(insn != end());
757
758 switch (insn.opcode()) {
759 case spv::OpTypePointer:
760 // See through the ptr -- this is only ever at the toplevel for graphics shaders we're never actually passing
761 // pointers around.
762 return GetComponentsConsumedByType(insn.word(3), strip_array_level);
763 case spv::OpTypeStruct: {
764 uint32_t sum = 0;
765 for (uint32_t i = 2; i < insn.len(); i++) { // i=2 to skip word(0) and word(1)=ID of struct
766 sum += GetComponentsConsumedByType(insn.word(i), false);
767 }
768 return sum;
769 }
770 case spv::OpTypeArray:
771 if (strip_array_level) {
772 return GetComponentsConsumedByType(insn.word(2), false);
773 } else {
774 return GetConstantValueById(insn.word(3)) * GetComponentsConsumedByType(insn.word(2), false);
775 }
776 case spv::OpTypeMatrix:
777 // Num locations is the dimension * element size
778 return insn.word(3) * GetComponentsConsumedByType(insn.word(2), false);
779 case spv::OpTypeVector: {
780 auto scalar_type = get_def(insn.word(2));
781 auto bit_width =
782 (scalar_type.opcode() == spv::OpTypeInt || scalar_type.opcode() == spv::OpTypeFloat) ? scalar_type.word(2) : 32;
783 // One component is 32-bit
784 return (bit_width * insn.word(3) + 31) / 32;
785 }
786 case spv::OpTypeFloat: {
787 auto bit_width = insn.word(2);
788 return (bit_width + 31) / 32;
789 }
790 case spv::OpTypeInt: {
791 auto bit_width = insn.word(2);
792 return (bit_width + 31) / 32;
793 }
794 case spv::OpConstant:
795 return GetComponentsConsumedByType(insn.word(1), false);
796 default:
797 return 0;
798 }
799}
800
801// characterizes a SPIR-V type appearing in an interface to a FF stage, for comparison to a VkFormat's characterization above.
802// also used for input attachments, as we statically know their format.
sfricke-samsung7fac88a2022-01-26 11:44:22 -0800803uint32_t SHADER_MODULE_STATE::GetFundamentalType(uint32_t type) const {
sfricke-samsung962cad92021-04-13 00:46:29 -0700804 auto insn = get_def(type);
805 assert(insn != end());
806
807 switch (insn.opcode()) {
808 case spv::OpTypeInt:
809 return insn.word(3) ? FORMAT_TYPE_SINT : FORMAT_TYPE_UINT;
810 case spv::OpTypeFloat:
811 return FORMAT_TYPE_FLOAT;
812 case spv::OpTypeVector:
813 case spv::OpTypeMatrix:
814 case spv::OpTypeArray:
815 case spv::OpTypeRuntimeArray:
816 case spv::OpTypeImage:
817 return GetFundamentalType(insn.word(2));
818 case spv::OpTypePointer:
819 return GetFundamentalType(insn.word(3));
820
821 default:
822 return 0;
823 }
824}
825
826spirv_inst_iter SHADER_MODULE_STATE::GetStructType(spirv_inst_iter def, bool is_array_of_verts) const {
827 while (true) {
828 if (def.opcode() == spv::OpTypePointer) {
829 def = get_def(def.word(3));
830 } else if (def.opcode() == spv::OpTypeArray && is_array_of_verts) {
831 def = get_def(def.word(2));
832 is_array_of_verts = false;
833 } else if (def.opcode() == spv::OpTypeStruct) {
834 return def;
835 } else {
836 return end();
837 }
838 }
839}
840
sfricke-samsungad55ccc2022-01-19 20:06:17 -0800841void SHADER_MODULE_STATE::DefineStructMember(const spirv_inst_iter &it, const std::vector<uint32_t> &member_decorate_offsets,
sfricke-samsung962cad92021-04-13 00:46:29 -0700842 shader_struct_member &data) const {
843 const auto struct_it = GetStructType(it, false);
844 assert(struct_it != end());
845 data.size = 0;
846
847 shader_struct_member data1;
848 uint32_t i = 2;
849 uint32_t local_offset = 0;
850 std::vector<uint32_t> offsets;
851 offsets.resize(struct_it.len() - i);
852
853 // The members of struct in SPRIV_R aren't always sort, so we need to know their order.
sfricke-samsungad55ccc2022-01-19 20:06:17 -0800854 for (const auto offset : member_decorate_offsets) {
sfricke-samsung962cad92021-04-13 00:46:29 -0700855 const auto member_decorate = at(offset);
856 if (member_decorate.word(1) != struct_it.word(1)) {
857 continue;
858 }
859
860 offsets[member_decorate.word(2)] = member_decorate.word(4);
861 }
862
863 for (const auto offset : offsets) {
864 local_offset = offset;
865 data1 = {};
866 data1.root = data.root;
867 data1.offset = local_offset;
868 auto def_member = get_def(struct_it.word(i));
869
870 // Array could be multi-dimensional
871 while (def_member.opcode() == spv::OpTypeArray) {
872 const auto len_id = def_member.word(3);
873 const auto def_len = get_def(len_id);
874 data1.array_length_hierarchy.emplace_back(def_len.word(3)); // array length
875 def_member = get_def(def_member.word(2));
876 }
877
878 if (def_member.opcode() == spv::OpTypeStruct) {
sfricke-samsungad55ccc2022-01-19 20:06:17 -0800879 DefineStructMember(def_member, member_decorate_offsets, data1);
sfricke-samsung962cad92021-04-13 00:46:29 -0700880 } else if (def_member.opcode() == spv::OpTypePointer) {
881 if (def_member.word(2) == spv::StorageClassPhysicalStorageBuffer) {
882 // If it's a pointer with PhysicalStorageBuffer class, this member is essentially a uint64_t containing an address
883 // that "points to something."
884 data1.size = 8;
885 } else {
886 // If it's OpTypePointer. it means the member is a buffer, the type will be TypePointer, and then struct
sfricke-samsungad55ccc2022-01-19 20:06:17 -0800887 DefineStructMember(def_member, member_decorate_offsets, data1);
sfricke-samsung962cad92021-04-13 00:46:29 -0700888 }
889 } else {
890 if (def_member.opcode() == spv::OpTypeMatrix) {
891 data1.array_length_hierarchy.emplace_back(def_member.word(3)); // matrix's columns. matrix's row is vector.
892 def_member = get_def(def_member.word(2));
893 }
894
895 if (def_member.opcode() == spv::OpTypeVector) {
896 data1.array_length_hierarchy.emplace_back(def_member.word(3)); // vector length
897 def_member = get_def(def_member.word(2));
898 }
899
900 // Get scalar type size. The value in SPRV-R is bit. It needs to translate to byte.
901 data1.size = (def_member.word(2) / 8);
902 }
903 const auto array_length_hierarchy_szie = data1.array_length_hierarchy.size();
904 if (array_length_hierarchy_szie > 0) {
905 data1.array_block_size.resize(array_length_hierarchy_szie, 1);
906
907 for (int i2 = static_cast<int>(array_length_hierarchy_szie - 1); i2 > 0; --i2) {
908 data1.array_block_size[i2 - 1] = data1.array_length_hierarchy[i2] * data1.array_block_size[i2];
909 }
910 }
911 data.struct_members.emplace_back(data1);
912 ++i;
913 }
914 uint32_t total_array_length = 1;
915 for (const auto length : data1.array_length_hierarchy) {
916 total_array_length *= length;
917 }
918 data.size = local_offset + data1.size * total_array_length;
919}
920
921static uint32_t UpdateOffset(uint32_t offset, const std::vector<uint32_t> &array_indices, const shader_struct_member &data) {
922 int array_indices_size = static_cast<int>(array_indices.size());
923 if (array_indices_size) {
924 uint32_t array_index = 0;
925 uint32_t i = 0;
926 for (const auto index : array_indices) {
927 array_index += (data.array_block_size[i] * index);
928 ++i;
929 }
930 offset += (array_index * data.size);
931 }
932 return offset;
933}
934
935static void SetUsedBytes(uint32_t offset, const std::vector<uint32_t> &array_indices, const shader_struct_member &data) {
936 int array_indices_size = static_cast<int>(array_indices.size());
937 uint32_t block_memory_size = data.size;
938 for (uint32_t i = static_cast<int>(array_indices_size); i < data.array_length_hierarchy.size(); ++i) {
939 block_memory_size *= data.array_length_hierarchy[i];
940 }
941
942 offset = UpdateOffset(offset, array_indices, data);
943
944 uint32_t end = offset + block_memory_size;
945 auto used_bytes = data.GetUsedbytes();
946 if (used_bytes->size() < end) {
947 used_bytes->resize(end, 0);
948 }
949 std::memset(used_bytes->data() + offset, true, static_cast<std::size_t>(block_memory_size));
950}
951
952void SHADER_MODULE_STATE::RunUsedArray(uint32_t offset, std::vector<uint32_t> array_indices, uint32_t access_chain_word_index,
953 spirv_inst_iter &access_chain_it, const shader_struct_member &data) const {
954 if (access_chain_word_index < access_chain_it.len()) {
955 if (data.array_length_hierarchy.size() > array_indices.size()) {
956 auto def_it = get_def(access_chain_it.word(access_chain_word_index));
957 ++access_chain_word_index;
958
959 if (def_it != end() && def_it.opcode() == spv::OpConstant) {
960 array_indices.emplace_back(def_it.word(3));
961 RunUsedArray(offset, array_indices, access_chain_word_index, access_chain_it, data);
962 } else {
963 // If it is a variable, set the all array is used.
964 if (access_chain_word_index < access_chain_it.len()) {
965 uint32_t array_length = data.array_length_hierarchy[array_indices.size()];
966 for (uint32_t i = 0; i < array_length; ++i) {
967 auto array_indices2 = array_indices;
968 array_indices2.emplace_back(i);
969 RunUsedArray(offset, array_indices2, access_chain_word_index, access_chain_it, data);
970 }
971 } else {
972 SetUsedBytes(offset, array_indices, data);
973 }
974 }
975 } else {
976 offset = UpdateOffset(offset, array_indices, data);
977 RunUsedStruct(offset, access_chain_word_index, access_chain_it, data);
978 }
979 } else {
980 SetUsedBytes(offset, array_indices, data);
981 }
982}
983
984void SHADER_MODULE_STATE::RunUsedStruct(uint32_t offset, uint32_t access_chain_word_index, spirv_inst_iter &access_chain_it,
985 const shader_struct_member &data) const {
986 std::vector<uint32_t> array_indices_emptry;
987
988 if (access_chain_word_index < access_chain_it.len()) {
989 auto strcut_member_index = GetConstantValueById(access_chain_it.word(access_chain_word_index));
990 ++access_chain_word_index;
991
992 auto data1 = data.struct_members[strcut_member_index];
993 RunUsedArray(offset + data1.offset, array_indices_emptry, access_chain_word_index, access_chain_it, data1);
994 }
995}
996
997void SHADER_MODULE_STATE::SetUsedStructMember(const uint32_t variable_id, const std::vector<function_set> &function_set_list,
998 const shader_struct_member &data) const {
999 for (const auto &func_set : function_set_list) {
1000 auto range = func_set.op_lists.equal_range(spv::OpAccessChain);
1001 for (auto it = range.first; it != range.second; ++it) {
1002 auto access_chain = at(it->second);
1003 if (access_chain.word(3) == variable_id) {
1004 RunUsedStruct(0, 4, access_chain, data);
1005 }
1006 }
1007 }
1008}
1009
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -06001010// static
1011void SHADER_MODULE_STATE::SetPushConstantUsedInShader(
sfricke-samsungef15e482022-01-26 11:32:49 -08001012 const SHADER_MODULE_STATE &module_state, std::unordered_multimap<std::string, SHADER_MODULE_STATE::EntryPoint> &entry_points) {
sfricke-samsung962cad92021-04-13 00:46:29 -07001013 for (auto &entrypoint : entry_points) {
1014 auto range = entrypoint.second.decorate_list.equal_range(spv::OpVariable);
1015 for (auto it = range.first; it != range.second; ++it) {
sfricke-samsungef15e482022-01-26 11:32:49 -08001016 const auto def_insn = module_state.at(it->second);
sfricke-samsung962cad92021-04-13 00:46:29 -07001017
1018 if (def_insn.word(3) == spv::StorageClassPushConstant) {
sfricke-samsungef15e482022-01-26 11:32:49 -08001019 spirv_inst_iter type = module_state.get_def(def_insn.word(1));
sfricke-samsung962cad92021-04-13 00:46:29 -07001020 const auto range2 = entrypoint.second.decorate_list.equal_range(spv::OpMemberDecorate);
1021 std::vector<uint32_t> offsets;
1022
1023 for (auto it2 = range2.first; it2 != range2.second; ++it2) {
sfricke-samsungef15e482022-01-26 11:32:49 -08001024 auto member_decorate = module_state.at(it2->second);
sfricke-samsung962cad92021-04-13 00:46:29 -07001025 if (member_decorate.len() == 5 && member_decorate.word(3) == spv::DecorationOffset) {
1026 offsets.emplace_back(member_decorate.offset());
1027 }
1028 }
1029 entrypoint.second.push_constant_used_in_shader.root = &entrypoint.second.push_constant_used_in_shader;
sfricke-samsungef15e482022-01-26 11:32:49 -08001030 module_state.DefineStructMember(type, offsets, entrypoint.second.push_constant_used_in_shader);
1031 module_state.SetUsedStructMember(def_insn.word(2), entrypoint.second.function_set_list,
1032 entrypoint.second.push_constant_used_in_shader);
sfricke-samsung962cad92021-04-13 00:46:29 -07001033 }
1034 }
1035 }
1036}
1037
1038uint32_t SHADER_MODULE_STATE::DescriptorTypeToReqs(uint32_t type_id) const {
1039 auto type = get_def(type_id);
1040
1041 while (true) {
1042 switch (type.opcode()) {
1043 case spv::OpTypeArray:
1044 case spv::OpTypeRuntimeArray:
1045 case spv::OpTypeSampledImage:
1046 type = get_def(type.word(2));
1047 break;
1048 case spv::OpTypePointer:
1049 type = get_def(type.word(3));
1050 break;
1051 case spv::OpTypeImage: {
1052 auto dim = type.word(3);
1053 auto arrayed = type.word(5);
1054 auto msaa = type.word(6);
1055
1056 uint32_t bits = 0;
1057 switch (GetFundamentalType(type.word(2))) {
1058 case FORMAT_TYPE_FLOAT:
1059 bits = DESCRIPTOR_REQ_COMPONENT_TYPE_FLOAT;
1060 break;
1061 case FORMAT_TYPE_UINT:
1062 bits = DESCRIPTOR_REQ_COMPONENT_TYPE_UINT;
1063 break;
1064 case FORMAT_TYPE_SINT:
1065 bits = DESCRIPTOR_REQ_COMPONENT_TYPE_SINT;
1066 break;
1067 default:
1068 break;
1069 }
1070
1071 switch (dim) {
1072 case spv::Dim1D:
1073 bits |= arrayed ? DESCRIPTOR_REQ_VIEW_TYPE_1D_ARRAY : DESCRIPTOR_REQ_VIEW_TYPE_1D;
1074 return bits;
1075 case spv::Dim2D:
1076 bits |= msaa ? DESCRIPTOR_REQ_MULTI_SAMPLE : DESCRIPTOR_REQ_SINGLE_SAMPLE;
1077 bits |= arrayed ? DESCRIPTOR_REQ_VIEW_TYPE_2D_ARRAY : DESCRIPTOR_REQ_VIEW_TYPE_2D;
1078 return bits;
1079 case spv::Dim3D:
1080 bits |= DESCRIPTOR_REQ_VIEW_TYPE_3D;
1081 return bits;
1082 case spv::DimCube:
1083 bits |= arrayed ? DESCRIPTOR_REQ_VIEW_TYPE_CUBE_ARRAY : DESCRIPTOR_REQ_VIEW_TYPE_CUBE;
1084 return bits;
1085 case spv::DimSubpassData:
1086 bits |= msaa ? DESCRIPTOR_REQ_MULTI_SAMPLE : DESCRIPTOR_REQ_SINGLE_SAMPLE;
1087 return bits;
1088 default: // buffer, etc.
1089 return bits;
1090 }
1091 }
1092 default:
1093 return 0;
1094 }
1095 }
1096}
1097
1098// For some built-in analysis we need to know if the variable decorated with as the built-in was actually written to.
1099// This function examines instructions in the static call tree for a write to this variable.
1100bool SHADER_MODULE_STATE::IsBuiltInWritten(spirv_inst_iter builtin_instr, spirv_inst_iter entrypoint) const {
1101 auto type = builtin_instr.opcode();
1102 uint32_t target_id = builtin_instr.word(1);
1103 bool init_complete = false;
Nathaniel Cesario80c0e0f2021-10-14 13:25:54 -06001104 uint32_t target_member_offset = 0;
sfricke-samsung962cad92021-04-13 00:46:29 -07001105
1106 if (type == spv::OpMemberDecorate) {
1107 // Built-in is part of a structure -- examine instructions up to first function body to get initial IDs
1108 auto insn = entrypoint;
1109 while (!init_complete && (insn.opcode() != spv::OpFunction)) {
1110 switch (insn.opcode()) {
1111 case spv::OpTypePointer:
Nathaniel Cesario58fc2282021-08-18 12:20:40 -06001112 if (insn.word(2) == spv::StorageClassOutput) {
1113 const auto type_id = insn.word(3);
1114 if (type_id == target_id) {
1115 target_id = insn.word(1);
1116 } else {
1117 // If the output is an array, check if the element type is what we're looking for
1118 const auto type_insn = get_def(type_id);
1119 if ((type_insn.opcode() == spv::OpTypeArray) && (type_insn.word(2) == target_id)) {
1120 target_id = insn.word(1);
Nathaniel Cesario80c0e0f2021-10-14 13:25:54 -06001121 target_member_offset = 1;
Nathaniel Cesario58fc2282021-08-18 12:20:40 -06001122 }
1123 }
sfricke-samsung962cad92021-04-13 00:46:29 -07001124 }
1125 break;
1126 case spv::OpVariable:
1127 if (insn.word(1) == target_id) {
1128 target_id = insn.word(2);
1129 init_complete = true;
1130 }
1131 break;
1132 }
1133 insn++;
1134 }
1135 }
1136
1137 if (!init_complete && (type == spv::OpMemberDecorate)) return false;
1138
1139 bool found_write = false;
1140 layer_data::unordered_set<uint32_t> worklist;
1141 worklist.insert(entrypoint.word(2));
1142
1143 // Follow instructions in call graph looking for writes to target
1144 while (!worklist.empty() && !found_write) {
1145 auto id_iter = worklist.begin();
1146 auto id = *id_iter;
1147 worklist.erase(id_iter);
1148
1149 auto insn = get_def(id);
1150 if (insn == end()) {
1151 continue;
1152 }
1153
1154 if (insn.opcode() == spv::OpFunction) {
1155 // Scan body of function looking for other function calls or items in our ID chain
Nathaniel Cesario58fc2282021-08-18 12:20:40 -06001156 while (++insn, (insn.opcode() != spv::OpFunctionEnd) && !found_write) {
sfricke-samsung962cad92021-04-13 00:46:29 -07001157 switch (insn.opcode()) {
1158 case spv::OpAccessChain:
sjfrickeeaa434d2022-06-22 04:55:28 +09001159 case spv::OpInBoundsAccessChain:
sfricke-samsung962cad92021-04-13 00:46:29 -07001160 if (insn.word(3) == target_id) {
1161 if (type == spv::OpMemberDecorate) {
Nathaniel Cesario80c0e0f2021-10-14 13:25:54 -06001162 // Get the target member of the struct
1163 // NOTE: this will only work for structs and arrays of structs. Deeper levels of nesting (e.g.,
1164 // arrays of structs of structs) is not currently supported.
1165 const auto value_itr = GetConstantDef(insn.word(4 + target_member_offset));
1166 if (value_itr != end()) {
1167 auto value = GetConstantValue(value_itr);
1168 if (value == builtin_instr.word(2)) {
1169 target_id = insn.word(2);
1170 }
sfricke-samsung962cad92021-04-13 00:46:29 -07001171 }
1172 } else {
1173 target_id = insn.word(2);
1174 }
1175 }
1176 break;
1177 case spv::OpStore:
1178 if (insn.word(1) == target_id) {
1179 found_write = true;
1180 }
1181 break;
1182 case spv::OpFunctionCall:
1183 worklist.insert(insn.word(3));
1184 break;
1185 }
1186 }
1187 }
1188 }
1189 return found_write;
1190}
1191
1192// Used by the collection functions to help aid in state tracking
1193struct shader_module_used_operators {
1194 bool updated;
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001195 std::vector<uint32_t> image_read_members;
1196 std::vector<uint32_t> image_write_members;
1197 std::vector<uint32_t> atomic_members;
1198 std::vector<uint32_t> store_members;
1199 std::vector<uint32_t> atomic_store_members;
1200 std::vector<uint32_t> sampler_implicitLod_dref_proj_members; // sampler Load id
1201 std::vector<uint32_t> sampler_bias_offset_members; // sampler Load id
1202 std::vector<uint32_t> image_dref_members;
1203 std::vector<std::pair<uint32_t, uint32_t>> sampled_image_members; // <image,sampler> Load id
1204 layer_data::unordered_map<uint32_t, uint32_t> load_members;
1205 layer_data::unordered_map<uint32_t, std::pair<uint32_t, uint32_t>> accesschain_members;
1206 layer_data::unordered_map<uint32_t, uint32_t> image_texel_pointer_members;
sfricke-samsung962cad92021-04-13 00:46:29 -07001207
1208 shader_module_used_operators() : updated(false) {}
1209
1210 bool CheckImageOperandsBiasOffset(uint32_t type) {
1211 return type & (spv::ImageOperandsBiasMask | spv::ImageOperandsConstOffsetMask | spv::ImageOperandsOffsetMask |
1212 spv::ImageOperandsConstOffsetsMask)
1213 ? true
1214 : false;
1215 }
1216
sfricke-samsungef15e482022-01-26 11:32:49 -08001217 void update(SHADER_MODULE_STATE const *module_state) {
sfricke-samsung962cad92021-04-13 00:46:29 -07001218 if (updated) return;
1219 updated = true;
1220
sfricke-samsungef15e482022-01-26 11:32:49 -08001221 for (auto insn : *module_state) {
sfricke-samsung962cad92021-04-13 00:46:29 -07001222 switch (insn.opcode()) {
1223 case spv::OpImageSampleImplicitLod:
1224 case spv::OpImageSampleProjImplicitLod:
1225 case spv::OpImageSampleProjExplicitLod:
1226 case spv::OpImageSparseSampleImplicitLod:
1227 case spv::OpImageSparseSampleProjImplicitLod:
1228 case spv::OpImageSparseSampleProjExplicitLod: {
sfricke-samsungdb3f3f82022-01-18 06:41:15 -08001229 // combined image samples are just OpLoad, but also can be separate image and sampler
sfricke-samsungef15e482022-01-26 11:32:49 -08001230 auto id = module_state->get_def(insn.word(3)); // <id> Sampled Image
sfricke-samsungdb3f3f82022-01-18 06:41:15 -08001231 auto load_id = (id.opcode() == spv::OpSampledImage) ? id.word(4) : insn.word(3);
1232 sampler_implicitLod_dref_proj_members.emplace_back(load_id);
sfricke-samsung962cad92021-04-13 00:46:29 -07001233 // ImageOperands in index: 5
1234 if (insn.len() > 5 && CheckImageOperandsBiasOffset(insn.word(5))) {
sfricke-samsungdb3f3f82022-01-18 06:41:15 -08001235 sampler_bias_offset_members.emplace_back(load_id);
sfricke-samsung962cad92021-04-13 00:46:29 -07001236 }
1237 break;
1238 }
Lionel Landwerlincdbe8682021-12-08 15:10:37 +02001239 case spv::OpImageDrefGather:
1240 case spv::OpImageSparseDrefGather: {
sfricke-samsungdb3f3f82022-01-18 06:41:15 -08001241 // combined image samples are just OpLoad, but also can be separate image and sampler
sfricke-samsungef15e482022-01-26 11:32:49 -08001242 auto id = module_state->get_def(insn.word(3)); // <id> Sampled Image
sfricke-samsungdb3f3f82022-01-18 06:41:15 -08001243 auto load_id = (id.opcode() == spv::OpSampledImage) ? id.word(3) : insn.word(3);
1244 image_dref_members.emplace_back(load_id);
Lionel Landwerlincdbe8682021-12-08 15:10:37 +02001245 break;
1246 }
sfricke-samsung962cad92021-04-13 00:46:29 -07001247 case spv::OpImageSampleDrefImplicitLod:
1248 case spv::OpImageSampleDrefExplicitLod:
1249 case spv::OpImageSampleProjDrefImplicitLod:
1250 case spv::OpImageSampleProjDrefExplicitLod:
1251 case spv::OpImageSparseSampleDrefImplicitLod:
1252 case spv::OpImageSparseSampleDrefExplicitLod:
1253 case spv::OpImageSparseSampleProjDrefImplicitLod:
1254 case spv::OpImageSparseSampleProjDrefExplicitLod: {
sfricke-samsungdb3f3f82022-01-18 06:41:15 -08001255 // combined image samples are just OpLoad, but also can be separate image and sampler
sfricke-samsungef15e482022-01-26 11:32:49 -08001256 auto id = module_state->get_def(insn.word(3)); // <id> Sampled Image
sfricke-samsungdb3f3f82022-01-18 06:41:15 -08001257 auto sampler_load_id = (id.opcode() == spv::OpSampledImage) ? id.word(4) : insn.word(3);
1258 auto image_load_id = (id.opcode() == spv::OpSampledImage) ? id.word(3) : insn.word(3);
1259
1260 image_dref_members.emplace_back(image_load_id);
1261 sampler_implicitLod_dref_proj_members.emplace_back(sampler_load_id);
sfricke-samsung962cad92021-04-13 00:46:29 -07001262 // ImageOperands in index: 6
1263 if (insn.len() > 6 && CheckImageOperandsBiasOffset(insn.word(6))) {
sfricke-samsungdb3f3f82022-01-18 06:41:15 -08001264 sampler_bias_offset_members.emplace_back(sampler_load_id);
sfricke-samsung962cad92021-04-13 00:46:29 -07001265 }
1266 break;
1267 }
1268 case spv::OpImageSampleExplicitLod:
1269 case spv::OpImageSparseSampleExplicitLod: {
1270 // ImageOperands in index: 5
1271 if (insn.len() > 5 && CheckImageOperandsBiasOffset(insn.word(5))) {
sfricke-samsungdb3f3f82022-01-18 06:41:15 -08001272 // combined image samples are just OpLoad, but also can be separate image and sampler
sfricke-samsungef15e482022-01-26 11:32:49 -08001273 auto id = module_state->get_def(insn.word(3)); // <id> Sampled Image
sfricke-samsungdb3f3f82022-01-18 06:41:15 -08001274 auto load_id = (id.opcode() == spv::OpSampledImage) ? id.word(4) : insn.word(3);
1275 sampler_bias_offset_members.emplace_back(load_id);
sfricke-samsung962cad92021-04-13 00:46:29 -07001276 }
1277 break;
1278 }
1279 case spv::OpStore: {
1280 store_members.emplace_back(insn.word(1)); // object id or AccessChain id
1281 break;
1282 }
Lionel Landwerlinbc7401b2021-12-07 15:43:05 +02001283 case spv::OpImageRead:
1284 case spv::OpImageSparseRead: {
sfricke-samsungad55ccc2022-01-19 20:06:17 -08001285 image_read_members.emplace_back(insn.word(3)); // Load id
Lionel Landwerlinbc7401b2021-12-07 15:43:05 +02001286 break;
1287 }
sfricke-samsung962cad92021-04-13 00:46:29 -07001288 case spv::OpImageWrite: {
sfricke-samsungad55ccc2022-01-19 20:06:17 -08001289 image_write_members.emplace_back(insn.word(1)); // Load id
sfricke-samsung962cad92021-04-13 00:46:29 -07001290 break;
1291 }
1292 case spv::OpSampledImage: {
1293 // 3: image load id, 4: sampler load id
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001294 sampled_image_members.emplace_back(std::pair<uint32_t, uint32_t>(insn.word(3), insn.word(4)));
sfricke-samsung962cad92021-04-13 00:46:29 -07001295 break;
1296 }
1297 case spv::OpLoad: {
1298 // 2: Load id, 3: object id or AccessChain id
1299 load_members.emplace(insn.word(2), insn.word(3));
1300 break;
1301 }
sjfrickeeaa434d2022-06-22 04:55:28 +09001302 case spv::OpAccessChain:
1303 case spv::OpInBoundsAccessChain: {
sfricke-samsung962cad92021-04-13 00:46:29 -07001304 if (insn.len() == 4) {
1305 // If it is for struct, the length is only 4.
1306 // 2: AccessChain id, 3: object id
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001307 accesschain_members.emplace(insn.word(2), std::pair<uint32_t, uint32_t>(insn.word(3), 0));
sfricke-samsung962cad92021-04-13 00:46:29 -07001308 } else {
1309 // 2: AccessChain id, 3: object id, 4: object id of array index
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001310 accesschain_members.emplace(insn.word(2), std::pair<uint32_t, uint32_t>(insn.word(3), insn.word(4)));
sfricke-samsung962cad92021-04-13 00:46:29 -07001311 }
1312 break;
1313 }
1314 case spv::OpImageTexelPointer: {
1315 // 2: ImageTexelPointer id, 3: object id
1316 image_texel_pointer_members.emplace(insn.word(2), insn.word(3));
1317 break;
1318 }
1319 default: {
1320 if (AtomicOperation(insn.opcode())) {
1321 if (insn.opcode() == spv::OpAtomicStore) {
1322 atomic_store_members.emplace_back(insn.word(1)); // ImageTexelPointer id
1323 } else {
1324 atomic_members.emplace_back(insn.word(3)); // ImageTexelPointer id
1325 }
1326 }
1327 break;
1328 }
1329 }
1330 }
1331 }
1332};
1333
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001334static bool CheckObjectIDFromOpLoad(uint32_t object_id, const std::vector<uint32_t> &operator_members,
1335 const layer_data::unordered_map<uint32_t, uint32_t> &load_members,
1336 const layer_data::unordered_map<uint32_t, std::pair<uint32_t, uint32_t>> &accesschain_members) {
sfricke-samsung962cad92021-04-13 00:46:29 -07001337 for (auto load_id : operator_members) {
1338 if (object_id == load_id) return true;
1339 auto load_it = load_members.find(load_id);
1340 if (load_it == load_members.end()) {
1341 continue;
1342 }
1343 if (load_it->second == object_id) {
1344 return true;
1345 }
1346
1347 auto accesschain_it = accesschain_members.find(load_it->second);
1348 if (accesschain_it == accesschain_members.end()) {
1349 continue;
1350 }
1351 if (accesschain_it->second.first == object_id) {
1352 return true;
1353 }
1354 }
1355 return false;
1356}
1357
1358// Takes a OpVariable and looks at the the descriptor type it uses. This will find things such as if the variable is writable, image
1359// atomic operation, matching images to samplers, etc
1360void SHADER_MODULE_STATE::IsSpecificDescriptorType(const spirv_inst_iter &id_it, bool is_storage_buffer, bool is_check_writable,
sjfrickeeaa434d2022-06-22 04:55:28 +09001361 interface_var &out_interface_var) const {
1362 shader_module_used_operators used_operators;
sfricke-samsung962cad92021-04-13 00:46:29 -07001363 uint32_t type_id = id_it.word(1);
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001364 uint32_t id = id_it.word(2);
sfricke-samsung962cad92021-04-13 00:46:29 -07001365
1366 auto type = get_def(type_id);
1367
1368 // Strip off any array or ptrs. Where we remove array levels, adjust the descriptor count for each dimension.
1369 while (type.opcode() == spv::OpTypeArray || type.opcode() == spv::OpTypePointer || type.opcode() == spv::OpTypeRuntimeArray ||
1370 type.opcode() == spv::OpTypeSampledImage) {
1371 if (type.opcode() == spv::OpTypeArray || type.opcode() == spv::OpTypeRuntimeArray ||
1372 type.opcode() == spv::OpTypeSampledImage) {
1373 type = get_def(type.word(2)); // Element type
1374 } else {
1375 type = get_def(type.word(3)); // Pointer type
1376 }
1377 }
Lionel Landwerlinbc7401b2021-12-07 15:43:05 +02001378
sfricke-samsung962cad92021-04-13 00:46:29 -07001379 switch (type.opcode()) {
1380 case spv::OpTypeImage: {
1381 auto dim = type.word(3);
1382 if (dim != spv::DimSubpassData) {
1383 used_operators.update(this);
1384
Lionel Landwerlinbc7401b2021-12-07 15:43:05 +02001385 // Sampled == 2 indicates used without a sampler (a storage image)
1386 bool is_image_without_format = false;
1387 if (type.word(7) == 2) is_image_without_format = type.word(8) == spv::ImageFormatUnknown;
1388
sfricke-samsungad55ccc2022-01-19 20:06:17 -08001389 if (CheckObjectIDFromOpLoad(id, used_operators.image_write_members, used_operators.load_members,
sfricke-samsung962cad92021-04-13 00:46:29 -07001390 used_operators.accesschain_members)) {
1391 out_interface_var.is_writable = true;
Lionel Landwerlinbc7401b2021-12-07 15:43:05 +02001392 if (is_image_without_format) out_interface_var.is_write_without_format = true;
1393 }
sfricke-samsungad55ccc2022-01-19 20:06:17 -08001394 if (CheckObjectIDFromOpLoad(id, used_operators.image_read_members, used_operators.load_members,
Lionel Landwerlinbc7401b2021-12-07 15:43:05 +02001395 used_operators.accesschain_members)) {
1396 out_interface_var.is_readable = true;
1397 if (is_image_without_format) out_interface_var.is_read_without_format = true;
sfricke-samsung962cad92021-04-13 00:46:29 -07001398 }
1399 if (CheckObjectIDFromOpLoad(id, used_operators.sampler_implicitLod_dref_proj_members, used_operators.load_members,
1400 used_operators.accesschain_members)) {
1401 out_interface_var.is_sampler_implicitLod_dref_proj = true;
1402 }
1403 if (CheckObjectIDFromOpLoad(id, used_operators.sampler_bias_offset_members, used_operators.load_members,
1404 used_operators.accesschain_members)) {
1405 out_interface_var.is_sampler_bias_offset = true;
1406 }
1407 if (CheckObjectIDFromOpLoad(id, used_operators.atomic_members, used_operators.image_texel_pointer_members,
1408 used_operators.accesschain_members) ||
1409 CheckObjectIDFromOpLoad(id, used_operators.atomic_store_members, used_operators.image_texel_pointer_members,
1410 used_operators.accesschain_members)) {
1411 out_interface_var.is_atomic_operation = true;
1412 }
Lionel Landwerlincdbe8682021-12-08 15:10:37 +02001413 if (CheckObjectIDFromOpLoad(id, used_operators.image_dref_members, used_operators.load_members,
1414 used_operators.accesschain_members)) {
1415 out_interface_var.is_dref_operation = true;
1416 }
sfricke-samsung962cad92021-04-13 00:46:29 -07001417
sfricke-samsungad55ccc2022-01-19 20:06:17 -08001418 for (auto &itp_id : used_operators.sampled_image_members) {
sfricke-samsung962cad92021-04-13 00:46:29 -07001419 // Find if image id match.
1420 uint32_t image_index = 0;
1421 auto load_it = used_operators.load_members.find(itp_id.first);
1422 if (load_it == used_operators.load_members.end()) {
1423 continue;
1424 } else {
1425 if (load_it->second != id) {
1426 auto accesschain_it = used_operators.accesschain_members.find(load_it->second);
1427 if (accesschain_it == used_operators.accesschain_members.end()) {
1428 continue;
1429 } else {
1430 if (accesschain_it->second.first != id) {
1431 continue;
1432 }
1433
1434 const auto const_itr = GetConstantDef(accesschain_it->second.second);
1435 if (const_itr == end()) {
1436 // access chain index not a constant, skip.
1437 break;
1438 }
1439 image_index = GetConstantValue(const_itr);
1440 }
1441 }
1442 }
1443 // Find sampler's set binding.
1444 load_it = used_operators.load_members.find(itp_id.second);
1445 if (load_it == used_operators.load_members.end()) {
1446 continue;
1447 } else {
1448 uint32_t sampler_id = load_it->second;
1449 uint32_t sampler_index = 0;
1450 auto accesschain_it = used_operators.accesschain_members.find(load_it->second);
1451
1452 if (accesschain_it != used_operators.accesschain_members.end()) {
1453 const auto const_itr = GetConstantDef(accesschain_it->second.second);
1454 if (const_itr == end()) {
1455 // access chain index representing sampler index is not a constant, skip.
1456 break;
1457 }
1458 sampler_id = const_itr.offset();
1459 sampler_index = GetConstantValue(const_itr);
1460 }
1461 auto sampler_dec = get_decorations(sampler_id);
1462 if (image_index >= out_interface_var.samplers_used_by_image.size()) {
1463 out_interface_var.samplers_used_by_image.resize(image_index + 1);
1464 }
sfricke-samsungdb3f3f82022-01-18 06:41:15 -08001465
1466 // Need to check again for these properties in case not using a combined image sampler
1467 if (CheckObjectIDFromOpLoad(sampler_id, used_operators.sampler_implicitLod_dref_proj_members,
1468 used_operators.load_members, used_operators.accesschain_members)) {
1469 out_interface_var.is_sampler_implicitLod_dref_proj = true;
1470 }
1471 if (CheckObjectIDFromOpLoad(sampler_id, used_operators.sampler_bias_offset_members,
1472 used_operators.load_members, used_operators.accesschain_members)) {
1473 out_interface_var.is_sampler_bias_offset = true;
1474 }
1475
sfricke-samsung962cad92021-04-13 00:46:29 -07001476 out_interface_var.samplers_used_by_image[image_index].emplace(
Jeremy Gebben7fc88a22021-08-25 13:30:45 -06001477 SamplerUsedByImage{DescriptorSlot{sampler_dec.descriptor_set, sampler_dec.binding}, sampler_index});
sfricke-samsung962cad92021-04-13 00:46:29 -07001478 }
1479 }
1480 }
1481 return;
1482 }
1483
1484 case spv::OpTypeStruct: {
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001485 layer_data::unordered_set<uint32_t> nonwritable_members;
sfricke-samsung962cad92021-04-13 00:46:29 -07001486 if (get_decorations(type.word(1)).flags & decoration_set::buffer_block_bit) is_storage_buffer = true;
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -06001487 for (auto insn : static_data_.member_decoration_inst) {
sfricke-samsung962cad92021-04-13 00:46:29 -07001488 if (insn.word(1) == type.word(1) && insn.word(3) == spv::DecorationNonWritable) {
1489 nonwritable_members.insert(insn.word(2));
1490 }
1491 }
1492
1493 // A buffer is writable if it's either flavor of storage buffer, and has any member not decorated
1494 // as nonwritable.
1495 if (is_storage_buffer && nonwritable_members.size() != type.len() - 2) {
1496 used_operators.update(this);
1497
1498 for (auto oid : used_operators.store_members) {
1499 if (id == oid) {
1500 out_interface_var.is_writable = true;
1501 return;
1502 }
1503 auto accesschain_it = used_operators.accesschain_members.find(oid);
1504 if (accesschain_it == used_operators.accesschain_members.end()) {
1505 continue;
1506 }
1507 if (accesschain_it->second.first == id) {
1508 out_interface_var.is_writable = true;
1509 return;
1510 }
1511 }
1512 if (CheckObjectIDFromOpLoad(id, used_operators.atomic_store_members, used_operators.image_texel_pointer_members,
1513 used_operators.accesschain_members)) {
1514 out_interface_var.is_writable = true;
1515 return;
1516 }
1517 }
1518 }
1519 }
1520}
1521
Jeremy Gebben7fc88a22021-08-25 13:30:45 -06001522std::vector<std::pair<DescriptorSlot, interface_var>> SHADER_MODULE_STATE::CollectInterfaceByDescriptorSlot(
ziga-lunargc2de4782022-04-14 19:49:07 +02001523 layer_data::unordered_set<uint32_t> const &accessible_ids) const {
Jeremy Gebben7fc88a22021-08-25 13:30:45 -06001524 std::vector<std::pair<DescriptorSlot, interface_var>> out;
sfricke-samsung962cad92021-04-13 00:46:29 -07001525
ziga-lunargc2de4782022-04-14 19:49:07 +02001526 for (auto id : accessible_ids) {
sfricke-samsung962cad92021-04-13 00:46:29 -07001527 auto insn = get_def(id);
1528 assert(insn != end());
1529
1530 if (insn.opcode() == spv::OpVariable &&
Lionel Landwerlin6a9f89c2021-12-07 15:46:46 +02001531 (insn.word(3) == spv::StorageClassUniform ||
1532 insn.word(3) == spv::StorageClassUniformConstant ||
sfricke-samsung962cad92021-04-13 00:46:29 -07001533 insn.word(3) == spv::StorageClassStorageBuffer)) {
1534 auto d = get_decorations(insn.word(2));
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001535 uint32_t set = d.descriptor_set;
1536 uint32_t binding = d.binding;
sfricke-samsung962cad92021-04-13 00:46:29 -07001537
1538 interface_var v = {};
1539 v.id = insn.word(2);
1540 v.type_id = insn.word(1);
1541
1542 IsSpecificDescriptorType(insn, insn.word(3) == spv::StorageClassStorageBuffer,
sjfrickeeaa434d2022-06-22 04:55:28 +09001543 !(d.flags & decoration_set::nonwritable_bit), v);
Jeremy Gebben84b838b2021-08-23 08:41:39 -06001544 out.emplace_back(DescriptorSlot{set, binding}, v);
sfricke-samsung962cad92021-04-13 00:46:29 -07001545 }
1546 }
1547
1548 return out;
1549}
1550
1551layer_data::unordered_set<uint32_t> SHADER_MODULE_STATE::CollectWritableOutputLocationinFS(
Jeremy Gebben84b838b2021-08-23 08:41:39 -06001552 const spirv_inst_iter &entrypoint) const {
sfricke-samsung962cad92021-04-13 00:46:29 -07001553 layer_data::unordered_set<uint32_t> location_list;
sfricke-samsung962cad92021-04-13 00:46:29 -07001554 const auto outputs = CollectInterfaceByLocation(entrypoint, spv::StorageClassOutput, false);
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001555 layer_data::unordered_set<uint32_t> store_members;
1556 layer_data::unordered_map<uint32_t, uint32_t> accesschain_members;
sfricke-samsung962cad92021-04-13 00:46:29 -07001557
1558 for (auto insn : *this) {
1559 switch (insn.opcode()) {
1560 case spv::OpStore:
1561 case spv::OpAtomicStore: {
1562 store_members.insert(insn.word(1)); // object id or AccessChain id
1563 break;
1564 }
sjfrickeeaa434d2022-06-22 04:55:28 +09001565 case spv::OpAccessChain:
1566 case spv::OpInBoundsAccessChain: {
sfricke-samsung962cad92021-04-13 00:46:29 -07001567 // 2: AccessChain id, 3: object id
1568 if (insn.word(3)) accesschain_members.emplace(insn.word(2), insn.word(3));
1569 break;
1570 }
1571 default:
1572 break;
1573 }
1574 }
1575 if (store_members.empty()) {
1576 return location_list;
1577 }
1578 for (auto output : outputs) {
1579 auto store_it = store_members.find(output.second.id);
1580 if (store_it != store_members.end()) {
1581 location_list.insert(output.first.first);
1582 store_members.erase(store_it);
1583 continue;
1584 }
1585 store_it = store_members.begin();
1586 while (store_it != store_members.end()) {
1587 auto accesschain_it = accesschain_members.find(*store_it);
1588 if (accesschain_it == accesschain_members.end()) {
1589 ++store_it;
1590 continue;
1591 }
1592 if (accesschain_it->second == output.second.id) {
1593 location_list.insert(output.first.first);
1594 store_members.erase(store_it);
1595 accesschain_members.erase(accesschain_it);
1596 break;
1597 }
1598 ++store_it;
1599 }
1600 }
1601 return location_list;
1602}
1603
1604bool SHADER_MODULE_STATE::CollectInterfaceBlockMembers(std::map<location_t, interface_var> *out, bool is_array_of_verts,
ziga-lunarg9e94e112021-09-27 00:21:10 +02001605 uint32_t id, uint32_t type_id, bool is_patch,
1606 uint32_t /*first_location*/) const {
sfricke-samsung962cad92021-04-13 00:46:29 -07001607 // Walk down the type_id presented, trying to determine whether it's actually an interface block.
1608 auto type = GetStructType(get_def(type_id), is_array_of_verts && !is_patch);
1609 if (type == end() || !(get_decorations(type.word(1)).flags & decoration_set::block_bit)) {
1610 // This isn't an interface block.
1611 return false;
1612 }
1613
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001614 layer_data::unordered_map<uint32_t, uint32_t> member_components;
1615 layer_data::unordered_map<uint32_t, uint32_t> member_relaxed_precision;
1616 layer_data::unordered_map<uint32_t, uint32_t> member_patch;
sfricke-samsung962cad92021-04-13 00:46:29 -07001617
1618 // Walk all the OpMemberDecorate for type's result id -- first pass, collect components.
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -06001619 for (auto insn : static_data_.member_decoration_inst) {
sfricke-samsung962cad92021-04-13 00:46:29 -07001620 if (insn.word(1) == type.word(1)) {
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001621 uint32_t member_index = insn.word(2);
sfricke-samsung962cad92021-04-13 00:46:29 -07001622
1623 if (insn.word(3) == spv::DecorationComponent) {
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001624 uint32_t component = insn.word(4);
sfricke-samsung962cad92021-04-13 00:46:29 -07001625 member_components[member_index] = component;
1626 }
1627
1628 if (insn.word(3) == spv::DecorationRelaxedPrecision) {
1629 member_relaxed_precision[member_index] = 1;
1630 }
1631
1632 if (insn.word(3) == spv::DecorationPatch) {
1633 member_patch[member_index] = 1;
1634 }
1635 }
1636 }
1637
1638 // TODO: correctly handle location assignment from outside
1639
1640 // Second pass -- produce the output, from Location decorations
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -06001641 for (auto insn : static_data_.member_decoration_inst) {
sfricke-samsung962cad92021-04-13 00:46:29 -07001642 if (insn.word(1) == type.word(1)) {
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001643 uint32_t member_index = insn.word(2);
1644 uint32_t member_type_id = type.word(2 + member_index);
sfricke-samsung962cad92021-04-13 00:46:29 -07001645
1646 if (insn.word(3) == spv::DecorationLocation) {
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001647 uint32_t location = insn.word(4);
1648 uint32_t num_locations = GetLocationsConsumedByType(member_type_id, false);
sfricke-samsung962cad92021-04-13 00:46:29 -07001649 auto component_it = member_components.find(member_index);
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001650 uint32_t component = component_it == member_components.end() ? 0 : component_it->second;
sfricke-samsung962cad92021-04-13 00:46:29 -07001651 bool is_relaxed_precision = member_relaxed_precision.find(member_index) != member_relaxed_precision.end();
1652 bool member_is_patch = is_patch || member_patch.count(member_index) > 0;
1653
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001654 for (uint32_t offset = 0; offset < num_locations; offset++) {
sfricke-samsung962cad92021-04-13 00:46:29 -07001655 interface_var v = {};
1656 v.id = id;
1657 // TODO: member index in interface_var too?
1658 v.type_id = member_type_id;
1659 v.offset = offset;
1660 v.is_patch = member_is_patch;
1661 v.is_block_member = true;
1662 v.is_relaxed_precision = is_relaxed_precision;
1663 (*out)[std::make_pair(location + offset, component)] = v;
1664 }
1665 }
1666 }
1667 }
1668
1669 return true;
1670}
1671
1672std::map<location_t, interface_var> SHADER_MODULE_STATE::CollectInterfaceByLocation(spirv_inst_iter entrypoint,
1673 spv::StorageClass sinterface,
1674 bool is_array_of_verts) const {
1675 // TODO: handle index=1 dual source outputs from FS -- two vars will have the same location, and we DON'T want to clobber.
1676
1677 std::map<location_t, interface_var> out;
1678
1679 for (uint32_t iid : FindEntrypointInterfaces(entrypoint)) {
1680 auto insn = get_def(iid);
1681 assert(insn != end());
1682 assert(insn.opcode() == spv::OpVariable);
1683
ziga-lunarg9e94e112021-09-27 00:21:10 +02001684 const auto d = get_decorations(iid);
1685 bool passthrough = sinterface == spv::StorageClassOutput && insn.word(3) == spv::StorageClassInput &&
1686 (d.flags & decoration_set::passthrough_bit) != 0;
1687 if (insn.word(3) == static_cast<uint32_t>(sinterface) || passthrough) {
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001688 uint32_t id = insn.word(2);
1689 uint32_t type = insn.word(1);
sfricke-samsung962cad92021-04-13 00:46:29 -07001690
ziga-lunarg9e94e112021-09-27 00:21:10 +02001691 auto location = d.location;
sfricke-samsung962cad92021-04-13 00:46:29 -07001692 int builtin = d.builtin;
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001693 uint32_t component = d.component;
sfricke-samsung962cad92021-04-13 00:46:29 -07001694 bool is_patch = (d.flags & decoration_set::patch_bit) != 0;
1695 bool is_relaxed_precision = (d.flags & decoration_set::relaxed_precision_bit) != 0;
ziga-lunarg9e94e112021-09-27 00:21:10 +02001696 bool is_per_vertex = (d.flags & decoration_set::per_vertex_bit) != 0;
sfricke-samsung962cad92021-04-13 00:46:29 -07001697
1698 if (builtin != -1) {
1699 continue;
ziga-lunarg9e94e112021-09-27 00:21:10 +02001700 } else if (!CollectInterfaceBlockMembers(&out, is_array_of_verts, id, type, is_patch, location) ||
1701 location != decoration_set::kInvalidValue) {
sfricke-samsung962cad92021-04-13 00:46:29 -07001702 // A user-defined interface variable, with a location. Where a variable occupied multiple locations, emit
1703 // one result for each.
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001704 uint32_t num_locations = GetLocationsConsumedByType(type, (is_array_of_verts && !is_patch) || is_per_vertex);
1705 for (uint32_t offset = 0; offset < num_locations; offset++) {
sfricke-samsung962cad92021-04-13 00:46:29 -07001706 interface_var v = {};
1707 v.id = id;
1708 v.type_id = type;
1709 v.offset = offset;
1710 v.is_patch = is_patch;
1711 v.is_relaxed_precision = is_relaxed_precision;
1712 out[std::make_pair(location + offset, component)] = v;
1713 }
1714 }
1715 }
1716 }
1717
1718 return out;
1719}
1720
1721std::vector<uint32_t> SHADER_MODULE_STATE::CollectBuiltinBlockMembers(spirv_inst_iter entrypoint, uint32_t storageClass) const {
sfricke-samsung962cad92021-04-13 00:46:29 -07001722 // Find all interface variables belonging to the entrypoint and matching the storage class
sfricke-samsung0df5ee72021-07-24 23:27:16 -07001723 std::vector<uint32_t> variables;
sfricke-samsung962cad92021-04-13 00:46:29 -07001724 for (uint32_t id : FindEntrypointInterfaces(entrypoint)) {
1725 auto def = get_def(id);
1726 assert(def != end());
1727 assert(def.opcode() == spv::OpVariable);
1728
1729 if (def.word(3) == storageClass) variables.push_back(def.word(1));
1730 }
1731
1732 // Find all members belonging to the builtin block selected
1733 std::vector<uint32_t> builtin_block_members;
1734 for (auto &var : variables) {
1735 auto def = get_def(get_def(var).word(3));
1736
1737 // It could be an array of IO blocks. The element type should be the struct defining the block contents
1738 if (def.opcode() == spv::OpTypeArray) def = get_def(def.word(2));
1739
1740 // Now find all members belonging to the struct defining the IO block
1741 if (def.opcode() == spv::OpTypeStruct) {
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -06001742 for (auto set : static_data_.builtin_decoration_list) {
sfricke-samsung0df5ee72021-07-24 23:27:16 -07001743 auto insn = at(set.offset);
1744 if ((insn.opcode() == spv::OpMemberDecorate) && (def.word(1) == insn.word(1))) {
1745 // Start with undefined builtin for each struct member.
1746 // But only when confirmed the struct is the built-in inteface block (can only be one per shader)
1747 if (builtin_block_members.size() == 0) {
1748 builtin_block_members.resize(def.len() - 2, spv::BuiltInMax);
sfricke-samsung962cad92021-04-13 00:46:29 -07001749 }
sfricke-samsung0df5ee72021-07-24 23:27:16 -07001750 auto struct_index = insn.word(2);
1751 assert(struct_index < builtin_block_members.size());
1752 builtin_block_members[struct_index] = insn.word(4);
sfricke-samsung962cad92021-04-13 00:46:29 -07001753 }
1754 }
1755 }
1756 }
1757
1758 return builtin_block_members;
1759}
1760
1761std::vector<std::pair<uint32_t, interface_var>> SHADER_MODULE_STATE::CollectInterfaceByInputAttachmentIndex(
1762 layer_data::unordered_set<uint32_t> const &accessible_ids) const {
1763 std::vector<std::pair<uint32_t, interface_var>> out;
1764
Nathaniel Cesario77cd59b2021-10-11 23:52:24 -06001765 for (auto insn : static_data_.decoration_inst) {
sfricke-samsung962cad92021-04-13 00:46:29 -07001766 if (insn.word(2) == spv::DecorationInputAttachmentIndex) {
1767 auto attachment_index = insn.word(3);
1768 auto id = insn.word(1);
1769
1770 if (accessible_ids.count(id)) {
1771 auto def = get_def(id);
1772 assert(def != end());
1773 if (def.opcode() == spv::OpVariable && def.word(3) == spv::StorageClassUniformConstant) {
1774 auto num_locations = GetLocationsConsumedByType(def.word(1), false);
sfricke-samsung7fac88a2022-01-26 11:44:22 -08001775 for (uint32_t offset = 0; offset < num_locations; offset++) {
sfricke-samsung962cad92021-04-13 00:46:29 -07001776 interface_var v = {};
1777 v.id = id;
1778 v.type_id = def.word(1);
1779 v.offset = offset;
1780 out.emplace_back(attachment_index + offset, v);
1781 }
1782 }
1783 }
1784 }
1785 }
1786
1787 return out;
1788}
1789
ziga-lunarg8346fe82021-08-22 17:30:50 +02001790uint32_t SHADER_MODULE_STATE::GetNumComponentsInBaseType(const spirv_inst_iter &iter) const {
1791 const uint32_t opcode = iter.opcode();
1792 if (opcode == spv::OpTypeFloat || opcode == spv::OpTypeInt) {
1793 return 1;
1794 } else if (opcode == spv::OpTypeVector) {
1795 const uint32_t component_count = iter.word(3);
1796 return component_count;
1797 } else if (opcode == spv::OpTypeMatrix) {
1798 const auto column_type = get_def(iter.word(2));
1799 const uint32_t vector_length = GetNumComponentsInBaseType(column_type);
ziga-lunarg38e44982022-04-05 00:10:46 +02001800 // Because we are calculating components for a single location we do not care about column count
1801 return vector_length;
ziga-lunarg8346fe82021-08-22 17:30:50 +02001802 } else if (opcode == spv::OpTypeArray) {
1803 const auto element_type = get_def(iter.word(2));
1804 const uint32_t element_length = GetNumComponentsInBaseType(element_type);
1805 return element_length;
1806 } else if (opcode == spv::OpTypeStruct) {
1807 uint32_t total_size = 0;
1808 for (uint32_t i = 2; i < iter.len(); ++i) {
1809 total_size += GetNumComponentsInBaseType(get_def(iter.word(i)));
1810 }
1811 return total_size;
1812 } else if (opcode == spv::OpTypePointer) {
1813 const auto type = get_def(iter.word(3));
1814 return GetNumComponentsInBaseType(type);
1815 }
1816 return 0;
1817}
1818
ziga-lunarga26b3602021-08-08 15:53:00 +02001819uint32_t SHADER_MODULE_STATE::GetTypeBitsSize(const spirv_inst_iter &iter) const {
1820 const uint32_t opcode = iter.opcode();
1821 if (opcode == spv::OpTypeFloat || opcode == spv::OpTypeInt) {
1822 return iter.word(2);
1823 } else if (opcode == spv::OpTypeVector) {
1824 const auto component_type = get_def(iter.word(2));
1825 uint32_t scalar_width = GetTypeBitsSize(component_type);
1826 uint32_t component_count = iter.word(3);
1827 return scalar_width * component_count;
1828 } else if (opcode == spv::OpTypeMatrix) {
1829 const auto column_type = get_def(iter.word(2));
1830 uint32_t vector_width = GetTypeBitsSize(column_type);
1831 uint32_t column_count = iter.word(3);
1832 return vector_width * column_count;
1833 } else if (opcode == spv::OpTypeArray) {
1834 const auto element_type = get_def(iter.word(2));
1835 uint32_t element_width = GetTypeBitsSize(element_type);
1836 const auto length_type = get_def(iter.word(3));
1837 uint32_t length = GetConstantValue(length_type);
1838 return element_width * length;
1839 } else if (opcode == spv::OpTypeStruct) {
1840 uint32_t total_size = 0;
1841 for (uint32_t i = 2; i < iter.len(); ++i) {
1842 total_size += GetTypeBitsSize(get_def(iter.word(i)));
1843 }
1844 return total_size;
ziga-lunarg8346fe82021-08-22 17:30:50 +02001845 } else if (opcode == spv::OpTypePointer) {
1846 const auto type = get_def(iter.word(3));
1847 return GetTypeBitsSize(type);
ziga-lunargef2c3172021-11-07 10:35:29 +01001848 } else if (opcode == spv::OpVariable) {
1849 const auto type = get_def(iter.word(1));
1850 return GetTypeBitsSize(type);
ziga-lunarga26b3602021-08-08 15:53:00 +02001851 }
1852 return 0;
1853}
1854
1855uint32_t SHADER_MODULE_STATE::GetTypeBytesSize(const spirv_inst_iter &iter) const { return GetTypeBitsSize(iter) / 8; }
1856
ziga-lunarg19fc6ae2021-09-09 00:05:19 +02001857// Returns the base type (float, int or unsigned int) or struct (can have multiple different base types inside)
ziga-lunarg8346fe82021-08-22 17:30:50 +02001858uint32_t SHADER_MODULE_STATE::GetBaseType(const spirv_inst_iter &iter) const {
1859 const uint32_t opcode = iter.opcode();
sjfricke10f74a82022-08-18 18:12:56 +09001860 if (opcode == spv::OpTypeFloat || opcode == spv::OpTypeInt || opcode == spv::OpTypeBool || opcode == spv::OpTypeStruct) {
1861 // point to itself as its the base type (or a struct that needs to be traversed still)
ziga-lunarg8346fe82021-08-22 17:30:50 +02001862 return iter.word(1);
1863 } else if (opcode == spv::OpTypeVector) {
1864 const auto& component_type = get_def(iter.word(2));
1865 return GetBaseType(component_type);
1866 } else if (opcode == spv::OpTypeMatrix) {
1867 const auto& column_type = get_def(iter.word(2));
1868 return GetBaseType(column_type);
sjfricke6a03e012022-06-23 17:54:11 +09001869 } else if (opcode == spv::OpTypeArray || opcode == spv::OpTypeRuntimeArray) {
ziga-lunarg8346fe82021-08-22 17:30:50 +02001870 const auto& element_type = get_def(iter.word(2));
1871 return GetBaseType(element_type);
1872 } else if (opcode == spv::OpTypePointer) {
1873 const auto& type = get_def(iter.word(3));
1874 return GetBaseType(type);
1875 }
sjfricke10f74a82022-08-18 18:12:56 +09001876 // If we assert here, we are missing a valid base type that must be handled. Without this assert, a return value of 0 will
1877 // produce a hard bug to track
1878 assert(false);
ziga-lunarg8346fe82021-08-22 17:30:50 +02001879 return 0;
1880}
1881
sfricke-samsunga6c1ddc2022-01-23 14:15:40 -08001882// Returns type_id if id has type or zero otherwise
1883uint32_t SHADER_MODULE_STATE::GetTypeId(uint32_t id) const {
1884 const auto type = get_def(id);
1885 return OpcodeHasType(type.opcode()) ? type.word(1) : 0;
1886}
1887
sfricke-samsung962cad92021-04-13 00:46:29 -07001888std::vector<uint32_t> FindEntrypointInterfaces(const spirv_inst_iter &entrypoint) {
1889 assert(entrypoint.opcode() == spv::OpEntryPoint);
1890
1891 std::vector<uint32_t> interfaces;
1892 // Find the end of the entrypoint's name string. additional zero bytes follow the actual null terminator, to fill out the
1893 // rest of the word - so we only need to look at the last byte in the word to determine which word contains the terminator.
1894 uint32_t word = 3;
1895 while (entrypoint.word(word) & 0xff000000u) {
1896 ++word;
1897 }
1898 ++word;
1899
1900 for (; word < entrypoint.len(); word++) interfaces.push_back(entrypoint.word(word));
1901
1902 return interfaces;
1903}