90679f1dcf
This effectively disables validation of the size of the data provided.
5437 lines
212 KiB
C
5437 lines
212 KiB
C
/*
|
|
Copyright 2017-2022 Google Inc.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
#include "spirv_reflect.h"
|
|
|
|
#include <assert.h>
|
|
#include <stdbool.h>
|
|
#include <string.h>
|
|
|
|
#if defined(WIN32)
|
|
#define _CRTDBG_MAP_ALLOC
|
|
#include <crtdbg.h>
|
|
#include <stdlib.h>
|
|
#else
|
|
#include <stdlib.h>
|
|
#endif
|
|
|
|
#if defined(__clang__) || (defined(__GNUC__) && __GNUC__ >= 7) || defined(__APPLE_CC__)
|
|
#define FALLTHROUGH __attribute__((fallthrough))
|
|
#else
|
|
#define FALLTHROUGH
|
|
#endif
|
|
|
|
#if defined(SPIRV_REFLECT_ENABLE_ASSERTS)
|
|
#define SPV_REFLECT_ASSERT(COND) assert(COND);
|
|
#else
|
|
#define SPV_REFLECT_ASSERT(COND)
|
|
#endif
|
|
|
|
// clang-format off
|
|
enum {
|
|
SPIRV_STARTING_WORD_INDEX = 5,
|
|
SPIRV_WORD_SIZE = sizeof(uint32_t),
|
|
SPIRV_BYTE_WIDTH = 8,
|
|
SPIRV_MINIMUM_FILE_SIZE = SPIRV_STARTING_WORD_INDEX * SPIRV_WORD_SIZE,
|
|
SPIRV_DATA_ALIGNMENT = 4 * SPIRV_WORD_SIZE, // 16
|
|
SPIRV_ACCESS_CHAIN_INDEX_OFFSET = 4,
|
|
};
|
|
|
|
enum {
|
|
INVALID_VALUE = 0xFFFFFFFF,
|
|
};
|
|
|
|
enum {
|
|
MAX_NODE_NAME_LENGTH = 1024,
|
|
// Number of unique PhysicalStorageBuffer structs tracked to detect recursion
|
|
MAX_RECURSIVE_PHYSICAL_POINTER_CHECK = 128,
|
|
};
|
|
|
|
enum {
|
|
IMAGE_SAMPLED = 1,
|
|
IMAGE_STORAGE = 2,
|
|
};
|
|
|
|
typedef struct SpvReflectPrvArrayTraits {
|
|
uint32_t element_type_id;
|
|
uint32_t length_id;
|
|
} SpvReflectPrvArrayTraits;
|
|
|
|
typedef struct SpvReflectPrvImageTraits {
|
|
uint32_t sampled_type_id;
|
|
SpvDim dim;
|
|
uint32_t depth;
|
|
uint32_t arrayed;
|
|
uint32_t ms;
|
|
uint32_t sampled;
|
|
SpvImageFormat image_format;
|
|
} SpvReflectPrvImageTraits;
|
|
|
|
typedef struct SpvReflectPrvNumberDecoration {
|
|
uint32_t word_offset;
|
|
uint32_t value;
|
|
} SpvReflectPrvNumberDecoration;
|
|
|
|
typedef struct SpvReflectPrvStringDecoration {
|
|
uint32_t word_offset;
|
|
const char* value;
|
|
} SpvReflectPrvStringDecoration;
|
|
|
|
typedef struct SpvReflectPrvDecorations {
|
|
bool is_relaxed_precision;
|
|
bool is_block;
|
|
bool is_buffer_block;
|
|
bool is_row_major;
|
|
bool is_column_major;
|
|
bool is_built_in;
|
|
bool is_noperspective;
|
|
bool is_flat;
|
|
bool is_non_writable;
|
|
bool is_non_readable;
|
|
bool is_patch;
|
|
bool is_per_vertex;
|
|
bool is_per_task;
|
|
bool is_weight_texture;
|
|
bool is_block_match_texture;
|
|
SpvReflectUserType user_type;
|
|
SpvReflectPrvNumberDecoration set;
|
|
SpvReflectPrvNumberDecoration binding;
|
|
SpvReflectPrvNumberDecoration input_attachment_index;
|
|
SpvReflectPrvNumberDecoration location;
|
|
SpvReflectPrvNumberDecoration component;
|
|
SpvReflectPrvNumberDecoration offset;
|
|
SpvReflectPrvNumberDecoration uav_counter_buffer;
|
|
SpvReflectPrvStringDecoration semantic;
|
|
uint32_t array_stride;
|
|
uint32_t matrix_stride;
|
|
uint32_t spec_id;
|
|
SpvBuiltIn built_in;
|
|
} SpvReflectPrvDecorations;
|
|
|
|
typedef struct SpvReflectPrvNode {
|
|
uint32_t result_id;
|
|
SpvOp op;
|
|
uint32_t result_type_id;
|
|
uint32_t type_id;
|
|
SpvCapability capability;
|
|
SpvStorageClass storage_class;
|
|
uint32_t word_offset;
|
|
uint32_t word_count;
|
|
bool is_type;
|
|
|
|
SpvReflectPrvArrayTraits array_traits;
|
|
SpvReflectPrvImageTraits image_traits;
|
|
uint32_t image_type_id;
|
|
|
|
const char* name;
|
|
SpvReflectPrvDecorations decorations;
|
|
uint32_t member_count;
|
|
const char** member_names;
|
|
SpvReflectPrvDecorations* member_decorations;
|
|
} SpvReflectPrvNode;
|
|
|
|
typedef struct SpvReflectPrvString {
|
|
uint32_t result_id;
|
|
const char* string;
|
|
} SpvReflectPrvString;
|
|
|
|
// There are a limit set of instructions that can touch an OpVariable,
|
|
// these are represented here with how it was accessed
|
|
// Examples:
|
|
// OpImageRead -> OpLoad -> OpVariable
|
|
// OpImageWrite -> OpLoad -> OpVariable
|
|
// OpStore -> OpAccessChain -> OpAccessChain -> OpVariable
|
|
// OpAtomicIAdd -> OpAccessChain -> OpVariable
|
|
// OpAtomicLoad -> OpImageTexelPointer -> OpVariable
|
|
typedef struct SpvReflectPrvAccessedVariable {
|
|
SpvReflectPrvNode* p_node;
|
|
uint32_t result_id;
|
|
uint32_t variable_ptr;
|
|
} SpvReflectPrvAccessedVariable;
|
|
|
|
typedef struct SpvReflectPrvFunction {
|
|
uint32_t id;
|
|
uint32_t callee_count;
|
|
uint32_t* callees;
|
|
struct SpvReflectPrvFunction** callee_ptrs;
|
|
uint32_t accessed_variable_count;
|
|
SpvReflectPrvAccessedVariable* accessed_variables;
|
|
} SpvReflectPrvFunction;
|
|
|
|
typedef struct SpvReflectPrvAccessChain {
|
|
uint32_t result_id;
|
|
uint32_t result_type_id;
|
|
//
|
|
// Pointing to the base of a composite object.
|
|
// Generally the id of descriptor block variable
|
|
uint32_t base_id;
|
|
//
|
|
// From spec:
|
|
// The first index in Indexes will select the
|
|
// top-level member/element/component/element
|
|
// of the base composite
|
|
uint32_t index_count;
|
|
uint32_t* indexes;
|
|
//
|
|
// Block variable ac is pointing to (for block references)
|
|
SpvReflectBlockVariable* block_var;
|
|
} SpvReflectPrvAccessChain;
|
|
|
|
// To prevent infinite recursion, we never walk down a
|
|
// PhysicalStorageBuffer struct twice, but incase a 2nd variable
|
|
// needs to use that struct, save a copy
|
|
typedef struct SpvReflectPrvPhysicalPointerStruct {
|
|
uint32_t struct_id;
|
|
// first variable to see the PhysicalStorageBuffer struct
|
|
SpvReflectBlockVariable* p_var;
|
|
} SpvReflectPrvPhysicalPointerStruct;
|
|
|
|
typedef struct SpvReflectPrvParser {
|
|
size_t spirv_word_count;
|
|
uint32_t* spirv_code;
|
|
uint32_t string_count;
|
|
SpvReflectPrvString* strings;
|
|
SpvSourceLanguage source_language;
|
|
uint32_t source_language_version;
|
|
uint32_t source_file_id;
|
|
const char* source_embedded;
|
|
size_t node_count;
|
|
SpvReflectPrvNode* nodes;
|
|
uint32_t entry_point_count;
|
|
uint32_t capability_count;
|
|
uint32_t function_count;
|
|
SpvReflectPrvFunction* functions;
|
|
uint32_t access_chain_count;
|
|
SpvReflectPrvAccessChain* access_chains;
|
|
|
|
uint32_t type_count;
|
|
uint32_t descriptor_count;
|
|
uint32_t push_constant_count;
|
|
|
|
SpvReflectTypeDescription* physical_pointer_check[MAX_RECURSIVE_PHYSICAL_POINTER_CHECK];
|
|
uint32_t physical_pointer_count;
|
|
|
|
SpvReflectPrvPhysicalPointerStruct* physical_pointer_structs;
|
|
uint32_t physical_pointer_struct_count;
|
|
} SpvReflectPrvParser;
|
|
// clang-format on
|
|
|
|
static uint32_t Max(uint32_t a, uint32_t b) { return a > b ? a : b; }
|
|
static uint32_t Min(uint32_t a, uint32_t b) { return a < b ? a : b; }
|
|
|
|
static uint32_t RoundUp(uint32_t value, uint32_t multiple) {
|
|
assert(multiple && ((multiple & (multiple - 1)) == 0));
|
|
return (value + multiple - 1) & ~(multiple - 1);
|
|
}
|
|
|
|
#define IsNull(ptr) (ptr == NULL)
|
|
|
|
#define IsNotNull(ptr) (ptr != NULL)
|
|
|
|
#define SafeFree(ptr) \
|
|
{ \
|
|
free((void*)ptr); \
|
|
ptr = NULL; \
|
|
}
|
|
|
|
static int SortCompareUint32(const void* a, const void* b) {
|
|
const uint32_t* p_a = (const uint32_t*)a;
|
|
const uint32_t* p_b = (const uint32_t*)b;
|
|
|
|
return (int)*p_a - (int)*p_b;
|
|
}
|
|
|
|
static int SortCompareAccessedVariable(const void* a, const void* b) {
|
|
const SpvReflectPrvAccessedVariable* p_a = (const SpvReflectPrvAccessedVariable*)a;
|
|
const SpvReflectPrvAccessedVariable* p_b = (const SpvReflectPrvAccessedVariable*)b;
|
|
|
|
return (int)p_a->variable_ptr - (int)p_b->variable_ptr;
|
|
}
|
|
|
|
//
|
|
// De-duplicates a sorted array and returns the new size.
|
|
//
|
|
// Note: The array doesn't actually need to be sorted, just
|
|
// arranged into "runs" so that all the entries with one
|
|
// value are adjacent.
|
|
//
|
|
static size_t DedupSortedUint32(uint32_t* arr, size_t size) {
|
|
if (size == 0) {
|
|
return 0;
|
|
}
|
|
size_t dedup_idx = 0;
|
|
for (size_t i = 0; i < size; ++i) {
|
|
if (arr[dedup_idx] != arr[i]) {
|
|
++dedup_idx;
|
|
arr[dedup_idx] = arr[i];
|
|
}
|
|
}
|
|
return dedup_idx + 1;
|
|
}
|
|
|
|
static bool SearchSortedUint32(const uint32_t* arr, size_t size, uint32_t target) {
|
|
size_t lo = 0;
|
|
size_t hi = size;
|
|
while (lo < hi) {
|
|
size_t mid = (hi - lo) / 2 + lo;
|
|
if (arr[mid] == target) {
|
|
return true;
|
|
} else if (arr[mid] < target) {
|
|
lo = mid + 1;
|
|
} else {
|
|
hi = mid;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static SpvReflectResult IntersectSortedAccessedVariable(const SpvReflectPrvAccessedVariable* p_arr0, size_t arr0_size,
|
|
const uint32_t* p_arr1, size_t arr1_size, uint32_t** pp_res,
|
|
size_t* res_size) {
|
|
*pp_res = NULL;
|
|
*res_size = 0;
|
|
if (IsNull(p_arr0) || IsNull(p_arr1)) {
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
const SpvReflectPrvAccessedVariable* p_arr0_end = p_arr0 + arr0_size;
|
|
const uint32_t* p_arr1_end = p_arr1 + arr1_size;
|
|
|
|
const SpvReflectPrvAccessedVariable* p_idx0 = p_arr0;
|
|
const uint32_t* p_idx1 = p_arr1;
|
|
while (p_idx0 != p_arr0_end && p_idx1 != p_arr1_end) {
|
|
if (p_idx0->variable_ptr < *p_idx1) {
|
|
++p_idx0;
|
|
} else if (p_idx0->variable_ptr > *p_idx1) {
|
|
++p_idx1;
|
|
} else {
|
|
++*res_size;
|
|
++p_idx0;
|
|
++p_idx1;
|
|
}
|
|
}
|
|
|
|
if (*res_size > 0) {
|
|
*pp_res = (uint32_t*)calloc(*res_size, sizeof(**pp_res));
|
|
if (IsNull(*pp_res)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
uint32_t* p_idxr = *pp_res;
|
|
p_idx0 = p_arr0;
|
|
p_idx1 = p_arr1;
|
|
while (p_idx0 != p_arr0_end && p_idx1 != p_arr1_end) {
|
|
if (p_idx0->variable_ptr < *p_idx1) {
|
|
++p_idx0;
|
|
} else if (p_idx0->variable_ptr > *p_idx1) {
|
|
++p_idx1;
|
|
} else {
|
|
*(p_idxr++) = p_idx0->variable_ptr;
|
|
++p_idx0;
|
|
++p_idx1;
|
|
}
|
|
}
|
|
}
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static bool InRange(const SpvReflectPrvParser* p_parser, uint32_t index) {
|
|
bool in_range = false;
|
|
if (IsNotNull(p_parser)) {
|
|
in_range = (index < p_parser->spirv_word_count);
|
|
}
|
|
return in_range;
|
|
}
|
|
|
|
static SpvReflectResult ReadU32(SpvReflectPrvParser* p_parser, uint32_t word_offset, uint32_t* p_value) {
|
|
assert(IsNotNull(p_parser));
|
|
assert(IsNotNull(p_parser->spirv_code));
|
|
assert(InRange(p_parser, word_offset));
|
|
SpvReflectResult result = SPV_REFLECT_RESULT_ERROR_SPIRV_UNEXPECTED_EOF;
|
|
if (IsNotNull(p_parser) && IsNotNull(p_parser->spirv_code) && InRange(p_parser, word_offset)) {
|
|
*p_value = *(p_parser->spirv_code + word_offset);
|
|
result = SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
#define UNCHECKED_READU32(parser, word_offset, value) \
|
|
{ (void)ReadU32(parser, word_offset, (uint32_t*)&(value)); }
|
|
|
|
#define CHECKED_READU32(parser, word_offset, value) \
|
|
{ \
|
|
SpvReflectResult checked_readu32_result = ReadU32(parser, word_offset, (uint32_t*)&(value)); \
|
|
if (checked_readu32_result != SPV_REFLECT_RESULT_SUCCESS) { \
|
|
return checked_readu32_result; \
|
|
} \
|
|
}
|
|
|
|
#define CHECKED_READU32_CAST(parser, word_offset, cast_to_type, value) \
|
|
{ \
|
|
uint32_t checked_readu32_cast_u32 = UINT32_MAX; \
|
|
SpvReflectResult checked_readu32_cast_result = ReadU32(parser, word_offset, (uint32_t*)&(checked_readu32_cast_u32)); \
|
|
if (checked_readu32_cast_result != SPV_REFLECT_RESULT_SUCCESS) { \
|
|
return checked_readu32_cast_result; \
|
|
} \
|
|
value = (cast_to_type)checked_readu32_cast_u32; \
|
|
}
|
|
|
|
#define IF_READU32(result, parser, word_offset, value) \
|
|
if ((result) == SPV_REFLECT_RESULT_SUCCESS) { \
|
|
result = ReadU32(parser, word_offset, (uint32_t*)&(value)); \
|
|
}
|
|
|
|
#define IF_READU32_CAST(result, parser, word_offset, cast_to_type, value) \
|
|
if ((result) == SPV_REFLECT_RESULT_SUCCESS) { \
|
|
uint32_t if_readu32_cast_u32 = UINT32_MAX; \
|
|
result = ReadU32(parser, word_offset, &if_readu32_cast_u32); \
|
|
if ((result) == SPV_REFLECT_RESULT_SUCCESS) { \
|
|
value = (cast_to_type)if_readu32_cast_u32; \
|
|
} \
|
|
}
|
|
|
|
static SpvReflectResult ReadStr(SpvReflectPrvParser* p_parser, uint32_t word_offset, uint32_t word_index, uint32_t word_count,
|
|
uint32_t* p_buf_size, char* p_buf) {
|
|
uint32_t limit = (word_offset + word_count);
|
|
assert(IsNotNull(p_parser));
|
|
assert(IsNotNull(p_parser->spirv_code));
|
|
assert(InRange(p_parser, limit));
|
|
SpvReflectResult result = SPV_REFLECT_RESULT_ERROR_SPIRV_UNEXPECTED_EOF;
|
|
if (IsNotNull(p_parser) && IsNotNull(p_parser->spirv_code) && InRange(p_parser, limit)) {
|
|
const char* c_str = (const char*)(p_parser->spirv_code + word_offset + word_index);
|
|
uint32_t n = word_count * SPIRV_WORD_SIZE;
|
|
uint32_t length_with_terminator = 0;
|
|
for (uint32_t i = 0; i < n; ++i) {
|
|
char c = *(c_str + i);
|
|
if (c == 0) {
|
|
length_with_terminator = i + 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (length_with_terminator > 0) {
|
|
result = SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
if (IsNotNull(p_buf_size) && IsNotNull(p_buf)) {
|
|
result = SPV_REFLECT_RESULT_ERROR_RANGE_EXCEEDED;
|
|
if (length_with_terminator <= *p_buf_size) {
|
|
memset(p_buf, 0, *p_buf_size);
|
|
memcpy(p_buf, c_str, length_with_terminator);
|
|
result = SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
} else {
|
|
if (IsNotNull(p_buf_size)) {
|
|
*p_buf_size = length_with_terminator;
|
|
result = SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
static SpvReflectDecorationFlags ApplyDecorations(const SpvReflectPrvDecorations* p_decoration_fields) {
|
|
SpvReflectDecorationFlags decorations = SPV_REFLECT_DECORATION_NONE;
|
|
if (p_decoration_fields->is_relaxed_precision) {
|
|
decorations |= SPV_REFLECT_DECORATION_RELAXED_PRECISION;
|
|
}
|
|
if (p_decoration_fields->is_block) {
|
|
decorations |= SPV_REFLECT_DECORATION_BLOCK;
|
|
}
|
|
if (p_decoration_fields->is_buffer_block) {
|
|
decorations |= SPV_REFLECT_DECORATION_BUFFER_BLOCK;
|
|
}
|
|
if (p_decoration_fields->is_row_major) {
|
|
decorations |= SPV_REFLECT_DECORATION_ROW_MAJOR;
|
|
}
|
|
if (p_decoration_fields->is_column_major) {
|
|
decorations |= SPV_REFLECT_DECORATION_COLUMN_MAJOR;
|
|
}
|
|
if (p_decoration_fields->is_built_in) {
|
|
decorations |= SPV_REFLECT_DECORATION_BUILT_IN;
|
|
}
|
|
if (p_decoration_fields->is_noperspective) {
|
|
decorations |= SPV_REFLECT_DECORATION_NOPERSPECTIVE;
|
|
}
|
|
if (p_decoration_fields->is_flat) {
|
|
decorations |= SPV_REFLECT_DECORATION_FLAT;
|
|
}
|
|
if (p_decoration_fields->is_non_writable) {
|
|
decorations |= SPV_REFLECT_DECORATION_NON_WRITABLE;
|
|
}
|
|
if (p_decoration_fields->is_non_readable) {
|
|
decorations |= SPV_REFLECT_DECORATION_NON_READABLE;
|
|
}
|
|
if (p_decoration_fields->is_patch) {
|
|
decorations |= SPV_REFLECT_DECORATION_PATCH;
|
|
}
|
|
if (p_decoration_fields->is_per_vertex) {
|
|
decorations |= SPV_REFLECT_DECORATION_PER_VERTEX;
|
|
}
|
|
if (p_decoration_fields->is_per_task) {
|
|
decorations |= SPV_REFLECT_DECORATION_PER_TASK;
|
|
}
|
|
if (p_decoration_fields->is_weight_texture) {
|
|
decorations |= SPV_REFLECT_DECORATION_WEIGHT_TEXTURE;
|
|
}
|
|
if (p_decoration_fields->is_block_match_texture) {
|
|
decorations |= SPV_REFLECT_DECORATION_BLOCK_MATCH_TEXTURE;
|
|
}
|
|
return decorations;
|
|
}
|
|
|
|
static void ApplyNumericTraits(const SpvReflectTypeDescription* p_type, SpvReflectNumericTraits* p_numeric_traits) {
|
|
memcpy(p_numeric_traits, &p_type->traits.numeric, sizeof(p_type->traits.numeric));
|
|
}
|
|
|
|
static void ApplyArrayTraits(const SpvReflectTypeDescription* p_type, SpvReflectArrayTraits* p_array_traits) {
|
|
memcpy(p_array_traits, &p_type->traits.array, sizeof(p_type->traits.array));
|
|
}
|
|
|
|
static bool IsSpecConstant(const SpvReflectPrvNode* p_node) {
|
|
return (p_node->op == SpvOpSpecConstant || p_node->op == SpvOpSpecConstantOp || p_node->op == SpvOpSpecConstantTrue ||
|
|
p_node->op == SpvOpSpecConstantFalse);
|
|
}
|
|
|
|
static SpvReflectPrvNode* FindNode(SpvReflectPrvParser* p_parser, uint32_t result_id) {
|
|
SpvReflectPrvNode* p_node = NULL;
|
|
for (size_t i = 0; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_elem = &(p_parser->nodes[i]);
|
|
if (p_elem->result_id == result_id) {
|
|
p_node = p_elem;
|
|
break;
|
|
}
|
|
}
|
|
return p_node;
|
|
}
|
|
|
|
static SpvReflectTypeDescription* FindType(SpvReflectShaderModule* p_module, uint32_t type_id) {
|
|
SpvReflectTypeDescription* p_type = NULL;
|
|
for (size_t i = 0; i < p_module->_internal->type_description_count; ++i) {
|
|
SpvReflectTypeDescription* p_elem = &(p_module->_internal->type_descriptions[i]);
|
|
if (p_elem->id == type_id) {
|
|
p_type = p_elem;
|
|
break;
|
|
}
|
|
}
|
|
return p_type;
|
|
}
|
|
|
|
static SpvReflectPrvAccessChain* FindAccessChain(SpvReflectPrvParser* p_parser, uint32_t id) {
|
|
const uint32_t ac_count = p_parser->access_chain_count;
|
|
for (uint32_t i = 0; i < ac_count; i++) {
|
|
if (p_parser->access_chains[i].result_id == id) {
|
|
return &p_parser->access_chains[i];
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
// Access Chains mostly have their Base ID pointed directly to a OpVariable, but sometimes
|
|
// it will be through a load and this funciton handles the edge cases how to find that
|
|
static uint32_t FindAccessChainBaseVariable(SpvReflectPrvParser* p_parser, SpvReflectPrvAccessChain* p_access_chain) {
|
|
uint32_t base_id = p_access_chain->base_id;
|
|
SpvReflectPrvNode* base_node = FindNode(p_parser, base_id);
|
|
// TODO - This is just a band-aid to fix crashes.
|
|
// Need to understand why here and hopefully remove
|
|
// https://github.com/KhronosGroup/SPIRV-Reflect/pull/206
|
|
if (IsNull(base_node)) {
|
|
return 0;
|
|
}
|
|
while (base_node->op != SpvOpVariable) {
|
|
switch (base_node->op) {
|
|
case SpvOpLoad: {
|
|
UNCHECKED_READU32(p_parser, base_node->word_offset + 3, base_id);
|
|
} break;
|
|
case SpvOpFunctionParameter: {
|
|
UNCHECKED_READU32(p_parser, base_node->word_offset + 2, base_id);
|
|
} break;
|
|
case SpvOpBitcast:
|
|
// This can be caused by something like GL_EXT_buffer_reference_uvec2 trying to load a pointer.
|
|
// We currently call from a push constant, so no way to have a reference loop back into the PC block
|
|
return 0;
|
|
default: {
|
|
assert(false);
|
|
} break;
|
|
}
|
|
|
|
SpvReflectPrvAccessChain* base_ac = FindAccessChain(p_parser, base_id);
|
|
if (base_ac == 0) {
|
|
return 0;
|
|
}
|
|
base_id = base_ac->base_id;
|
|
base_node = FindNode(p_parser, base_id);
|
|
if (IsNull(base_node)) {
|
|
return 0;
|
|
}
|
|
}
|
|
return base_id;
|
|
}
|
|
|
|
static SpvReflectBlockVariable* GetRefBlkVar(SpvReflectPrvParser* p_parser, SpvReflectPrvAccessChain* p_access_chain) {
|
|
uint32_t base_id = p_access_chain->base_id;
|
|
SpvReflectPrvNode* base_node = FindNode(p_parser, base_id);
|
|
assert(base_node->op == SpvOpLoad);
|
|
UNCHECKED_READU32(p_parser, base_node->word_offset + 3, base_id);
|
|
SpvReflectPrvAccessChain* base_ac = FindAccessChain(p_parser, base_id);
|
|
assert(base_ac != 0);
|
|
SpvReflectBlockVariable* base_var = base_ac->block_var;
|
|
assert(base_var != 0);
|
|
return base_var;
|
|
}
|
|
|
|
bool IsPointerToPointer(SpvReflectPrvParser* p_parser, uint32_t type_id) {
|
|
SpvReflectPrvNode* ptr_node = FindNode(p_parser, type_id);
|
|
if (IsNull(ptr_node) || (ptr_node->op != SpvOpTypePointer)) {
|
|
return false;
|
|
}
|
|
uint32_t pte_id = 0;
|
|
UNCHECKED_READU32(p_parser, ptr_node->word_offset + 3, pte_id);
|
|
SpvReflectPrvNode* pte_node = FindNode(p_parser, pte_id);
|
|
if (IsNull(pte_node)) {
|
|
return false;
|
|
}
|
|
return pte_node->op == SpvOpTypePointer;
|
|
}
|
|
|
|
static SpvReflectResult CreateParser(size_t size, void* p_code, SpvReflectPrvParser* p_parser) {
|
|
if (p_code == NULL) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
if (size < SPIRV_MINIMUM_FILE_SIZE) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_CODE_SIZE;
|
|
}
|
|
if ((size % 4) != 0) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_CODE_SIZE;
|
|
}
|
|
|
|
p_parser->spirv_word_count = size / SPIRV_WORD_SIZE;
|
|
p_parser->spirv_code = (uint32_t*)p_code;
|
|
|
|
if (p_parser->spirv_code[0] != SpvMagicNumber) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_MAGIC_NUMBER;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static void DestroyParser(SpvReflectPrvParser* p_parser) {
|
|
if (!IsNull(p_parser->nodes)) {
|
|
// Free nodes
|
|
for (size_t i = 0; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
if (IsNotNull(p_node->member_names)) {
|
|
SafeFree(p_node->member_names);
|
|
}
|
|
if (IsNotNull(p_node->member_decorations)) {
|
|
SafeFree(p_node->member_decorations);
|
|
}
|
|
}
|
|
|
|
// Free functions
|
|
for (size_t i = 0; i < p_parser->function_count; ++i) {
|
|
SafeFree(p_parser->functions[i].callees);
|
|
SafeFree(p_parser->functions[i].callee_ptrs);
|
|
SafeFree(p_parser->functions[i].accessed_variables);
|
|
}
|
|
|
|
// Free access chains
|
|
for (uint32_t i = 0; i < p_parser->access_chain_count; ++i) {
|
|
SafeFree(p_parser->access_chains[i].indexes);
|
|
}
|
|
|
|
SafeFree(p_parser->nodes);
|
|
SafeFree(p_parser->strings);
|
|
SafeFree(p_parser->source_embedded);
|
|
SafeFree(p_parser->functions);
|
|
SafeFree(p_parser->access_chains);
|
|
|
|
if (IsNotNull(p_parser->physical_pointer_structs)) {
|
|
SafeFree(p_parser->physical_pointer_structs);
|
|
}
|
|
p_parser->node_count = 0;
|
|
}
|
|
}
|
|
|
|
static SpvReflectResult ParseNodes(SpvReflectPrvParser* p_parser) {
|
|
assert(IsNotNull(p_parser));
|
|
assert(IsNotNull(p_parser->spirv_code));
|
|
|
|
uint32_t* p_spirv = p_parser->spirv_code;
|
|
uint32_t spirv_word_index = SPIRV_STARTING_WORD_INDEX;
|
|
|
|
// Count nodes
|
|
uint32_t node_count = 0;
|
|
while (spirv_word_index < p_parser->spirv_word_count) {
|
|
uint32_t word = p_spirv[spirv_word_index];
|
|
SpvOp op = (SpvOp)(word & 0xFFFF);
|
|
uint32_t node_word_count = (word >> 16) & 0xFFFF;
|
|
if (node_word_count == 0) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_INSTRUCTION;
|
|
}
|
|
if (op == SpvOpAccessChain) {
|
|
++(p_parser->access_chain_count);
|
|
}
|
|
spirv_word_index += node_word_count;
|
|
++node_count;
|
|
}
|
|
|
|
if (node_count == 0) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_UNEXPECTED_EOF;
|
|
}
|
|
|
|
// Allocate nodes
|
|
p_parser->node_count = node_count;
|
|
p_parser->nodes = (SpvReflectPrvNode*)calloc(p_parser->node_count, sizeof(*(p_parser->nodes)));
|
|
if (IsNull(p_parser->nodes)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
// Mark all nodes with an invalid state
|
|
for (uint32_t i = 0; i < node_count; ++i) {
|
|
p_parser->nodes[i].op = (SpvOp)INVALID_VALUE;
|
|
p_parser->nodes[i].storage_class = (SpvStorageClass)INVALID_VALUE;
|
|
p_parser->nodes[i].decorations.set.value = (uint32_t)INVALID_VALUE;
|
|
p_parser->nodes[i].decorations.binding.value = (uint32_t)INVALID_VALUE;
|
|
p_parser->nodes[i].decorations.location.value = (uint32_t)INVALID_VALUE;
|
|
p_parser->nodes[i].decorations.component.value = (uint32_t)INVALID_VALUE;
|
|
p_parser->nodes[i].decorations.offset.value = (uint32_t)INVALID_VALUE;
|
|
p_parser->nodes[i].decorations.uav_counter_buffer.value = (uint32_t)INVALID_VALUE;
|
|
p_parser->nodes[i].decorations.spec_id = (uint32_t)INVALID_VALUE;
|
|
p_parser->nodes[i].decorations.built_in = (SpvBuiltIn)INVALID_VALUE;
|
|
}
|
|
// Mark source file id node
|
|
p_parser->source_file_id = (uint32_t)INVALID_VALUE;
|
|
p_parser->source_embedded = NULL;
|
|
|
|
// Function node
|
|
uint32_t function_node = (uint32_t)INVALID_VALUE;
|
|
|
|
// Allocate access chain
|
|
if (p_parser->access_chain_count > 0) {
|
|
p_parser->access_chains = (SpvReflectPrvAccessChain*)calloc(p_parser->access_chain_count, sizeof(*(p_parser->access_chains)));
|
|
if (IsNull(p_parser->access_chains)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
}
|
|
|
|
// Parse nodes
|
|
uint32_t node_index = 0;
|
|
uint32_t access_chain_index = 0;
|
|
spirv_word_index = SPIRV_STARTING_WORD_INDEX;
|
|
while (spirv_word_index < p_parser->spirv_word_count) {
|
|
uint32_t word = p_spirv[spirv_word_index];
|
|
SpvOp op = (SpvOp)(word & 0xFFFF);
|
|
uint32_t node_word_count = (word >> 16) & 0xFFFF;
|
|
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[node_index]);
|
|
p_node->op = op;
|
|
p_node->word_offset = spirv_word_index;
|
|
p_node->word_count = node_word_count;
|
|
|
|
switch (p_node->op) {
|
|
default:
|
|
break;
|
|
|
|
case SpvOpString: {
|
|
++(p_parser->string_count);
|
|
} break;
|
|
|
|
case SpvOpSource: {
|
|
CHECKED_READU32_CAST(p_parser, p_node->word_offset + 1, SpvSourceLanguage, p_parser->source_language);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, p_parser->source_language_version);
|
|
if (p_node->word_count >= 4) {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 3, p_parser->source_file_id);
|
|
}
|
|
if (p_node->word_count >= 5) {
|
|
const char* p_source = (const char*)(p_parser->spirv_code + p_node->word_offset + 4);
|
|
|
|
const size_t source_len = strlen(p_source);
|
|
char* p_source_temp = (char*)calloc(source_len + 1, sizeof(char));
|
|
|
|
if (IsNull(p_source_temp)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
strcpy_s(p_source_temp, source_len + 1, p_source);
|
|
#else
|
|
strcpy(p_source_temp, p_source);
|
|
#endif
|
|
|
|
SafeFree(p_parser->source_embedded);
|
|
p_parser->source_embedded = p_source_temp;
|
|
}
|
|
} break;
|
|
|
|
case SpvOpSourceContinued: {
|
|
const char* p_source = (const char*)(p_parser->spirv_code + p_node->word_offset + 1);
|
|
|
|
const size_t source_len = strlen(p_source);
|
|
const size_t embedded_source_len = strlen(p_parser->source_embedded);
|
|
char* p_continued_source = (char*)calloc(source_len + embedded_source_len + 1, sizeof(char));
|
|
|
|
if (IsNull(p_continued_source)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
strcpy_s(p_continued_source, embedded_source_len + 1, p_parser->source_embedded);
|
|
strcat_s(p_continued_source, embedded_source_len + source_len + 1, p_source);
|
|
#else
|
|
strcpy(p_continued_source, p_parser->source_embedded);
|
|
strcat(p_continued_source, p_source);
|
|
#endif
|
|
|
|
SafeFree(p_parser->source_embedded);
|
|
p_parser->source_embedded = p_continued_source;
|
|
} break;
|
|
|
|
case SpvOpEntryPoint: {
|
|
++(p_parser->entry_point_count);
|
|
} break;
|
|
|
|
case SpvOpCapability: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, p_node->capability);
|
|
++(p_parser->capability_count);
|
|
} break;
|
|
|
|
case SpvOpName:
|
|
case SpvOpMemberName: {
|
|
uint32_t member_offset = (p_node->op == SpvOpMemberName) ? 1 : 0;
|
|
uint32_t name_start = p_node->word_offset + member_offset + 2;
|
|
p_node->name = (const char*)(p_parser->spirv_code + name_start);
|
|
} break;
|
|
|
|
case SpvOpTypeStruct: {
|
|
p_node->member_count = p_node->word_count - 2;
|
|
FALLTHROUGH;
|
|
} // Fall through
|
|
|
|
// This is all the rest of OpType* that need to be tracked
|
|
// Possible new extensions might expose new type, will need to be added
|
|
// here
|
|
case SpvOpTypeVoid:
|
|
case SpvOpTypeBool:
|
|
case SpvOpTypeInt:
|
|
case SpvOpTypeFloat:
|
|
case SpvOpTypeVector:
|
|
case SpvOpTypeMatrix:
|
|
case SpvOpTypeSampler:
|
|
case SpvOpTypeOpaque:
|
|
case SpvOpTypeFunction:
|
|
case SpvOpTypeEvent:
|
|
case SpvOpTypeDeviceEvent:
|
|
case SpvOpTypeReserveId:
|
|
case SpvOpTypeQueue:
|
|
case SpvOpTypePipe:
|
|
case SpvOpTypeAccelerationStructureKHR:
|
|
case SpvOpTypeRayQueryKHR:
|
|
case SpvOpTypeHitObjectNV:
|
|
case SpvOpTypeCooperativeMatrixNV:
|
|
case SpvOpTypeCooperativeMatrixKHR: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, p_node->result_id);
|
|
p_node->is_type = true;
|
|
} break;
|
|
|
|
case SpvOpTypeImage: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, p_node->result_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, p_node->image_traits.sampled_type_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 3, p_node->image_traits.dim);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 4, p_node->image_traits.depth);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 5, p_node->image_traits.arrayed);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 6, p_node->image_traits.ms);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 7, p_node->image_traits.sampled);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 8, p_node->image_traits.image_format);
|
|
p_node->is_type = true;
|
|
} break;
|
|
|
|
case SpvOpTypeSampledImage: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, p_node->result_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, p_node->image_type_id);
|
|
p_node->is_type = true;
|
|
} break;
|
|
|
|
case SpvOpTypeArray: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, p_node->result_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, p_node->array_traits.element_type_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 3, p_node->array_traits.length_id);
|
|
p_node->is_type = true;
|
|
} break;
|
|
|
|
case SpvOpTypeRuntimeArray: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, p_node->result_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, p_node->array_traits.element_type_id);
|
|
p_node->is_type = true;
|
|
} break;
|
|
|
|
case SpvOpTypePointer: {
|
|
uint32_t result_id;
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, result_id);
|
|
// Look for forward pointer. Clear result id if found
|
|
SpvReflectPrvNode* p_fwd_node = FindNode(p_parser, result_id);
|
|
if (p_fwd_node) {
|
|
p_fwd_node->result_id = 0;
|
|
}
|
|
// Register pointer type
|
|
p_node->result_id = result_id;
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, p_node->storage_class);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 3, p_node->type_id);
|
|
p_node->is_type = true;
|
|
} break;
|
|
|
|
case SpvOpTypeForwardPointer: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, p_node->result_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, p_node->storage_class);
|
|
p_node->is_type = true;
|
|
} break;
|
|
|
|
case SpvOpConstantTrue:
|
|
case SpvOpConstantFalse:
|
|
case SpvOpConstant:
|
|
case SpvOpConstantComposite:
|
|
case SpvOpConstantSampler:
|
|
case SpvOpConstantNull: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, p_node->result_type_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, p_node->result_id);
|
|
} break;
|
|
|
|
case SpvOpSpecConstantTrue:
|
|
case SpvOpSpecConstantFalse:
|
|
case SpvOpSpecConstant:
|
|
case SpvOpSpecConstantComposite:
|
|
case SpvOpSpecConstantOp: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, p_node->result_type_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, p_node->result_id);
|
|
} break;
|
|
|
|
case SpvOpVariable: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, p_node->type_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, p_node->result_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 3, p_node->storage_class);
|
|
} break;
|
|
|
|
case SpvOpLoad: {
|
|
// Only load enough so OpDecorate can reference the node, skip the remaining operands.
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, p_node->result_type_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, p_node->result_id);
|
|
} break;
|
|
|
|
case SpvOpAccessChain: {
|
|
SpvReflectPrvAccessChain* p_access_chain = &(p_parser->access_chains[access_chain_index]);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, p_access_chain->result_type_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, p_access_chain->result_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 3, p_access_chain->base_id);
|
|
//
|
|
// SPIRV_ACCESS_CHAIN_INDEX_OFFSET (4) is the number of words up until the first index:
|
|
// [Node, Result Type Id, Result Id, Base Id, <Indexes>]
|
|
//
|
|
p_access_chain->index_count = (node_word_count - SPIRV_ACCESS_CHAIN_INDEX_OFFSET);
|
|
if (p_access_chain->index_count > 0) {
|
|
p_access_chain->indexes = (uint32_t*)calloc(p_access_chain->index_count, sizeof(*(p_access_chain->indexes)));
|
|
if (IsNull(p_access_chain->indexes)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
// Parse any index values for access chain
|
|
for (uint32_t index_index = 0; index_index < p_access_chain->index_count; ++index_index) {
|
|
// Read index id
|
|
uint32_t index_id = 0;
|
|
CHECKED_READU32(p_parser, p_node->word_offset + SPIRV_ACCESS_CHAIN_INDEX_OFFSET + index_index, index_id);
|
|
// Find OpConstant node that contains index value
|
|
SpvReflectPrvNode* p_index_value_node = FindNode(p_parser, index_id);
|
|
if ((p_index_value_node != NULL) &&
|
|
(p_index_value_node->op == SpvOpConstant || p_index_value_node->op == SpvOpSpecConstant)) {
|
|
// Read index value
|
|
uint32_t index_value = UINT32_MAX;
|
|
CHECKED_READU32(p_parser, p_index_value_node->word_offset + 3, index_value);
|
|
assert(index_value != UINT32_MAX);
|
|
// Write index value to array
|
|
p_access_chain->indexes[index_index] = index_value;
|
|
}
|
|
}
|
|
}
|
|
++access_chain_index;
|
|
} break;
|
|
|
|
case SpvOpFunction: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, p_node->result_id);
|
|
// Count function definitions, not function declarations. To determine
|
|
// the difference, set an in-function variable, and then if an OpLabel
|
|
// is reached before the end of the function increment the function
|
|
// count.
|
|
function_node = node_index;
|
|
} break;
|
|
|
|
case SpvOpLabel: {
|
|
if (function_node != (uint32_t)INVALID_VALUE) {
|
|
SpvReflectPrvNode* p_func_node = &(p_parser->nodes[function_node]);
|
|
CHECKED_READU32(p_parser, p_func_node->word_offset + 2, p_func_node->result_id);
|
|
++(p_parser->function_count);
|
|
}
|
|
FALLTHROUGH;
|
|
} // Fall through
|
|
|
|
case SpvOpFunctionEnd: {
|
|
function_node = (uint32_t)INVALID_VALUE;
|
|
} break;
|
|
case SpvOpFunctionParameter: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, p_node->result_id);
|
|
} break;
|
|
case SpvOpBitcast:
|
|
case SpvOpShiftRightLogical:
|
|
case SpvOpIAdd:
|
|
case SpvOpISub:
|
|
case SpvOpIMul:
|
|
case SpvOpUDiv:
|
|
case SpvOpSDiv: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, p_node->result_id);
|
|
} break;
|
|
}
|
|
|
|
if (p_node->is_type) {
|
|
++(p_parser->type_count);
|
|
}
|
|
|
|
spirv_word_index += node_word_count;
|
|
++node_index;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseStrings(SpvReflectPrvParser* p_parser) {
|
|
assert(IsNotNull(p_parser));
|
|
assert(IsNotNull(p_parser->spirv_code));
|
|
assert(IsNotNull(p_parser->nodes));
|
|
|
|
// Early out
|
|
if (p_parser->string_count == 0) {
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
if (IsNotNull(p_parser) && IsNotNull(p_parser->spirv_code) && IsNotNull(p_parser->nodes)) {
|
|
// Allocate string storage
|
|
p_parser->strings = (SpvReflectPrvString*)calloc(p_parser->string_count, sizeof(*(p_parser->strings)));
|
|
|
|
uint32_t string_index = 0;
|
|
for (size_t i = 0; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
if (p_node->op != SpvOpString) {
|
|
continue;
|
|
}
|
|
|
|
// Paranoid check against string count
|
|
assert(string_index < p_parser->string_count);
|
|
if (string_index >= p_parser->string_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_COUNT_MISMATCH;
|
|
}
|
|
|
|
// Result id
|
|
SpvReflectPrvString* p_string = &(p_parser->strings[string_index]);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, p_string->result_id);
|
|
|
|
// String
|
|
uint32_t string_start = p_node->word_offset + 2;
|
|
p_string->string = (const char*)(p_parser->spirv_code + string_start);
|
|
|
|
// Increment string index
|
|
++string_index;
|
|
}
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseSource(SpvReflectPrvParser* p_parser, SpvReflectShaderModule* p_module) {
|
|
assert(IsNotNull(p_parser));
|
|
assert(IsNotNull(p_parser->spirv_code));
|
|
|
|
if (IsNotNull(p_parser) && IsNotNull(p_parser->spirv_code)) {
|
|
// Source file
|
|
if (IsNotNull(p_parser->strings)) {
|
|
for (uint32_t i = 0; i < p_parser->string_count; ++i) {
|
|
SpvReflectPrvString* p_string = &(p_parser->strings[i]);
|
|
if (p_string->result_id == p_parser->source_file_id) {
|
|
p_module->source_file = p_string->string;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Source code
|
|
if (IsNotNull(p_parser->source_embedded)) {
|
|
const size_t source_len = strlen(p_parser->source_embedded);
|
|
char* p_source = (char*)calloc(source_len + 1, sizeof(char));
|
|
|
|
if (IsNull(p_source)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
strcpy_s(p_source, source_len + 1, p_parser->source_embedded);
|
|
#else
|
|
strcpy(p_source, p_parser->source_embedded);
|
|
#endif
|
|
|
|
p_module->source_source = p_source;
|
|
}
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseFunction(SpvReflectPrvParser* p_parser, SpvReflectPrvNode* p_func_node, SpvReflectPrvFunction* p_func,
|
|
size_t first_label_index) {
|
|
p_func->id = p_func_node->result_id;
|
|
|
|
p_func->callee_count = 0;
|
|
p_func->accessed_variable_count = 0;
|
|
|
|
// First get count to know how much to allocate
|
|
for (size_t i = first_label_index; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
if (p_node->op == SpvOpFunctionEnd) {
|
|
break;
|
|
}
|
|
switch (p_node->op) {
|
|
case SpvOpFunctionCall: {
|
|
++(p_func->callee_count);
|
|
} break;
|
|
case SpvOpLoad:
|
|
case SpvOpAccessChain:
|
|
case SpvOpInBoundsAccessChain:
|
|
case SpvOpPtrAccessChain:
|
|
case SpvOpArrayLength:
|
|
case SpvOpGenericPtrMemSemantics:
|
|
case SpvOpInBoundsPtrAccessChain:
|
|
case SpvOpStore:
|
|
case SpvOpImageTexelPointer: {
|
|
++(p_func->accessed_variable_count);
|
|
} break;
|
|
case SpvOpCopyMemory:
|
|
case SpvOpCopyMemorySized: {
|
|
p_func->accessed_variable_count += 2;
|
|
} break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (p_func->callee_count > 0) {
|
|
p_func->callees = (uint32_t*)calloc(p_func->callee_count, sizeof(*(p_func->callees)));
|
|
if (IsNull(p_func->callees)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
}
|
|
|
|
if (p_func->accessed_variable_count > 0) {
|
|
p_func->accessed_variables =
|
|
(SpvReflectPrvAccessedVariable*)calloc(p_func->accessed_variable_count, sizeof(*(p_func->accessed_variables)));
|
|
if (IsNull(p_func->accessed_variables)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
}
|
|
|
|
p_func->callee_count = 0;
|
|
p_func->accessed_variable_count = 0;
|
|
// Now have allocation, fill in values
|
|
for (size_t i = first_label_index; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
if (p_node->op == SpvOpFunctionEnd) {
|
|
break;
|
|
}
|
|
switch (p_node->op) {
|
|
case SpvOpFunctionCall: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 3, p_func->callees[p_func->callee_count]);
|
|
(++p_func->callee_count);
|
|
} break;
|
|
case SpvOpLoad:
|
|
case SpvOpAccessChain:
|
|
case SpvOpInBoundsAccessChain:
|
|
case SpvOpPtrAccessChain:
|
|
case SpvOpArrayLength:
|
|
case SpvOpGenericPtrMemSemantics:
|
|
case SpvOpInBoundsPtrAccessChain:
|
|
case SpvOpImageTexelPointer: {
|
|
const uint32_t result_index = p_node->word_offset + 2;
|
|
const uint32_t ptr_index = p_node->word_offset + 3;
|
|
SpvReflectPrvAccessedVariable* access_ptr = &p_func->accessed_variables[p_func->accessed_variable_count];
|
|
|
|
access_ptr->p_node = p_node;
|
|
// Need to track Result ID as not sure there has been any memory access through here yet
|
|
CHECKED_READU32(p_parser, result_index, access_ptr->result_id);
|
|
CHECKED_READU32(p_parser, ptr_index, access_ptr->variable_ptr);
|
|
(++p_func->accessed_variable_count);
|
|
} break;
|
|
case SpvOpStore: {
|
|
const uint32_t result_index = p_node->word_offset + 2;
|
|
CHECKED_READU32(p_parser, result_index, p_func->accessed_variables[p_func->accessed_variable_count].variable_ptr);
|
|
p_func->accessed_variables[p_func->accessed_variable_count].p_node = p_node;
|
|
(++p_func->accessed_variable_count);
|
|
} break;
|
|
case SpvOpCopyMemory:
|
|
case SpvOpCopyMemorySized: {
|
|
// There is no result_id or node, being zero is same as being invalid
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1,
|
|
p_func->accessed_variables[p_func->accessed_variable_count].variable_ptr);
|
|
(++p_func->accessed_variable_count);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2,
|
|
p_func->accessed_variables[p_func->accessed_variable_count].variable_ptr);
|
|
(++p_func->accessed_variable_count);
|
|
} break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (p_func->callee_count > 0) {
|
|
qsort(p_func->callees, p_func->callee_count, sizeof(*(p_func->callees)), SortCompareUint32);
|
|
}
|
|
p_func->callee_count = (uint32_t)DedupSortedUint32(p_func->callees, p_func->callee_count);
|
|
|
|
if (p_func->accessed_variable_count > 0) {
|
|
qsort(p_func->accessed_variables, p_func->accessed_variable_count, sizeof(*(p_func->accessed_variables)),
|
|
SortCompareAccessedVariable);
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static int SortCompareFunctions(const void* a, const void* b) {
|
|
const SpvReflectPrvFunction* af = (const SpvReflectPrvFunction*)a;
|
|
const SpvReflectPrvFunction* bf = (const SpvReflectPrvFunction*)b;
|
|
return (int)af->id - (int)bf->id;
|
|
}
|
|
|
|
static SpvReflectResult ParseFunctions(SpvReflectPrvParser* p_parser) {
|
|
assert(IsNotNull(p_parser));
|
|
assert(IsNotNull(p_parser->spirv_code));
|
|
assert(IsNotNull(p_parser->nodes));
|
|
|
|
if (IsNotNull(p_parser) && IsNotNull(p_parser->spirv_code) && IsNotNull(p_parser->nodes)) {
|
|
if (p_parser->function_count == 0) {
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
p_parser->functions = (SpvReflectPrvFunction*)calloc(p_parser->function_count, sizeof(*(p_parser->functions)));
|
|
if (IsNull(p_parser->functions)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
|
|
size_t function_index = 0;
|
|
for (size_t i = 0; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
if (p_node->op != SpvOpFunction) {
|
|
continue;
|
|
}
|
|
|
|
// Skip over function declarations that aren't definitions
|
|
bool func_definition = false;
|
|
// Intentionally reuse i to avoid iterating over these nodes more than
|
|
// once
|
|
for (; i < p_parser->node_count; ++i) {
|
|
if (p_parser->nodes[i].op == SpvOpLabel) {
|
|
func_definition = true;
|
|
break;
|
|
}
|
|
if (p_parser->nodes[i].op == SpvOpFunctionEnd) {
|
|
break;
|
|
}
|
|
}
|
|
if (!func_definition) {
|
|
continue;
|
|
}
|
|
|
|
SpvReflectPrvFunction* p_function = &(p_parser->functions[function_index]);
|
|
|
|
SpvReflectResult result = ParseFunction(p_parser, p_node, p_function, i);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
|
|
++function_index;
|
|
}
|
|
|
|
qsort(p_parser->functions, p_parser->function_count, sizeof(*(p_parser->functions)), SortCompareFunctions);
|
|
|
|
// Once they're sorted, link the functions with pointers to improve graph
|
|
// traversal efficiency
|
|
for (size_t i = 0; i < p_parser->function_count; ++i) {
|
|
SpvReflectPrvFunction* p_func = &(p_parser->functions[i]);
|
|
if (p_func->callee_count == 0) {
|
|
continue;
|
|
}
|
|
p_func->callee_ptrs = (SpvReflectPrvFunction**)calloc(p_func->callee_count, sizeof(*(p_func->callee_ptrs)));
|
|
for (size_t j = 0, k = 0; j < p_func->callee_count; ++j) {
|
|
while (p_parser->functions[k].id != p_func->callees[j]) {
|
|
++k;
|
|
if (k >= p_parser->function_count) {
|
|
// Invalid called function ID somewhere
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
}
|
|
p_func->callee_ptrs[j] = &(p_parser->functions[k]);
|
|
}
|
|
}
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseMemberCounts(SpvReflectPrvParser* p_parser) {
|
|
assert(IsNotNull(p_parser));
|
|
assert(IsNotNull(p_parser->spirv_code));
|
|
assert(IsNotNull(p_parser->nodes));
|
|
|
|
if (IsNotNull(p_parser) && IsNotNull(p_parser->spirv_code) && IsNotNull(p_parser->nodes)) {
|
|
for (size_t i = 0; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
if ((p_node->op != SpvOpMemberName) && (p_node->op != SpvOpMemberDecorate)) {
|
|
continue;
|
|
}
|
|
|
|
uint32_t target_id = 0;
|
|
uint32_t member_index = (uint32_t)INVALID_VALUE;
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, target_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, member_index);
|
|
SpvReflectPrvNode* p_target_node = FindNode(p_parser, target_id);
|
|
// Not all nodes get parsed, so FindNode returning NULL is expected.
|
|
if (IsNull(p_target_node)) {
|
|
continue;
|
|
}
|
|
|
|
if (member_index == INVALID_VALUE) {
|
|
return SPV_REFLECT_RESULT_ERROR_RANGE_EXCEEDED;
|
|
}
|
|
|
|
p_target_node->member_count = Max(p_target_node->member_count, member_index + 1);
|
|
}
|
|
|
|
for (uint32_t i = 0; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
if (p_node->member_count == 0) {
|
|
continue;
|
|
}
|
|
|
|
p_node->member_names = (const char**)calloc(p_node->member_count, sizeof(*(p_node->member_names)));
|
|
if (IsNull(p_node->member_names)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
|
|
p_node->member_decorations = (SpvReflectPrvDecorations*)calloc(p_node->member_count, sizeof(*(p_node->member_decorations)));
|
|
if (IsNull(p_node->member_decorations)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
}
|
|
}
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseNames(SpvReflectPrvParser* p_parser) {
|
|
assert(IsNotNull(p_parser));
|
|
assert(IsNotNull(p_parser->spirv_code));
|
|
assert(IsNotNull(p_parser->nodes));
|
|
|
|
if (IsNotNull(p_parser) && IsNotNull(p_parser->spirv_code) && IsNotNull(p_parser->nodes)) {
|
|
for (size_t i = 0; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
if ((p_node->op != SpvOpName) && (p_node->op != SpvOpMemberName)) {
|
|
continue;
|
|
}
|
|
|
|
uint32_t target_id = 0;
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, target_id);
|
|
SpvReflectPrvNode* p_target_node = FindNode(p_parser, target_id);
|
|
// Not all nodes get parsed, so FindNode returning NULL is expected.
|
|
if (IsNull(p_target_node)) {
|
|
continue;
|
|
}
|
|
|
|
const char** pp_target_name = &(p_target_node->name);
|
|
if (p_node->op == SpvOpMemberName) {
|
|
uint32_t member_index = UINT32_MAX;
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, member_index);
|
|
pp_target_name = &(p_target_node->member_names[member_index]);
|
|
}
|
|
|
|
*pp_target_name = p_node->name;
|
|
}
|
|
}
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
// Returns true if user_type matches pattern or if user_type begins with pattern and the next character is ':'
|
|
// For example, UserTypeMatches("rwbuffer", "rwbuffer") will be true, UserTypeMatches("rwbuffer", "rwbuffer:<S>") will be true, and
|
|
// UserTypeMatches("rwbuffer", "rwbufferfoo") will be false.
|
|
static bool UserTypeMatches(const char* user_type, const char* pattern) {
|
|
const size_t pattern_length = strlen(pattern);
|
|
if (strncmp(user_type, pattern, pattern_length) == 0) {
|
|
if (user_type[pattern_length] == ':' || user_type[pattern_length] == '\0') {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static SpvReflectResult ParseDecorations(SpvReflectPrvParser* p_parser, SpvReflectShaderModule* p_module) {
|
|
uint32_t spec_constant_count = 0;
|
|
for (uint32_t i = 0; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
|
|
if ((p_node->op != SpvOpDecorate) && (p_node->op != SpvOpMemberDecorate) && (p_node->op != SpvOpDecorateId) &&
|
|
(p_node->op != SpvOpDecorateString) && (p_node->op != SpvOpMemberDecorateString)) {
|
|
continue;
|
|
}
|
|
|
|
// Need to adjust the read offset if this is a member decoration
|
|
uint32_t member_offset = 0;
|
|
if (p_node->op == SpvOpMemberDecorate) {
|
|
member_offset = 1;
|
|
}
|
|
|
|
// Get decoration
|
|
uint32_t decoration = (uint32_t)INVALID_VALUE;
|
|
CHECKED_READU32(p_parser, p_node->word_offset + member_offset + 2, decoration);
|
|
|
|
// Filter out the decoration that do not affect reflection, otherwise
|
|
// there will be random crashes because the nodes aren't found.
|
|
bool skip = false;
|
|
switch (decoration) {
|
|
default: {
|
|
skip = true;
|
|
} break;
|
|
case SpvDecorationRelaxedPrecision:
|
|
case SpvDecorationBlock:
|
|
case SpvDecorationBufferBlock:
|
|
case SpvDecorationColMajor:
|
|
case SpvDecorationRowMajor:
|
|
case SpvDecorationArrayStride:
|
|
case SpvDecorationMatrixStride:
|
|
case SpvDecorationBuiltIn:
|
|
case SpvDecorationNoPerspective:
|
|
case SpvDecorationFlat:
|
|
case SpvDecorationNonWritable:
|
|
case SpvDecorationNonReadable:
|
|
case SpvDecorationPatch:
|
|
case SpvDecorationPerVertexKHR:
|
|
case SpvDecorationPerTaskNV:
|
|
case SpvDecorationLocation:
|
|
case SpvDecorationComponent:
|
|
case SpvDecorationBinding:
|
|
case SpvDecorationDescriptorSet:
|
|
case SpvDecorationOffset:
|
|
case SpvDecorationInputAttachmentIndex:
|
|
case SpvDecorationSpecId:
|
|
case SpvDecorationWeightTextureQCOM:
|
|
case SpvDecorationBlockMatchTextureQCOM:
|
|
case SpvDecorationUserTypeGOOGLE:
|
|
case SpvDecorationHlslCounterBufferGOOGLE:
|
|
case SpvDecorationHlslSemanticGOOGLE: {
|
|
skip = false;
|
|
} break;
|
|
}
|
|
if (skip) {
|
|
continue;
|
|
}
|
|
|
|
// Find target node
|
|
uint32_t target_id = 0;
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, target_id);
|
|
SpvReflectPrvNode* p_target_node = FindNode(p_parser, target_id);
|
|
if (IsNull(p_target_node)) {
|
|
if ((p_node->op == (uint32_t)SpvOpDecorate) && (decoration == SpvDecorationRelaxedPrecision)) {
|
|
// Many OPs can be decorated that we don't care about. Ignore those.
|
|
// See https://github.com/KhronosGroup/SPIRV-Reflect/issues/134
|
|
continue;
|
|
}
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
// Get decorations
|
|
SpvReflectPrvDecorations* p_target_decorations = &(p_target_node->decorations);
|
|
// Update pointer if this is a member decoration
|
|
if (p_node->op == SpvOpMemberDecorate) {
|
|
uint32_t member_index = (uint32_t)INVALID_VALUE;
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, member_index);
|
|
p_target_decorations = &(p_target_node->member_decorations[member_index]);
|
|
}
|
|
|
|
switch (decoration) {
|
|
default:
|
|
break;
|
|
|
|
case SpvDecorationRelaxedPrecision: {
|
|
p_target_decorations->is_relaxed_precision = true;
|
|
} break;
|
|
|
|
case SpvDecorationBlock: {
|
|
p_target_decorations->is_block = true;
|
|
} break;
|
|
|
|
case SpvDecorationBufferBlock: {
|
|
p_target_decorations->is_buffer_block = true;
|
|
} break;
|
|
|
|
case SpvDecorationColMajor: {
|
|
p_target_decorations->is_column_major = true;
|
|
} break;
|
|
|
|
case SpvDecorationRowMajor: {
|
|
p_target_decorations->is_row_major = true;
|
|
} break;
|
|
|
|
case SpvDecorationArrayStride: {
|
|
uint32_t word_offset = p_node->word_offset + member_offset + 3;
|
|
CHECKED_READU32(p_parser, word_offset, p_target_decorations->array_stride);
|
|
} break;
|
|
|
|
case SpvDecorationMatrixStride: {
|
|
uint32_t word_offset = p_node->word_offset + member_offset + 3;
|
|
CHECKED_READU32(p_parser, word_offset, p_target_decorations->matrix_stride);
|
|
} break;
|
|
|
|
case SpvDecorationBuiltIn: {
|
|
p_target_decorations->is_built_in = true;
|
|
uint32_t word_offset = p_node->word_offset + member_offset + 3;
|
|
CHECKED_READU32_CAST(p_parser, word_offset, SpvBuiltIn, p_target_decorations->built_in);
|
|
} break;
|
|
|
|
case SpvDecorationNoPerspective: {
|
|
p_target_decorations->is_noperspective = true;
|
|
} break;
|
|
|
|
case SpvDecorationFlat: {
|
|
p_target_decorations->is_flat = true;
|
|
} break;
|
|
|
|
case SpvDecorationNonWritable: {
|
|
p_target_decorations->is_non_writable = true;
|
|
} break;
|
|
|
|
case SpvDecorationNonReadable: {
|
|
p_target_decorations->is_non_readable = true;
|
|
} break;
|
|
|
|
case SpvDecorationPatch: {
|
|
p_target_decorations->is_patch = true;
|
|
} break;
|
|
|
|
case SpvDecorationPerVertexKHR: {
|
|
p_target_decorations->is_per_vertex = true;
|
|
} break;
|
|
|
|
case SpvDecorationPerTaskNV: {
|
|
p_target_decorations->is_per_task = true;
|
|
} break;
|
|
|
|
case SpvDecorationLocation: {
|
|
uint32_t word_offset = p_node->word_offset + member_offset + 3;
|
|
CHECKED_READU32(p_parser, word_offset, p_target_decorations->location.value);
|
|
p_target_decorations->location.word_offset = word_offset;
|
|
} break;
|
|
|
|
case SpvDecorationComponent: {
|
|
uint32_t word_offset = p_node->word_offset + member_offset + 3;
|
|
CHECKED_READU32(p_parser, word_offset, p_target_decorations->component.value);
|
|
p_target_decorations->component.word_offset = word_offset;
|
|
} break;
|
|
|
|
case SpvDecorationBinding: {
|
|
uint32_t word_offset = p_node->word_offset + member_offset + 3;
|
|
CHECKED_READU32(p_parser, word_offset, p_target_decorations->binding.value);
|
|
p_target_decorations->binding.word_offset = word_offset;
|
|
} break;
|
|
|
|
case SpvDecorationDescriptorSet: {
|
|
uint32_t word_offset = p_node->word_offset + member_offset + 3;
|
|
CHECKED_READU32(p_parser, word_offset, p_target_decorations->set.value);
|
|
p_target_decorations->set.word_offset = word_offset;
|
|
} break;
|
|
|
|
case SpvDecorationOffset: {
|
|
uint32_t word_offset = p_node->word_offset + member_offset + 3;
|
|
CHECKED_READU32(p_parser, word_offset, p_target_decorations->offset.value);
|
|
p_target_decorations->offset.word_offset = word_offset;
|
|
} break;
|
|
|
|
case SpvDecorationInputAttachmentIndex: {
|
|
uint32_t word_offset = p_node->word_offset + member_offset + 3;
|
|
CHECKED_READU32(p_parser, word_offset, p_target_decorations->input_attachment_index.value);
|
|
p_target_decorations->input_attachment_index.word_offset = word_offset;
|
|
} break;
|
|
|
|
case SpvDecorationSpecId: {
|
|
// -- GODOT begin --
|
|
uint32_t word_offset = p_node->word_offset + member_offset+ 3;
|
|
CHECKED_READU32(p_parser, word_offset, p_target_decorations->spec_id);
|
|
// -- GODOT end --
|
|
spec_constant_count++;
|
|
} break;
|
|
|
|
case SpvDecorationHlslCounterBufferGOOGLE: {
|
|
uint32_t word_offset = p_node->word_offset + member_offset + 3;
|
|
CHECKED_READU32(p_parser, word_offset, p_target_decorations->uav_counter_buffer.value);
|
|
p_target_decorations->uav_counter_buffer.word_offset = word_offset;
|
|
} break;
|
|
|
|
case SpvDecorationHlslSemanticGOOGLE: {
|
|
uint32_t word_offset = p_node->word_offset + member_offset + 3;
|
|
p_target_decorations->semantic.value = (const char*)(p_parser->spirv_code + word_offset);
|
|
p_target_decorations->semantic.word_offset = word_offset;
|
|
} break;
|
|
|
|
case SpvDecorationWeightTextureQCOM: {
|
|
p_target_decorations->is_weight_texture = true;
|
|
} break;
|
|
|
|
case SpvDecorationBlockMatchTextureQCOM: {
|
|
p_target_decorations->is_block_match_texture = true;
|
|
} break;
|
|
}
|
|
|
|
if (p_node->op == SpvOpDecorateString && decoration == SpvDecorationUserTypeGOOGLE) {
|
|
uint32_t terminator = 0;
|
|
SpvReflectResult result = ReadStr(p_parser, p_node->word_offset + 3, 0, p_node->word_count, &terminator, NULL);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
const char* name = (const char*)(p_parser->spirv_code + p_node->word_offset + 3);
|
|
if (UserTypeMatches(name, "cbuffer")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_CBUFFER;
|
|
} else if (UserTypeMatches(name, "tbuffer")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_TBUFFER;
|
|
} else if (UserTypeMatches(name, "appendstructuredbuffer")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_APPEND_STRUCTURED_BUFFER;
|
|
} else if (UserTypeMatches(name, "buffer")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_BUFFER;
|
|
} else if (UserTypeMatches(name, "byteaddressbuffer")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_BYTE_ADDRESS_BUFFER;
|
|
} else if (UserTypeMatches(name, "constantbuffer")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_CONSTANT_BUFFER;
|
|
} else if (UserTypeMatches(name, "consumestructuredbuffer")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_CONSUME_STRUCTURED_BUFFER;
|
|
} else if (UserTypeMatches(name, "inputpatch")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_INPUT_PATCH;
|
|
} else if (UserTypeMatches(name, "outputpatch")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_OUTPUT_PATCH;
|
|
} else if (UserTypeMatches(name, "rasterizerorderedbuffer")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RASTERIZER_ORDERED_BUFFER;
|
|
} else if (UserTypeMatches(name, "rasterizerorderedbyteaddressbuffer")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RASTERIZER_ORDERED_BYTE_ADDRESS_BUFFER;
|
|
} else if (UserTypeMatches(name, "rasterizerorderedstructuredbuffer")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RASTERIZER_ORDERED_STRUCTURED_BUFFER;
|
|
} else if (UserTypeMatches(name, "rasterizerorderedtexture1d")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RASTERIZER_ORDERED_TEXTURE_1D;
|
|
} else if (UserTypeMatches(name, "rasterizerorderedtexture1darray")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RASTERIZER_ORDERED_TEXTURE_1D_ARRAY;
|
|
} else if (UserTypeMatches(name, "rasterizerorderedtexture2d")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RASTERIZER_ORDERED_TEXTURE_2D;
|
|
} else if (UserTypeMatches(name, "rasterizerorderedtexture2darray")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RASTERIZER_ORDERED_TEXTURE_2D_ARRAY;
|
|
} else if (UserTypeMatches(name, "rasterizerorderedtexture3d")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RASTERIZER_ORDERED_TEXTURE_3D;
|
|
} else if (UserTypeMatches(name, "raytracingaccelerationstructure")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RAYTRACING_ACCELERATION_STRUCTURE;
|
|
} else if (UserTypeMatches(name, "rwbuffer")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RW_BUFFER;
|
|
} else if (UserTypeMatches(name, "rwbyteaddressbuffer")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RW_BYTE_ADDRESS_BUFFER;
|
|
} else if (UserTypeMatches(name, "rwstructuredbuffer")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RW_STRUCTURED_BUFFER;
|
|
} else if (UserTypeMatches(name, "rwtexture1d")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RW_TEXTURE_1D;
|
|
} else if (UserTypeMatches(name, "rwtexture1darray")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RW_TEXTURE_1D_ARRAY;
|
|
} else if (UserTypeMatches(name, "rwtexture2d")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RW_TEXTURE_2D;
|
|
} else if (UserTypeMatches(name, "rwtexture2darray")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RW_TEXTURE_2D_ARRAY;
|
|
} else if (UserTypeMatches(name, "rwtexture3d")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_RW_TEXTURE_3D;
|
|
} else if (UserTypeMatches(name, "structuredbuffer")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_STRUCTURED_BUFFER;
|
|
} else if (UserTypeMatches(name, "subpassinput")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_SUBPASS_INPUT;
|
|
} else if (UserTypeMatches(name, "subpassinputms")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_SUBPASS_INPUT_MS;
|
|
} else if (UserTypeMatches(name, "texture1d")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_TEXTURE_1D;
|
|
} else if (UserTypeMatches(name, "texture1darray")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_TEXTURE_1D_ARRAY;
|
|
} else if (UserTypeMatches(name, "texture2d")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_TEXTURE_2D;
|
|
} else if (UserTypeMatches(name, "texture2darray")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_TEXTURE_2D_ARRAY;
|
|
} else if (UserTypeMatches(name, "texture2dms")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_TEXTURE_2DMS;
|
|
} else if (UserTypeMatches(name, "texture2dmsarray")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_TEXTURE_2DMS_ARRAY;
|
|
} else if (UserTypeMatches(name, "texture3d")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_TEXTURE_3D;
|
|
} else if (UserTypeMatches(name, "texturebuffer")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_TEXTURE_BUFFER;
|
|
} else if (UserTypeMatches(name, "texturecube")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_TEXTURE_CUBE;
|
|
} else if (UserTypeMatches(name, "texturecubearray")) {
|
|
p_target_decorations->user_type = SPV_REFLECT_USER_TYPE_TEXTURE_CUBE_ARRAY;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (spec_constant_count > 0) {
|
|
p_module->spec_constants = (SpvReflectSpecializationConstant*)calloc(spec_constant_count, sizeof(*p_module->spec_constants));
|
|
if (IsNull(p_module->spec_constants)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
}
|
|
for (uint32_t i = 0; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
// -- GODOT begin --
|
|
const uint32_t count = p_module->spec_constant_count;
|
|
switch(p_node->op) {
|
|
default: continue;
|
|
case SpvOpSpecConstantTrue: {
|
|
p_module->spec_constants[count].constant_type = SPV_REFLECT_SPECIALIZATION_CONSTANT_BOOL;
|
|
p_module->spec_constants[count].default_value.int_bool_value = 1;
|
|
} break;
|
|
case SpvOpSpecConstantFalse: {
|
|
p_module->spec_constants[count].constant_type = SPV_REFLECT_SPECIALIZATION_CONSTANT_BOOL;
|
|
p_module->spec_constants[count].default_value.int_bool_value = 0;
|
|
} break;
|
|
case SpvOpSpecConstant: {
|
|
SpvReflectResult result = SPV_REFLECT_RESULT_SUCCESS;
|
|
uint32_t element_type_id = (uint32_t)INVALID_VALUE;
|
|
uint32_t default_value = 0;
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, element_type_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 3, default_value);
|
|
|
|
SpvReflectPrvNode* p_next_node = FindNode(p_parser, element_type_id);
|
|
|
|
if (p_next_node->op == SpvOpTypeInt) {
|
|
p_module->spec_constants[count].constant_type = SPV_REFLECT_SPECIALIZATION_CONSTANT_INT;
|
|
} else if (p_next_node->op == SpvOpTypeFloat) {
|
|
p_module->spec_constants[count].constant_type = SPV_REFLECT_SPECIALIZATION_CONSTANT_FLOAT;
|
|
} else {
|
|
return SPV_REFLECT_RESULT_ERROR_PARSE_FAILED;
|
|
}
|
|
|
|
p_module->spec_constants[count].default_value.int_bool_value = default_value; //bits are the same for int and float
|
|
} break;
|
|
}
|
|
|
|
p_module->spec_constants[count].name = p_node->name;
|
|
p_module->spec_constants[count].constant_id = p_node->decorations.spec_id;
|
|
p_module->spec_constants[count].spirv_id = p_node->result_id;
|
|
|
|
p_module->spec_constant_count++;
|
|
// -- GODOT end --
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult EnumerateAllUniforms(SpvReflectShaderModule* p_module, size_t* p_uniform_count, uint32_t** pp_uniforms) {
|
|
*p_uniform_count = p_module->descriptor_binding_count;
|
|
if (*p_uniform_count == 0) {
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
*pp_uniforms = (uint32_t*)calloc(*p_uniform_count, sizeof(**pp_uniforms));
|
|
|
|
if (IsNull(*pp_uniforms)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
|
|
for (size_t i = 0; i < *p_uniform_count; ++i) {
|
|
(*pp_uniforms)[i] = p_module->descriptor_bindings[i].spirv_id;
|
|
}
|
|
qsort(*pp_uniforms, *p_uniform_count, sizeof(**pp_uniforms), SortCompareUint32);
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseType(SpvReflectPrvParser* p_parser, SpvReflectPrvNode* p_node,
|
|
SpvReflectPrvDecorations* p_struct_member_decorations, SpvReflectShaderModule* p_module,
|
|
SpvReflectTypeDescription* p_type) {
|
|
SpvReflectResult result = SPV_REFLECT_RESULT_SUCCESS;
|
|
|
|
if (p_node->member_count > 0) {
|
|
p_type->struct_type_description = FindType(p_module, p_node->result_id);
|
|
p_type->member_count = p_node->member_count;
|
|
p_type->members = (SpvReflectTypeDescription*)calloc(p_type->member_count, sizeof(*(p_type->members)));
|
|
if (IsNotNull(p_type->members)) {
|
|
// Mark all members types with an invalid state
|
|
for (size_t i = 0; i < p_type->members->member_count; ++i) {
|
|
SpvReflectTypeDescription* p_member_type = &(p_type->members[i]);
|
|
p_member_type->id = (uint32_t)INVALID_VALUE;
|
|
p_member_type->op = (SpvOp)INVALID_VALUE;
|
|
p_member_type->storage_class = (SpvStorageClass)INVALID_VALUE;
|
|
}
|
|
} else {
|
|
result = SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
}
|
|
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
// Since the parse descends on type information, these will get overwritten
|
|
// if not guarded against assignment. Only assign if the id is invalid.
|
|
if (p_type->id == INVALID_VALUE) {
|
|
p_type->id = p_node->result_id;
|
|
p_type->op = p_node->op;
|
|
p_type->decoration_flags = 0;
|
|
}
|
|
// Top level types need to pick up decorations from all types below it.
|
|
// Issue and fix here: https://github.com/chaoticbob/SPIRV-Reflect/issues/64
|
|
p_type->decoration_flags = ApplyDecorations(&p_node->decorations);
|
|
|
|
switch (p_node->op) {
|
|
default:
|
|
break;
|
|
case SpvOpTypeVoid:
|
|
p_type->type_flags |= SPV_REFLECT_TYPE_FLAG_VOID;
|
|
break;
|
|
|
|
case SpvOpTypeBool:
|
|
p_type->type_flags |= SPV_REFLECT_TYPE_FLAG_BOOL;
|
|
break;
|
|
|
|
case SpvOpTypeInt: {
|
|
p_type->type_flags |= SPV_REFLECT_TYPE_FLAG_INT;
|
|
IF_READU32(result, p_parser, p_node->word_offset + 2, p_type->traits.numeric.scalar.width);
|
|
IF_READU32(result, p_parser, p_node->word_offset + 3, p_type->traits.numeric.scalar.signedness);
|
|
} break;
|
|
|
|
case SpvOpTypeFloat: {
|
|
p_type->type_flags |= SPV_REFLECT_TYPE_FLAG_FLOAT;
|
|
IF_READU32(result, p_parser, p_node->word_offset + 2, p_type->traits.numeric.scalar.width);
|
|
} break;
|
|
|
|
case SpvOpTypeVector: {
|
|
p_type->type_flags |= SPV_REFLECT_TYPE_FLAG_VECTOR;
|
|
uint32_t component_type_id = (uint32_t)INVALID_VALUE;
|
|
IF_READU32(result, p_parser, p_node->word_offset + 2, component_type_id);
|
|
IF_READU32(result, p_parser, p_node->word_offset + 3, p_type->traits.numeric.vector.component_count);
|
|
// Parse component type
|
|
SpvReflectPrvNode* p_next_node = FindNode(p_parser, component_type_id);
|
|
if (IsNotNull(p_next_node)) {
|
|
result = ParseType(p_parser, p_next_node, NULL, p_module, p_type);
|
|
} else {
|
|
result = SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
SPV_REFLECT_ASSERT(false);
|
|
}
|
|
} break;
|
|
|
|
case SpvOpTypeMatrix: {
|
|
p_type->type_flags |= SPV_REFLECT_TYPE_FLAG_MATRIX;
|
|
uint32_t column_type_id = (uint32_t)INVALID_VALUE;
|
|
IF_READU32(result, p_parser, p_node->word_offset + 2, column_type_id);
|
|
IF_READU32(result, p_parser, p_node->word_offset + 3, p_type->traits.numeric.matrix.column_count);
|
|
SpvReflectPrvNode* p_next_node = FindNode(p_parser, column_type_id);
|
|
if (IsNotNull(p_next_node)) {
|
|
result = ParseType(p_parser, p_next_node, NULL, p_module, p_type);
|
|
} else {
|
|
result = SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
SPV_REFLECT_ASSERT(false);
|
|
}
|
|
p_type->traits.numeric.matrix.row_count = p_type->traits.numeric.vector.component_count;
|
|
p_type->traits.numeric.matrix.stride = p_node->decorations.matrix_stride;
|
|
// NOTE: Matrix stride is decorated using OpMemberDecoreate - not OpDecoreate.
|
|
if (IsNotNull(p_struct_member_decorations)) {
|
|
p_type->traits.numeric.matrix.stride = p_struct_member_decorations->matrix_stride;
|
|
}
|
|
} break;
|
|
|
|
case SpvOpTypeImage: {
|
|
p_type->type_flags |= SPV_REFLECT_TYPE_FLAG_EXTERNAL_IMAGE;
|
|
uint32_t sampled_type_id = (uint32_t)INVALID_VALUE;
|
|
IF_READU32(result, p_parser, p_node->word_offset + 2, sampled_type_id);
|
|
SpvReflectPrvNode* p_next_node = FindNode(p_parser, sampled_type_id);
|
|
if (IsNotNull(p_next_node)) {
|
|
result = ParseType(p_parser, p_next_node, NULL, p_module, p_type);
|
|
} else {
|
|
result = SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
IF_READU32_CAST(result, p_parser, p_node->word_offset + 3, SpvDim, p_type->traits.image.dim);
|
|
IF_READU32(result, p_parser, p_node->word_offset + 4, p_type->traits.image.depth);
|
|
IF_READU32(result, p_parser, p_node->word_offset + 5, p_type->traits.image.arrayed);
|
|
IF_READU32(result, p_parser, p_node->word_offset + 6, p_type->traits.image.ms);
|
|
IF_READU32(result, p_parser, p_node->word_offset + 7, p_type->traits.image.sampled);
|
|
IF_READU32_CAST(result, p_parser, p_node->word_offset + 8, SpvImageFormat, p_type->traits.image.image_format);
|
|
} break;
|
|
|
|
case SpvOpTypeSampler: {
|
|
p_type->type_flags |= SPV_REFLECT_TYPE_FLAG_EXTERNAL_SAMPLER;
|
|
} break;
|
|
|
|
case SpvOpTypeSampledImage: {
|
|
p_type->type_flags |= SPV_REFLECT_TYPE_FLAG_EXTERNAL_SAMPLED_IMAGE;
|
|
uint32_t image_type_id = (uint32_t)INVALID_VALUE;
|
|
IF_READU32(result, p_parser, p_node->word_offset + 2, image_type_id);
|
|
SpvReflectPrvNode* p_next_node = FindNode(p_parser, image_type_id);
|
|
if (IsNotNull(p_next_node)) {
|
|
result = ParseType(p_parser, p_next_node, NULL, p_module, p_type);
|
|
} else {
|
|
result = SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
SPV_REFLECT_ASSERT(false);
|
|
}
|
|
} break;
|
|
|
|
case SpvOpTypeArray: {
|
|
p_type->type_flags |= SPV_REFLECT_TYPE_FLAG_ARRAY;
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
uint32_t element_type_id = (uint32_t)INVALID_VALUE;
|
|
uint32_t length_id = (uint32_t)INVALID_VALUE;
|
|
IF_READU32(result, p_parser, p_node->word_offset + 2, element_type_id);
|
|
IF_READU32(result, p_parser, p_node->word_offset + 3, length_id);
|
|
// NOTE: Array stride is decorated using OpDecorate instead of
|
|
// OpMemberDecorate, even if the array is apart of a struct.
|
|
p_type->traits.array.stride = p_node->decorations.array_stride;
|
|
// Get length for current dimension
|
|
SpvReflectPrvNode* p_length_node = FindNode(p_parser, length_id);
|
|
if (IsNotNull(p_length_node)) {
|
|
uint32_t dim_index = p_type->traits.array.dims_count;
|
|
uint32_t length = 0;
|
|
IF_READU32(result, p_parser, p_length_node->word_offset + 3, length);
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
p_type->traits.array.dims[dim_index] = length;
|
|
p_type->traits.array.dims_count += 1;
|
|
p_type->traits.array.spec_constant_op_ids[dim_index] =
|
|
IsSpecConstant(p_length_node) ? p_length_node->decorations.spec_id : (uint32_t)INVALID_VALUE;
|
|
} else {
|
|
result = SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
SPV_REFLECT_ASSERT(false);
|
|
}
|
|
// Parse next dimension or element type
|
|
SpvReflectPrvNode* p_next_node = FindNode(p_parser, element_type_id);
|
|
if (IsNotNull(p_next_node)) {
|
|
result = ParseType(p_parser, p_next_node, NULL, p_module, p_type);
|
|
}
|
|
} else {
|
|
result = SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
SPV_REFLECT_ASSERT(false);
|
|
}
|
|
}
|
|
} break;
|
|
|
|
case SpvOpTypeRuntimeArray: {
|
|
p_type->type_flags |= SPV_REFLECT_TYPE_FLAG_ARRAY;
|
|
uint32_t element_type_id = (uint32_t)INVALID_VALUE;
|
|
IF_READU32(result, p_parser, p_node->word_offset + 2, element_type_id);
|
|
p_type->traits.array.stride = p_node->decorations.array_stride;
|
|
uint32_t dim_index = p_type->traits.array.dims_count;
|
|
p_type->traits.array.dims[dim_index] = (uint32_t)SPV_REFLECT_ARRAY_DIM_RUNTIME;
|
|
p_type->traits.array.spec_constant_op_ids[dim_index] = (uint32_t)INVALID_VALUE;
|
|
p_type->traits.array.dims_count += 1;
|
|
// Parse next dimension or element type
|
|
SpvReflectPrvNode* p_next_node = FindNode(p_parser, element_type_id);
|
|
if (IsNotNull(p_next_node)) {
|
|
result = ParseType(p_parser, p_next_node, NULL, p_module, p_type);
|
|
} else {
|
|
result = SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
SPV_REFLECT_ASSERT(false);
|
|
}
|
|
} break;
|
|
|
|
case SpvOpTypeStruct: {
|
|
p_type->type_flags |= SPV_REFLECT_TYPE_FLAG_STRUCT;
|
|
p_type->type_flags |= SPV_REFLECT_TYPE_FLAG_EXTERNAL_BLOCK;
|
|
uint32_t word_index = 2;
|
|
uint32_t member_index = 0;
|
|
for (; word_index < p_node->word_count; ++word_index, ++member_index) {
|
|
uint32_t member_id = (uint32_t)INVALID_VALUE;
|
|
IF_READU32(result, p_parser, p_node->word_offset + word_index, member_id);
|
|
// Find member node
|
|
SpvReflectPrvNode* p_member_node = FindNode(p_parser, member_id);
|
|
if (IsNull(p_member_node)) {
|
|
result = SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
SPV_REFLECT_ASSERT(false);
|
|
break;
|
|
}
|
|
|
|
// Member decorations
|
|
SpvReflectPrvDecorations* p_member_decorations = &p_node->member_decorations[member_index];
|
|
|
|
assert(member_index < p_type->member_count);
|
|
// Parse member type
|
|
SpvReflectTypeDescription* p_member_type = &(p_type->members[member_index]);
|
|
p_member_type->id = member_id;
|
|
p_member_type->op = p_member_node->op;
|
|
result = ParseType(p_parser, p_member_node, p_member_decorations, p_module, p_member_type);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
break;
|
|
}
|
|
// This looks wrong
|
|
// p_member_type->type_name = p_member_node->name;
|
|
p_member_type->struct_member_name = p_node->member_names[member_index];
|
|
}
|
|
} break;
|
|
|
|
case SpvOpTypeOpaque:
|
|
break;
|
|
|
|
case SpvOpTypePointer: {
|
|
p_type->type_flags |= SPV_REFLECT_TYPE_FLAG_REF;
|
|
IF_READU32_CAST(result, p_parser, p_node->word_offset + 2, SpvStorageClass, p_type->storage_class);
|
|
|
|
bool found_recursion = false;
|
|
if (p_type->storage_class == SpvStorageClassPhysicalStorageBuffer) {
|
|
// Need to make sure we haven't started an infinite recursive loop
|
|
for (uint32_t i = 0; i < p_parser->physical_pointer_count; i++) {
|
|
if (p_type->id == p_parser->physical_pointer_check[i]->id) {
|
|
found_recursion = true;
|
|
memcpy(p_type, p_parser->physical_pointer_check[i], sizeof(SpvReflectTypeDescription));
|
|
p_type->copied = 1;
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
}
|
|
if (!found_recursion) {
|
|
p_parser->physical_pointer_struct_count++;
|
|
p_parser->physical_pointer_check[p_parser->physical_pointer_count] = p_type;
|
|
p_parser->physical_pointer_count++;
|
|
if (p_parser->physical_pointer_count >= MAX_RECURSIVE_PHYSICAL_POINTER_CHECK) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_MAX_RECURSIVE_EXCEEDED;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Parse type
|
|
SpvReflectPrvNode* p_next_node = FindNode(p_parser, p_node->type_id);
|
|
if (IsNull(p_next_node)) {
|
|
result = SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
SPV_REFLECT_ASSERT(false);
|
|
} else if (!found_recursion) {
|
|
if (p_next_node->op == SpvOpTypeStruct) {
|
|
p_type->struct_type_description = FindType(p_module, p_next_node->result_id);
|
|
}
|
|
|
|
result = ParseType(p_parser, p_next_node, NULL, p_module, p_type);
|
|
}
|
|
} break;
|
|
|
|
case SpvOpTypeAccelerationStructureKHR: {
|
|
p_type->type_flags |= SPV_REFLECT_TYPE_FLAG_EXTERNAL_ACCELERATION_STRUCTURE;
|
|
} break;
|
|
}
|
|
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
// Names get assigned on the way down. Guard against names
|
|
// get overwritten on the way up.
|
|
if (IsNull(p_type->type_name)) {
|
|
p_type->type_name = p_node->name;
|
|
}
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
static SpvReflectResult ParseTypes(SpvReflectPrvParser* p_parser, SpvReflectShaderModule* p_module) {
|
|
if (p_parser->type_count == 0) {
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
p_module->_internal->type_description_count = p_parser->type_count;
|
|
p_module->_internal->type_descriptions = (SpvReflectTypeDescription*)calloc(p_module->_internal->type_description_count,
|
|
sizeof(*(p_module->_internal->type_descriptions)));
|
|
if (IsNull(p_module->_internal->type_descriptions)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
|
|
// Mark all types with an invalid state
|
|
for (size_t i = 0; i < p_module->_internal->type_description_count; ++i) {
|
|
SpvReflectTypeDescription* p_type = &(p_module->_internal->type_descriptions[i]);
|
|
p_type->id = (uint32_t)INVALID_VALUE;
|
|
p_type->op = (SpvOp)INVALID_VALUE;
|
|
p_type->storage_class = (SpvStorageClass)INVALID_VALUE;
|
|
}
|
|
|
|
size_t type_index = 0;
|
|
for (size_t i = 0; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
if (!p_node->is_type) {
|
|
continue;
|
|
}
|
|
|
|
SpvReflectTypeDescription* p_type = &(p_module->_internal->type_descriptions[type_index]);
|
|
p_parser->physical_pointer_count = 0;
|
|
SpvReflectResult result = ParseType(p_parser, p_node, NULL, p_module, p_type);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
++type_index;
|
|
}
|
|
|
|
// allocate now and fill in when parsing struct variable later
|
|
if (p_parser->physical_pointer_struct_count > 0) {
|
|
p_parser->physical_pointer_structs = (SpvReflectPrvPhysicalPointerStruct*)calloc(p_parser->physical_pointer_struct_count,
|
|
sizeof(*(p_parser->physical_pointer_structs)));
|
|
if (IsNull(p_parser->physical_pointer_structs)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
}
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseCapabilities(SpvReflectPrvParser* p_parser, SpvReflectShaderModule* p_module) {
|
|
if (p_parser->capability_count == 0) {
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
p_module->capability_count = p_parser->capability_count;
|
|
p_module->capabilities = (SpvReflectCapability*)calloc(p_module->capability_count, sizeof(*(p_module->capabilities)));
|
|
if (IsNull(p_module->capabilities)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
|
|
// Mark all types with an invalid state
|
|
for (size_t i = 0; i < p_module->capability_count; ++i) {
|
|
SpvReflectCapability* p_cap = &(p_module->capabilities[i]);
|
|
p_cap->value = SpvCapabilityMax;
|
|
p_cap->word_offset = (uint32_t)INVALID_VALUE;
|
|
}
|
|
|
|
size_t capability_index = 0;
|
|
for (size_t i = 0; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
if (SpvOpCapability != p_node->op) {
|
|
continue;
|
|
}
|
|
|
|
SpvReflectCapability* p_cap = &(p_module->capabilities[capability_index]);
|
|
p_cap->value = p_node->capability;
|
|
p_cap->word_offset = p_node->word_offset + 1;
|
|
++capability_index;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static int SortCompareDescriptorBinding(const void* a, const void* b) {
|
|
const SpvReflectDescriptorBinding* p_elem_a = (const SpvReflectDescriptorBinding*)a;
|
|
const SpvReflectDescriptorBinding* p_elem_b = (const SpvReflectDescriptorBinding*)b;
|
|
int value = (int)(p_elem_a->binding) - (int)(p_elem_b->binding);
|
|
if (value == 0) {
|
|
// use spirv-id as a tiebreaker to ensure a stable ordering, as they're guaranteed
|
|
// unique.
|
|
assert(p_elem_a->spirv_id != p_elem_b->spirv_id);
|
|
value = (int)(p_elem_a->spirv_id) - (int)(p_elem_b->spirv_id);
|
|
}
|
|
return value;
|
|
}
|
|
|
|
static SpvReflectResult ParseDescriptorBindings(SpvReflectPrvParser* p_parser, SpvReflectShaderModule* p_module) {
|
|
p_module->descriptor_binding_count = 0;
|
|
for (size_t i = 0; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
if ((p_node->op != SpvOpVariable) ||
|
|
((p_node->storage_class != SpvStorageClassUniform) && (p_node->storage_class != SpvStorageClassStorageBuffer) &&
|
|
(p_node->storage_class != SpvStorageClassUniformConstant))) {
|
|
continue;
|
|
}
|
|
if ((p_node->decorations.set.value == INVALID_VALUE) || (p_node->decorations.binding.value == INVALID_VALUE)) {
|
|
continue;
|
|
}
|
|
|
|
p_module->descriptor_binding_count += 1;
|
|
}
|
|
|
|
if (p_module->descriptor_binding_count == 0) {
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
p_module->descriptor_bindings =
|
|
(SpvReflectDescriptorBinding*)calloc(p_module->descriptor_binding_count, sizeof(*(p_module->descriptor_bindings)));
|
|
if (IsNull(p_module->descriptor_bindings)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
|
|
// Mark all types with an invalid state
|
|
for (uint32_t descriptor_index = 0; descriptor_index < p_module->descriptor_binding_count; ++descriptor_index) {
|
|
SpvReflectDescriptorBinding* p_descriptor = &(p_module->descriptor_bindings[descriptor_index]);
|
|
p_descriptor->binding = (uint32_t)INVALID_VALUE;
|
|
p_descriptor->input_attachment_index = (uint32_t)INVALID_VALUE;
|
|
p_descriptor->set = (uint32_t)INVALID_VALUE;
|
|
p_descriptor->descriptor_type = (SpvReflectDescriptorType)INVALID_VALUE;
|
|
p_descriptor->uav_counter_id = (uint32_t)INVALID_VALUE;
|
|
}
|
|
|
|
size_t descriptor_index = 0;
|
|
for (size_t i = 0; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
if ((p_node->op != SpvOpVariable) ||
|
|
((p_node->storage_class != SpvStorageClassUniform) && (p_node->storage_class != SpvStorageClassStorageBuffer) &&
|
|
(p_node->storage_class != SpvStorageClassUniformConstant))) {
|
|
continue;
|
|
}
|
|
if ((p_node->decorations.set.value == INVALID_VALUE) || (p_node->decorations.binding.value == INVALID_VALUE)) {
|
|
continue;
|
|
}
|
|
|
|
SpvReflectTypeDescription* p_type = FindType(p_module, p_node->type_id);
|
|
if (IsNull(p_type)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
// If the type is a pointer, resolve it. We need to retain the storage class
|
|
// from the pointer so that we can use it to deduce deescriptor types.
|
|
SpvStorageClass pointer_storage_class = SpvStorageClassMax;
|
|
if (p_type->op == SpvOpTypePointer) {
|
|
pointer_storage_class = p_type->storage_class;
|
|
// Find the type's node
|
|
SpvReflectPrvNode* p_type_node = FindNode(p_parser, p_type->id);
|
|
if (IsNull(p_type_node)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
// Should be the resolved type
|
|
p_type = FindType(p_module, p_type_node->type_id);
|
|
if (IsNull(p_type)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
}
|
|
|
|
SpvReflectDescriptorBinding* p_descriptor = &p_module->descriptor_bindings[descriptor_index];
|
|
p_descriptor->spirv_id = p_node->result_id;
|
|
p_descriptor->name = p_node->name;
|
|
p_descriptor->binding = p_node->decorations.binding.value;
|
|
p_descriptor->input_attachment_index = p_node->decorations.input_attachment_index.value;
|
|
p_descriptor->set = p_node->decorations.set.value;
|
|
p_descriptor->count = 1;
|
|
p_descriptor->uav_counter_id = p_node->decorations.uav_counter_buffer.value;
|
|
p_descriptor->type_description = p_type;
|
|
p_descriptor->decoration_flags = ApplyDecorations(&p_node->decorations);
|
|
p_descriptor->user_type = p_node->decorations.user_type;
|
|
|
|
// Flags like non-writable and non-readable are found as member decorations only.
|
|
// If all members have one of those decorations set, promote the decoration up
|
|
// to the whole descriptor.
|
|
const SpvReflectPrvNode* p_type_node = FindNode(p_parser, p_type->id);
|
|
if (IsNotNull(p_type_node) && p_type_node->member_count) {
|
|
SpvReflectPrvDecorations common_flags = p_type_node->member_decorations[0];
|
|
|
|
for (uint32_t m = 1; m < p_type_node->member_count; ++m) {
|
|
common_flags.is_relaxed_precision &= p_type_node->member_decorations[m].is_relaxed_precision;
|
|
common_flags.is_block &= p_type_node->member_decorations[m].is_block;
|
|
common_flags.is_buffer_block &= p_type_node->member_decorations[m].is_buffer_block;
|
|
common_flags.is_row_major &= p_type_node->member_decorations[m].is_row_major;
|
|
common_flags.is_column_major &= p_type_node->member_decorations[m].is_column_major;
|
|
common_flags.is_built_in &= p_type_node->member_decorations[m].is_built_in;
|
|
common_flags.is_noperspective &= p_type_node->member_decorations[m].is_noperspective;
|
|
common_flags.is_flat &= p_type_node->member_decorations[m].is_flat;
|
|
common_flags.is_non_writable &= p_type_node->member_decorations[m].is_non_writable;
|
|
common_flags.is_non_readable &= p_type_node->member_decorations[m].is_non_readable;
|
|
common_flags.is_patch &= p_type_node->member_decorations[m].is_patch;
|
|
common_flags.is_per_vertex &= p_type_node->member_decorations[m].is_per_vertex;
|
|
common_flags.is_per_task &= p_type_node->member_decorations[m].is_per_task;
|
|
common_flags.is_weight_texture &= p_type_node->member_decorations[m].is_weight_texture;
|
|
common_flags.is_block_match_texture &= p_type_node->member_decorations[m].is_block_match_texture;
|
|
}
|
|
|
|
p_descriptor->decoration_flags |= ApplyDecorations(&common_flags);
|
|
}
|
|
|
|
// If this is in the StorageBuffer storage class, it's for sure a storage
|
|
// buffer descriptor. We need to handle this case earlier because in SPIR-V
|
|
// there are two ways to indicate a storage buffer:
|
|
// 1) Uniform storage class + BufferBlock decoration, or
|
|
// 2) StorageBuffer storage class + Buffer decoration.
|
|
// The 1) way is deprecated since SPIR-V v1.3. But the Buffer decoration is
|
|
// also used together with Uniform storage class to mean uniform buffer..
|
|
// We'll handle the pre-v1.3 cases in ParseDescriptorType().
|
|
if (pointer_storage_class == SpvStorageClassStorageBuffer) {
|
|
p_descriptor->descriptor_type = SPV_REFLECT_DESCRIPTOR_TYPE_STORAGE_BUFFER;
|
|
}
|
|
|
|
// Copy image traits
|
|
if ((p_type->type_flags & SPV_REFLECT_TYPE_FLAG_EXTERNAL_MASK) == SPV_REFLECT_TYPE_FLAG_EXTERNAL_IMAGE) {
|
|
memcpy(&p_descriptor->image, &p_type->traits.image, sizeof(p_descriptor->image));
|
|
}
|
|
|
|
// This is a workaround for: https://github.com/KhronosGroup/glslang/issues/1096
|
|
{
|
|
const uint32_t resource_mask = SPV_REFLECT_TYPE_FLAG_EXTERNAL_SAMPLED_IMAGE | SPV_REFLECT_TYPE_FLAG_EXTERNAL_IMAGE;
|
|
if ((p_type->type_flags & resource_mask) == resource_mask) {
|
|
memcpy(&p_descriptor->image, &p_type->traits.image, sizeof(p_descriptor->image));
|
|
}
|
|
}
|
|
|
|
// Copy array traits
|
|
if (p_type->traits.array.dims_count > 0) {
|
|
p_descriptor->array.dims_count = p_type->traits.array.dims_count;
|
|
for (uint32_t dim_index = 0; dim_index < p_type->traits.array.dims_count; ++dim_index) {
|
|
uint32_t dim_value = p_type->traits.array.dims[dim_index];
|
|
p_descriptor->array.dims[dim_index] = dim_value;
|
|
p_descriptor->count *= dim_value;
|
|
}
|
|
}
|
|
|
|
// Count
|
|
|
|
p_descriptor->word_offset.binding = p_node->decorations.binding.word_offset;
|
|
p_descriptor->word_offset.set = p_node->decorations.set.word_offset;
|
|
|
|
++descriptor_index;
|
|
}
|
|
|
|
if (p_module->descriptor_binding_count > 0) {
|
|
qsort(p_module->descriptor_bindings, p_module->descriptor_binding_count, sizeof(*(p_module->descriptor_bindings)),
|
|
SortCompareDescriptorBinding);
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseDescriptorType(SpvReflectShaderModule* p_module) {
|
|
if (p_module->descriptor_binding_count == 0) {
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
for (uint32_t descriptor_index = 0; descriptor_index < p_module->descriptor_binding_count; ++descriptor_index) {
|
|
SpvReflectDescriptorBinding* p_descriptor = &(p_module->descriptor_bindings[descriptor_index]);
|
|
SpvReflectTypeDescription* p_type = p_descriptor->type_description;
|
|
|
|
if ((int)p_descriptor->descriptor_type == (int)INVALID_VALUE) {
|
|
switch (p_type->type_flags & SPV_REFLECT_TYPE_FLAG_EXTERNAL_MASK) {
|
|
default:
|
|
assert(false && "unknown type flag");
|
|
break;
|
|
|
|
case SPV_REFLECT_TYPE_FLAG_EXTERNAL_IMAGE: {
|
|
if (p_descriptor->image.dim == SpvDimBuffer) {
|
|
switch (p_descriptor->image.sampled) {
|
|
default:
|
|
assert(false && "unknown texel buffer sampled value");
|
|
break;
|
|
case IMAGE_SAMPLED:
|
|
p_descriptor->descriptor_type = SPV_REFLECT_DESCRIPTOR_TYPE_UNIFORM_TEXEL_BUFFER;
|
|
break;
|
|
case IMAGE_STORAGE:
|
|
p_descriptor->descriptor_type = SPV_REFLECT_DESCRIPTOR_TYPE_STORAGE_TEXEL_BUFFER;
|
|
break;
|
|
}
|
|
} else if (p_descriptor->image.dim == SpvDimSubpassData) {
|
|
p_descriptor->descriptor_type = SPV_REFLECT_DESCRIPTOR_TYPE_INPUT_ATTACHMENT;
|
|
} else {
|
|
switch (p_descriptor->image.sampled) {
|
|
default:
|
|
assert(false && "unknown image sampled value");
|
|
break;
|
|
case IMAGE_SAMPLED:
|
|
p_descriptor->descriptor_type = SPV_REFLECT_DESCRIPTOR_TYPE_SAMPLED_IMAGE;
|
|
break;
|
|
case IMAGE_STORAGE:
|
|
p_descriptor->descriptor_type = SPV_REFLECT_DESCRIPTOR_TYPE_STORAGE_IMAGE;
|
|
break;
|
|
}
|
|
}
|
|
} break;
|
|
|
|
case SPV_REFLECT_TYPE_FLAG_EXTERNAL_SAMPLER: {
|
|
p_descriptor->descriptor_type = SPV_REFLECT_DESCRIPTOR_TYPE_SAMPLER;
|
|
} break;
|
|
|
|
case (SPV_REFLECT_TYPE_FLAG_EXTERNAL_SAMPLED_IMAGE | SPV_REFLECT_TYPE_FLAG_EXTERNAL_IMAGE): {
|
|
// This is a workaround for: https://github.com/KhronosGroup/glslang/issues/1096
|
|
if (p_descriptor->image.dim == SpvDimBuffer) {
|
|
switch (p_descriptor->image.sampled) {
|
|
default:
|
|
assert(false && "unknown texel buffer sampled value");
|
|
break;
|
|
case IMAGE_SAMPLED:
|
|
p_descriptor->descriptor_type = SPV_REFLECT_DESCRIPTOR_TYPE_UNIFORM_TEXEL_BUFFER;
|
|
break;
|
|
case IMAGE_STORAGE:
|
|
p_descriptor->descriptor_type = SPV_REFLECT_DESCRIPTOR_TYPE_STORAGE_TEXEL_BUFFER;
|
|
break;
|
|
}
|
|
} else {
|
|
p_descriptor->descriptor_type = SPV_REFLECT_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER;
|
|
}
|
|
} break;
|
|
|
|
case SPV_REFLECT_TYPE_FLAG_EXTERNAL_BLOCK: {
|
|
if (p_type->decoration_flags & SPV_REFLECT_DECORATION_BLOCK) {
|
|
p_descriptor->descriptor_type = SPV_REFLECT_DESCRIPTOR_TYPE_UNIFORM_BUFFER;
|
|
} else if (p_type->decoration_flags & SPV_REFLECT_DECORATION_BUFFER_BLOCK) {
|
|
p_descriptor->descriptor_type = SPV_REFLECT_DESCRIPTOR_TYPE_STORAGE_BUFFER;
|
|
} else {
|
|
assert(false && "unknown struct");
|
|
}
|
|
} break;
|
|
|
|
case SPV_REFLECT_TYPE_FLAG_EXTERNAL_ACCELERATION_STRUCTURE: {
|
|
p_descriptor->descriptor_type = SPV_REFLECT_DESCRIPTOR_TYPE_ACCELERATION_STRUCTURE_KHR;
|
|
} break;
|
|
}
|
|
}
|
|
|
|
switch (p_descriptor->descriptor_type) {
|
|
case SPV_REFLECT_DESCRIPTOR_TYPE_SAMPLER:
|
|
p_descriptor->resource_type = SPV_REFLECT_RESOURCE_FLAG_SAMPLER;
|
|
break;
|
|
case SPV_REFLECT_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER:
|
|
p_descriptor->resource_type = (SpvReflectResourceType)(SPV_REFLECT_RESOURCE_FLAG_SAMPLER | SPV_REFLECT_RESOURCE_FLAG_SRV);
|
|
break;
|
|
case SPV_REFLECT_DESCRIPTOR_TYPE_SAMPLED_IMAGE:
|
|
p_descriptor->resource_type = SPV_REFLECT_RESOURCE_FLAG_SRV;
|
|
break;
|
|
case SPV_REFLECT_DESCRIPTOR_TYPE_STORAGE_IMAGE:
|
|
p_descriptor->resource_type = SPV_REFLECT_RESOURCE_FLAG_UAV;
|
|
break;
|
|
case SPV_REFLECT_DESCRIPTOR_TYPE_UNIFORM_TEXEL_BUFFER:
|
|
p_descriptor->resource_type = SPV_REFLECT_RESOURCE_FLAG_SRV;
|
|
break;
|
|
case SPV_REFLECT_DESCRIPTOR_TYPE_STORAGE_TEXEL_BUFFER:
|
|
p_descriptor->resource_type = SPV_REFLECT_RESOURCE_FLAG_UAV;
|
|
break;
|
|
case SPV_REFLECT_DESCRIPTOR_TYPE_UNIFORM_BUFFER:
|
|
p_descriptor->resource_type = SPV_REFLECT_RESOURCE_FLAG_CBV;
|
|
break;
|
|
case SPV_REFLECT_DESCRIPTOR_TYPE_UNIFORM_BUFFER_DYNAMIC:
|
|
p_descriptor->resource_type = SPV_REFLECT_RESOURCE_FLAG_CBV;
|
|
break;
|
|
case SPV_REFLECT_DESCRIPTOR_TYPE_STORAGE_BUFFER:
|
|
p_descriptor->resource_type = SPV_REFLECT_RESOURCE_FLAG_UAV;
|
|
break;
|
|
case SPV_REFLECT_DESCRIPTOR_TYPE_STORAGE_BUFFER_DYNAMIC:
|
|
p_descriptor->resource_type = SPV_REFLECT_RESOURCE_FLAG_UAV;
|
|
break;
|
|
case SPV_REFLECT_DESCRIPTOR_TYPE_INPUT_ATTACHMENT:
|
|
break;
|
|
case SPV_REFLECT_DESCRIPTOR_TYPE_ACCELERATION_STRUCTURE_KHR:
|
|
p_descriptor->resource_type = SPV_REFLECT_RESOURCE_FLAG_SRV;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseUAVCounterBindings(SpvReflectShaderModule* p_module) {
|
|
char name[MAX_NODE_NAME_LENGTH];
|
|
const char* k_count_tag = "@count";
|
|
|
|
for (uint32_t descriptor_index = 0; descriptor_index < p_module->descriptor_binding_count; ++descriptor_index) {
|
|
SpvReflectDescriptorBinding* p_descriptor = &(p_module->descriptor_bindings[descriptor_index]);
|
|
|
|
if (p_descriptor->descriptor_type != SPV_REFLECT_DESCRIPTOR_TYPE_STORAGE_BUFFER) {
|
|
continue;
|
|
}
|
|
|
|
SpvReflectDescriptorBinding* p_counter_descriptor = NULL;
|
|
// Use UAV counter buffer id if present...
|
|
if (p_descriptor->uav_counter_id != UINT32_MAX) {
|
|
for (uint32_t counter_descriptor_index = 0; counter_descriptor_index < p_module->descriptor_binding_count;
|
|
++counter_descriptor_index) {
|
|
SpvReflectDescriptorBinding* p_test_counter_descriptor = &(p_module->descriptor_bindings[counter_descriptor_index]);
|
|
if (p_test_counter_descriptor->descriptor_type != SPV_REFLECT_DESCRIPTOR_TYPE_STORAGE_BUFFER) {
|
|
continue;
|
|
}
|
|
if (p_descriptor->uav_counter_id == p_test_counter_descriptor->spirv_id) {
|
|
p_counter_descriptor = p_test_counter_descriptor;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
// ...otherwise use old @count convention.
|
|
else {
|
|
const size_t descriptor_name_length = p_descriptor->name ? strlen(p_descriptor->name) : 0;
|
|
|
|
memset(name, 0, MAX_NODE_NAME_LENGTH);
|
|
memcpy(name, p_descriptor->name, descriptor_name_length);
|
|
#if defined(_WIN32)
|
|
strcat_s(name, MAX_NODE_NAME_LENGTH, k_count_tag);
|
|
#else
|
|
strcat(name, k_count_tag);
|
|
#endif
|
|
|
|
for (uint32_t counter_descriptor_index = 0; counter_descriptor_index < p_module->descriptor_binding_count;
|
|
++counter_descriptor_index) {
|
|
SpvReflectDescriptorBinding* p_test_counter_descriptor = &(p_module->descriptor_bindings[counter_descriptor_index]);
|
|
if (p_test_counter_descriptor->descriptor_type != SPV_REFLECT_DESCRIPTOR_TYPE_STORAGE_BUFFER) {
|
|
continue;
|
|
}
|
|
if (p_test_counter_descriptor->name && strcmp(name, p_test_counter_descriptor->name) == 0) {
|
|
p_counter_descriptor = p_test_counter_descriptor;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (p_counter_descriptor != NULL) {
|
|
p_descriptor->uav_counter_binding = p_counter_descriptor;
|
|
}
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseDescriptorBlockVariable(SpvReflectPrvParser* p_parser, SpvReflectShaderModule* p_module,
|
|
SpvReflectTypeDescription* p_type, SpvReflectBlockVariable* p_var) {
|
|
bool has_non_writable = false;
|
|
|
|
if (IsNotNull(p_type->members) && (p_type->member_count > 0)) {
|
|
p_var->member_count = p_type->member_count;
|
|
p_var->members = (SpvReflectBlockVariable*)calloc(p_var->member_count, sizeof(*p_var->members));
|
|
if (IsNull(p_var->members)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
|
|
SpvReflectPrvNode* p_type_node = FindNode(p_parser, p_type->id);
|
|
if (IsNull(p_type_node)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
// Resolve to element type if current type is array or run time array
|
|
while (p_type_node->op == SpvOpTypeArray || p_type_node->op == SpvOpTypeRuntimeArray) {
|
|
if (p_type_node->op == SpvOpTypeArray) {
|
|
p_type_node = FindNode(p_parser, p_type_node->array_traits.element_type_id);
|
|
} else {
|
|
// Element type description
|
|
SpvReflectTypeDescription* p_type_temp = FindType(p_module, p_type_node->array_traits.element_type_id);
|
|
if (IsNull(p_type_temp)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
// Element type node
|
|
p_type_node = FindNode(p_parser, p_type_temp->id);
|
|
}
|
|
if (IsNull(p_type_node)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
}
|
|
|
|
// Parse members
|
|
for (uint32_t member_index = 0; member_index < p_type->member_count; ++member_index) {
|
|
SpvReflectTypeDescription* p_member_type = &p_type->members[member_index];
|
|
SpvReflectBlockVariable* p_member_var = &p_var->members[member_index];
|
|
// If pointer type, treat like reference and resolve to pointee type
|
|
SpvReflectTypeDescription* p_member_ptr_type = 0;
|
|
bool found_recursion = false;
|
|
|
|
if ((p_member_type->storage_class == SpvStorageClassPhysicalStorageBuffer) &&
|
|
(p_member_type->type_flags & SPV_REFLECT_TYPE_FLAG_REF)) {
|
|
// Remember the original type
|
|
p_member_ptr_type = p_member_type;
|
|
|
|
// strip array
|
|
if (p_member_type->op == SpvOpTypeArray) {
|
|
SpvReflectPrvNode* p_node = FindNode(p_parser, p_member_type->id);
|
|
if (p_node == NULL) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
uint32_t element_type_id = p_node->array_traits.element_type_id;
|
|
p_member_type = FindType(p_module, element_type_id);
|
|
if (p_member_type == NULL) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
}
|
|
|
|
// Need to make sure we haven't started an infinite recursive loop
|
|
for (uint32_t i = 0; i < p_parser->physical_pointer_count; i++) {
|
|
if (p_member_type->id == p_parser->physical_pointer_check[i]->id) {
|
|
found_recursion = true;
|
|
break; // still need to fill in p_member_type values
|
|
}
|
|
}
|
|
if (!found_recursion) {
|
|
uint32_t struct_id = FindType(p_module, p_member_type->id)->struct_type_description->id;
|
|
p_parser->physical_pointer_structs[p_parser->physical_pointer_struct_count].struct_id = struct_id;
|
|
p_parser->physical_pointer_structs[p_parser->physical_pointer_struct_count].p_var = p_member_var;
|
|
p_parser->physical_pointer_struct_count++;
|
|
|
|
p_parser->physical_pointer_check[p_parser->physical_pointer_count] = p_member_type;
|
|
p_parser->physical_pointer_count++;
|
|
if (p_parser->physical_pointer_count >= MAX_RECURSIVE_PHYSICAL_POINTER_CHECK) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_MAX_RECURSIVE_EXCEEDED;
|
|
}
|
|
}
|
|
|
|
SpvReflectPrvNode* p_member_type_node = FindNode(p_parser, p_member_type->id);
|
|
if (IsNull(p_member_type_node)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
// Should be the pointee type
|
|
p_member_type = FindType(p_module, p_member_type_node->type_id);
|
|
if (IsNull(p_member_type)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
}
|
|
bool is_struct = (p_member_type->type_flags & SPV_REFLECT_TYPE_FLAG_STRUCT) == SPV_REFLECT_TYPE_FLAG_STRUCT;
|
|
if (is_struct) {
|
|
if (!found_recursion) {
|
|
SpvReflectResult result = ParseDescriptorBlockVariable(p_parser, p_module, p_member_type, p_member_var);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
} else {
|
|
// if 2 member of structs are same PhysicalPointer type, copy the
|
|
// members values that aren't found skipping the recursion call
|
|
for (uint32_t i = 0; i < p_parser->physical_pointer_struct_count; i++) {
|
|
if (p_parser->physical_pointer_structs[i].struct_id == p_member_type->id) {
|
|
p_member_var->members = p_parser->physical_pointer_structs[i].p_var->members;
|
|
p_member_var->member_count = p_parser->physical_pointer_structs[i].p_var->member_count;
|
|
// Set here as it is the first time we need to walk down structs
|
|
p_member_var->flags |= SPV_REFLECT_VARIABLE_FLAGS_PHYSICAL_POINTER_COPY;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (p_type_node->storage_class == SpvStorageClassPhysicalStorageBuffer && !p_type_node->member_names) {
|
|
// TODO 212 - If a buffer ref has an array of itself, all members are null
|
|
continue;
|
|
}
|
|
|
|
p_member_var->name = p_type_node->member_names[member_index];
|
|
p_member_var->offset = p_type_node->member_decorations[member_index].offset.value;
|
|
p_member_var->decoration_flags = ApplyDecorations(&p_type_node->member_decorations[member_index]);
|
|
p_member_var->flags |= SPV_REFLECT_VARIABLE_FLAGS_UNUSED;
|
|
if (!has_non_writable && (p_member_var->decoration_flags & SPV_REFLECT_DECORATION_NON_WRITABLE)) {
|
|
has_non_writable = true;
|
|
}
|
|
ApplyNumericTraits(p_member_type, &p_member_var->numeric);
|
|
if (p_member_type->op == SpvOpTypeArray) {
|
|
ApplyArrayTraits(p_member_type, &p_member_var->array);
|
|
}
|
|
|
|
p_member_var->word_offset.offset = p_type_node->member_decorations[member_index].offset.word_offset;
|
|
p_member_var->type_description = p_member_ptr_type ? p_member_ptr_type : p_member_type;
|
|
}
|
|
}
|
|
|
|
p_var->name = p_type->type_name;
|
|
p_var->type_description = p_type;
|
|
if (has_non_writable) {
|
|
p_var->decoration_flags |= SPV_REFLECT_DECORATION_NON_WRITABLE;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static uint32_t GetPhysicalPointerStructSize(SpvReflectPrvParser* p_parser, uint32_t id) {
|
|
for (uint32_t i = 0; i < p_parser->physical_pointer_struct_count; i++) {
|
|
if (p_parser->physical_pointer_structs[i].struct_id == id) {
|
|
return p_parser->physical_pointer_structs[i].p_var->size;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static SpvReflectResult ParseDescriptorBlockVariableSizes(SpvReflectPrvParser* p_parser, SpvReflectShaderModule* p_module,
|
|
bool is_parent_root, bool is_parent_aos, bool is_parent_rta,
|
|
SpvReflectBlockVariable* p_var) {
|
|
if (p_var->member_count == 0) {
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
bool is_parent_ref = p_var->type_description->op == SpvOpTypePointer;
|
|
|
|
// Absolute offsets
|
|
for (uint32_t member_index = 0; member_index < p_var->member_count; ++member_index) {
|
|
SpvReflectBlockVariable* p_member_var = &p_var->members[member_index];
|
|
if (is_parent_root) {
|
|
p_member_var->absolute_offset = p_member_var->offset;
|
|
} else {
|
|
p_member_var->absolute_offset =
|
|
is_parent_aos ? 0 : (is_parent_ref ? p_member_var->offset : p_member_var->offset + p_var->absolute_offset);
|
|
}
|
|
}
|
|
|
|
// Size
|
|
for (uint32_t member_index = 0; member_index < p_var->member_count; ++member_index) {
|
|
SpvReflectBlockVariable* p_member_var = &p_var->members[member_index];
|
|
SpvReflectTypeDescription* p_member_type = p_member_var->type_description;
|
|
|
|
if (!p_member_type) {
|
|
// TODO 212 - If a buffer ref has an array of itself, all members are null
|
|
continue;
|
|
}
|
|
switch (p_member_type->op) {
|
|
case SpvOpTypeBool: {
|
|
p_member_var->size = SPIRV_WORD_SIZE;
|
|
} break;
|
|
|
|
case SpvOpTypeInt:
|
|
case SpvOpTypeFloat: {
|
|
p_member_var->size = p_member_type->traits.numeric.scalar.width / SPIRV_BYTE_WIDTH;
|
|
} break;
|
|
|
|
case SpvOpTypeVector: {
|
|
uint32_t size =
|
|
p_member_type->traits.numeric.vector.component_count * (p_member_type->traits.numeric.scalar.width / SPIRV_BYTE_WIDTH);
|
|
p_member_var->size = size;
|
|
} break;
|
|
|
|
case SpvOpTypeMatrix: {
|
|
if (p_member_var->decoration_flags & SPV_REFLECT_DECORATION_COLUMN_MAJOR) {
|
|
p_member_var->size = p_member_var->numeric.matrix.column_count * p_member_var->numeric.matrix.stride;
|
|
} else if (p_member_var->decoration_flags & SPV_REFLECT_DECORATION_ROW_MAJOR) {
|
|
p_member_var->size = p_member_var->numeric.matrix.row_count * p_member_var->numeric.matrix.stride;
|
|
}
|
|
} break;
|
|
|
|
case SpvOpTypeArray: {
|
|
// If array of structs, parse members first...
|
|
bool is_struct = (p_member_type->type_flags & SPV_REFLECT_TYPE_FLAG_STRUCT) == SPV_REFLECT_TYPE_FLAG_STRUCT;
|
|
if (is_struct) {
|
|
if (p_member_var->flags & SPV_REFLECT_VARIABLE_FLAGS_PHYSICAL_POINTER_COPY) {
|
|
p_member_var->size = GetPhysicalPointerStructSize(p_parser, p_member_type->id);
|
|
} else {
|
|
SpvReflectResult result =
|
|
ParseDescriptorBlockVariableSizes(p_parser, p_module, false, true, is_parent_rta, p_member_var);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
}
|
|
}
|
|
// ...then array
|
|
uint32_t element_count = (p_member_var->array.dims_count > 0 ? 1 : 0);
|
|
for (uint32_t i = 0; i < p_member_var->array.dims_count; ++i) {
|
|
// -- GODOT begin --
|
|
if (p_member_var->array.spec_constant_op_ids[i] != (uint32_t)INVALID_VALUE) {
|
|
// Force size to be reported as 0 to effectively disable buffer size validation, since
|
|
// the value is unreliable anyway as only valid for the default values of the SCs involved.
|
|
element_count = 0;
|
|
}
|
|
// -- GODOT end --
|
|
element_count *= p_member_var->array.dims[i];
|
|
}
|
|
p_member_var->size = element_count * p_member_var->array.stride;
|
|
} break;
|
|
|
|
case SpvOpTypeRuntimeArray: {
|
|
bool is_struct = (p_member_type->type_flags & SPV_REFLECT_TYPE_FLAG_STRUCT) == SPV_REFLECT_TYPE_FLAG_STRUCT;
|
|
if (is_struct) {
|
|
SpvReflectResult result = ParseDescriptorBlockVariableSizes(p_parser, p_module, false, true, true, p_member_var);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
}
|
|
} break;
|
|
|
|
case SpvOpTypePointer: {
|
|
// Reference. Get to underlying struct type.
|
|
SpvReflectPrvNode* p_member_type_node = FindNode(p_parser, p_member_type->id);
|
|
if (IsNull(p_member_type_node)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
// Get the pointee type
|
|
p_member_type = FindType(p_module, p_member_type_node->type_id);
|
|
if (IsNull(p_member_type)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
assert(p_member_type->op == SpvOpTypeStruct);
|
|
FALLTHROUGH;
|
|
}
|
|
|
|
case SpvOpTypeStruct: {
|
|
if (p_member_var->flags & SPV_REFLECT_VARIABLE_FLAGS_PHYSICAL_POINTER_COPY) {
|
|
p_member_var->size = GetPhysicalPointerStructSize(p_parser, p_member_type->id);
|
|
} else {
|
|
SpvReflectResult result =
|
|
ParseDescriptorBlockVariableSizes(p_parser, p_module, false, is_parent_aos, is_parent_rta, p_member_var);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
}
|
|
} break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Structs can offset order don't need to match the index order, so first order by offset
|
|
// example:
|
|
// OpMemberDecorate %struct 0 Offset 4
|
|
// OpMemberDecorate %struct 1 Offset 0
|
|
SpvReflectBlockVariable** pp_member_offset_order =
|
|
(SpvReflectBlockVariable**)calloc(p_var->member_count, sizeof(SpvReflectBlockVariable*));
|
|
if (IsNull(pp_member_offset_order)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
|
|
uint32_t bottom_bound = 0;
|
|
for (uint32_t i = 0; i < p_var->member_count; ++i) {
|
|
uint32_t lowest_offset = UINT32_MAX;
|
|
uint32_t member_index = 0;
|
|
for (uint32_t j = 0; j < p_var->member_count; ++j) {
|
|
const uint32_t offset = p_var->members[j].offset;
|
|
if (offset < lowest_offset && offset >= bottom_bound) {
|
|
member_index = j;
|
|
lowest_offset = offset;
|
|
}
|
|
}
|
|
pp_member_offset_order[i] = &p_var->members[member_index];
|
|
bottom_bound = lowest_offset + 1; // 2 index can't share the same offset
|
|
}
|
|
|
|
// Parse padded size using offset difference for all member except for the last entry...
|
|
for (uint32_t i = 0; i < (p_var->member_count - 1); ++i) {
|
|
SpvReflectBlockVariable* p_member_var = pp_member_offset_order[i];
|
|
SpvReflectBlockVariable* p_next_member_var = pp_member_offset_order[i + 1];
|
|
p_member_var->padded_size = p_next_member_var->offset - p_member_var->offset;
|
|
if (p_member_var->size > p_member_var->padded_size) {
|
|
p_member_var->size = p_member_var->padded_size;
|
|
}
|
|
if (is_parent_rta) {
|
|
p_member_var->padded_size = p_member_var->size;
|
|
}
|
|
}
|
|
|
|
// ...last entry just gets rounded up to near multiple of SPIRV_DATA_ALIGNMENT, which is 16 and
|
|
// subtract the offset.
|
|
// last entry == entry with largest offset value
|
|
SpvReflectBlockVariable* p_last_member_var = pp_member_offset_order[p_var->member_count - 1];
|
|
p_last_member_var->padded_size =
|
|
RoundUp(p_last_member_var->offset + p_last_member_var->size, SPIRV_DATA_ALIGNMENT) - p_last_member_var->offset;
|
|
if (p_last_member_var->size > p_last_member_var->padded_size) {
|
|
p_last_member_var->size = p_last_member_var->padded_size;
|
|
}
|
|
if (is_parent_rta) {
|
|
p_last_member_var->padded_size = p_last_member_var->size;
|
|
}
|
|
|
|
SafeFree(pp_member_offset_order);
|
|
|
|
// If buffer ref, sizes are same as uint64_t
|
|
if (is_parent_ref) {
|
|
p_var->size = p_var->padded_size = 8;
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
// @TODO validate this with assertion
|
|
p_var->size = p_last_member_var->offset + p_last_member_var->padded_size;
|
|
p_var->padded_size = p_var->size;
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static void MarkSelfAndAllMemberVarsAsUsed(SpvReflectBlockVariable* p_var) {
|
|
// Clear the current variable's UNUSED flag
|
|
p_var->flags &= ~SPV_REFLECT_VARIABLE_FLAGS_UNUSED;
|
|
|
|
SpvOp op_type = p_var->type_description->op;
|
|
switch (op_type) {
|
|
default:
|
|
break;
|
|
|
|
case SpvOpTypeArray: {
|
|
} break;
|
|
|
|
case SpvOpTypeStruct: {
|
|
for (uint32_t i = 0; i < p_var->member_count; ++i) {
|
|
SpvReflectBlockVariable* p_member_var = &p_var->members[i];
|
|
MarkSelfAndAllMemberVarsAsUsed(p_member_var);
|
|
}
|
|
} break;
|
|
}
|
|
}
|
|
|
|
static SpvReflectResult ParseDescriptorBlockVariableUsage(SpvReflectPrvParser* p_parser, SpvReflectShaderModule* p_module,
|
|
SpvReflectPrvAccessChain* p_access_chain, uint32_t index_index,
|
|
SpvOp override_op_type, SpvReflectBlockVariable* p_var) {
|
|
// Clear the current variable's UNUSED flag
|
|
p_var->flags &= ~SPV_REFLECT_VARIABLE_FLAGS_UNUSED;
|
|
|
|
// Parsing arrays requires overriding the op type for
|
|
// for the lowest dim's element type.
|
|
SpvReflectTypeDescription* p_type = p_var->type_description;
|
|
SpvOp op_type = p_type->op;
|
|
if (override_op_type != (SpvOp)INVALID_VALUE) {
|
|
op_type = override_op_type;
|
|
}
|
|
|
|
switch (op_type) {
|
|
default:
|
|
break;
|
|
|
|
case SpvOpTypeArray: {
|
|
// Parse through array's type hierarchy to find the actual/non-array element type
|
|
while ((p_type->op == SpvOpTypeArray) && (index_index < p_access_chain->index_count)) {
|
|
// Find the array element type id
|
|
SpvReflectPrvNode* p_node = FindNode(p_parser, p_type->id);
|
|
if (p_node == NULL) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
uint32_t element_type_id = p_node->array_traits.element_type_id;
|
|
// Get the array element type
|
|
p_type = FindType(p_module, element_type_id);
|
|
if (p_type == NULL) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
// Next access chain index
|
|
index_index += 1;
|
|
}
|
|
|
|
// Only continue parsing if there's remaining indices in the access
|
|
// chain. If the end of the access chain has been reached then all
|
|
// remaining variables (including those in struct hierarchies)
|
|
// are considered USED.
|
|
//
|
|
// See: https://github.com/KhronosGroup/SPIRV-Reflect/issues/78
|
|
//
|
|
if (index_index < p_access_chain->index_count) {
|
|
// Parse current var again with a type override and advanced index index
|
|
SpvReflectResult result =
|
|
ParseDescriptorBlockVariableUsage(p_parser, p_module, p_access_chain, index_index, p_type->op, p_var);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
} else {
|
|
// Clear UNUSED flag for remaining variables
|
|
MarkSelfAndAllMemberVarsAsUsed(p_var);
|
|
}
|
|
} break;
|
|
|
|
case SpvOpTypePointer: {
|
|
// Reference. Get to underlying struct type.
|
|
SpvReflectPrvNode* p_type_node = FindNode(p_parser, p_type->id);
|
|
if (IsNull(p_type_node)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
// Get the pointee type
|
|
p_type = FindType(p_module, p_type_node->type_id);
|
|
if (IsNull(p_type)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
if (p_type->op != SpvOpTypeStruct) {
|
|
break;
|
|
}
|
|
FALLTHROUGH;
|
|
}
|
|
|
|
case SpvOpTypeStruct: {
|
|
assert(p_var->member_count > 0);
|
|
if (p_var->member_count == 0) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_UNEXPECTED_BLOCK_DATA;
|
|
}
|
|
|
|
// The access chain can have zero indexes, if used for a runtime array
|
|
if (p_access_chain->index_count == 0) {
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
// Get member variable at the access's chain current index
|
|
uint32_t index = p_access_chain->indexes[index_index];
|
|
if (index >= p_var->member_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_BLOCK_MEMBER_REFERENCE;
|
|
}
|
|
SpvReflectBlockVariable* p_member_var = &p_var->members[index];
|
|
|
|
bool is_pointer_to_pointer = IsPointerToPointer(p_parser, p_access_chain->result_type_id);
|
|
if (is_pointer_to_pointer) {
|
|
// Remember block var for this access chain for downstream dereference
|
|
p_access_chain->block_var = p_member_var;
|
|
}
|
|
|
|
// Next access chain index
|
|
index_index += 1;
|
|
|
|
// Only continue parsing if there's remaining indices in the access
|
|
// chain. If the end of the access chain has been reach then all
|
|
// remaining variables (including those in struct hierarchies)
|
|
// are considered USED.
|
|
//
|
|
// See: https://github.com/KhronosGroup/SPIRV-Reflect/issues/78
|
|
//
|
|
if (index_index < p_access_chain->index_count) {
|
|
SpvReflectResult result =
|
|
ParseDescriptorBlockVariableUsage(p_parser, p_module, p_access_chain, index_index, (SpvOp)INVALID_VALUE, p_member_var);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
} else if (!is_pointer_to_pointer) {
|
|
// Clear UNUSED flag for remaining variables
|
|
MarkSelfAndAllMemberVarsAsUsed(p_member_var);
|
|
}
|
|
} break;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseDescriptorBlocks(SpvReflectPrvParser* p_parser, SpvReflectShaderModule* p_module) {
|
|
if (p_module->descriptor_binding_count == 0) {
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
p_parser->physical_pointer_struct_count = 0;
|
|
|
|
for (uint32_t descriptor_index = 0; descriptor_index < p_module->descriptor_binding_count; ++descriptor_index) {
|
|
SpvReflectDescriptorBinding* p_descriptor = &(p_module->descriptor_bindings[descriptor_index]);
|
|
SpvReflectTypeDescription* p_type = p_descriptor->type_description;
|
|
if ((p_descriptor->descriptor_type != SPV_REFLECT_DESCRIPTOR_TYPE_UNIFORM_BUFFER) &&
|
|
(p_descriptor->descriptor_type != SPV_REFLECT_DESCRIPTOR_TYPE_STORAGE_BUFFER)) {
|
|
continue;
|
|
}
|
|
|
|
// Mark UNUSED
|
|
p_descriptor->block.flags |= SPV_REFLECT_VARIABLE_FLAGS_UNUSED;
|
|
p_parser->physical_pointer_count = 0;
|
|
// Parse descriptor block
|
|
SpvReflectResult result = ParseDescriptorBlockVariable(p_parser, p_module, p_type, &p_descriptor->block);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
|
|
for (uint32_t access_chain_index = 0; access_chain_index < p_parser->access_chain_count; ++access_chain_index) {
|
|
SpvReflectPrvAccessChain* p_access_chain = &(p_parser->access_chains[access_chain_index]);
|
|
// Skip any access chains that aren't touching this descriptor block
|
|
if (p_descriptor->spirv_id != p_access_chain->base_id) {
|
|
continue;
|
|
}
|
|
result = ParseDescriptorBlockVariableUsage(p_parser, p_module, p_access_chain, 0, (SpvOp)INVALID_VALUE, &p_descriptor->block);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
}
|
|
|
|
p_descriptor->block.name = p_descriptor->name;
|
|
|
|
bool is_parent_rta = (p_descriptor->descriptor_type == SPV_REFLECT_DESCRIPTOR_TYPE_STORAGE_BUFFER);
|
|
result = ParseDescriptorBlockVariableSizes(p_parser, p_module, true, false, is_parent_rta, &p_descriptor->block);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
|
|
if (is_parent_rta) {
|
|
p_descriptor->block.size = 0;
|
|
p_descriptor->block.padded_size = 0;
|
|
}
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseFormat(const SpvReflectTypeDescription* p_type, SpvReflectFormat* p_format) {
|
|
SpvReflectResult result = SPV_REFLECT_RESULT_ERROR_INTERNAL_ERROR;
|
|
bool signedness = (p_type->traits.numeric.scalar.signedness != 0);
|
|
uint32_t bit_width = p_type->traits.numeric.scalar.width;
|
|
if (p_type->type_flags & SPV_REFLECT_TYPE_FLAG_VECTOR) {
|
|
uint32_t component_count = p_type->traits.numeric.vector.component_count;
|
|
if (p_type->type_flags & SPV_REFLECT_TYPE_FLAG_FLOAT) {
|
|
switch (bit_width) {
|
|
case 16: {
|
|
switch (component_count) {
|
|
case 2:
|
|
*p_format = SPV_REFLECT_FORMAT_R16G16_SFLOAT;
|
|
break;
|
|
case 3:
|
|
*p_format = SPV_REFLECT_FORMAT_R16G16B16_SFLOAT;
|
|
break;
|
|
case 4:
|
|
*p_format = SPV_REFLECT_FORMAT_R16G16B16A16_SFLOAT;
|
|
break;
|
|
}
|
|
} break;
|
|
|
|
case 32: {
|
|
switch (component_count) {
|
|
case 2:
|
|
*p_format = SPV_REFLECT_FORMAT_R32G32_SFLOAT;
|
|
break;
|
|
case 3:
|
|
*p_format = SPV_REFLECT_FORMAT_R32G32B32_SFLOAT;
|
|
break;
|
|
case 4:
|
|
*p_format = SPV_REFLECT_FORMAT_R32G32B32A32_SFLOAT;
|
|
break;
|
|
}
|
|
} break;
|
|
|
|
case 64: {
|
|
switch (component_count) {
|
|
case 2:
|
|
*p_format = SPV_REFLECT_FORMAT_R64G64_SFLOAT;
|
|
break;
|
|
case 3:
|
|
*p_format = SPV_REFLECT_FORMAT_R64G64B64_SFLOAT;
|
|
break;
|
|
case 4:
|
|
*p_format = SPV_REFLECT_FORMAT_R64G64B64A64_SFLOAT;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
result = SPV_REFLECT_RESULT_SUCCESS;
|
|
} else if (p_type->type_flags & (SPV_REFLECT_TYPE_FLAG_INT | SPV_REFLECT_TYPE_FLAG_BOOL)) {
|
|
switch (bit_width) {
|
|
case 16: {
|
|
switch (component_count) {
|
|
case 2:
|
|
*p_format = signedness ? SPV_REFLECT_FORMAT_R16G16_SINT : SPV_REFLECT_FORMAT_R16G16_UINT;
|
|
break;
|
|
case 3:
|
|
*p_format = signedness ? SPV_REFLECT_FORMAT_R16G16B16_SINT : SPV_REFLECT_FORMAT_R16G16B16_UINT;
|
|
break;
|
|
case 4:
|
|
*p_format = signedness ? SPV_REFLECT_FORMAT_R16G16B16A16_SINT : SPV_REFLECT_FORMAT_R16G16B16A16_UINT;
|
|
break;
|
|
}
|
|
} break;
|
|
|
|
case 32: {
|
|
switch (component_count) {
|
|
case 2:
|
|
*p_format = signedness ? SPV_REFLECT_FORMAT_R32G32_SINT : SPV_REFLECT_FORMAT_R32G32_UINT;
|
|
break;
|
|
case 3:
|
|
*p_format = signedness ? SPV_REFLECT_FORMAT_R32G32B32_SINT : SPV_REFLECT_FORMAT_R32G32B32_UINT;
|
|
break;
|
|
case 4:
|
|
*p_format = signedness ? SPV_REFLECT_FORMAT_R32G32B32A32_SINT : SPV_REFLECT_FORMAT_R32G32B32A32_UINT;
|
|
break;
|
|
}
|
|
} break;
|
|
|
|
case 64: {
|
|
switch (component_count) {
|
|
case 2:
|
|
*p_format = signedness ? SPV_REFLECT_FORMAT_R64G64_SINT : SPV_REFLECT_FORMAT_R64G64_UINT;
|
|
break;
|
|
case 3:
|
|
*p_format = signedness ? SPV_REFLECT_FORMAT_R64G64B64_SINT : SPV_REFLECT_FORMAT_R64G64B64_UINT;
|
|
break;
|
|
case 4:
|
|
*p_format = signedness ? SPV_REFLECT_FORMAT_R64G64B64A64_SINT : SPV_REFLECT_FORMAT_R64G64B64A64_UINT;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
result = SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
} else if (p_type->type_flags & SPV_REFLECT_TYPE_FLAG_FLOAT) {
|
|
switch (bit_width) {
|
|
case 16:
|
|
*p_format = SPV_REFLECT_FORMAT_R16_SFLOAT;
|
|
break;
|
|
case 32:
|
|
*p_format = SPV_REFLECT_FORMAT_R32_SFLOAT;
|
|
break;
|
|
case 64:
|
|
*p_format = SPV_REFLECT_FORMAT_R64_SFLOAT;
|
|
break;
|
|
}
|
|
result = SPV_REFLECT_RESULT_SUCCESS;
|
|
} else if (p_type->type_flags & (SPV_REFLECT_TYPE_FLAG_INT | SPV_REFLECT_TYPE_FLAG_BOOL)) {
|
|
switch (bit_width) {
|
|
case 16:
|
|
*p_format = signedness ? SPV_REFLECT_FORMAT_R16_SINT : SPV_REFLECT_FORMAT_R16_UINT;
|
|
break;
|
|
break;
|
|
case 32:
|
|
*p_format = signedness ? SPV_REFLECT_FORMAT_R32_SINT : SPV_REFLECT_FORMAT_R32_UINT;
|
|
break;
|
|
break;
|
|
case 64:
|
|
*p_format = signedness ? SPV_REFLECT_FORMAT_R64_SINT : SPV_REFLECT_FORMAT_R64_UINT;
|
|
break;
|
|
}
|
|
result = SPV_REFLECT_RESULT_SUCCESS;
|
|
} else if (p_type->type_flags & SPV_REFLECT_TYPE_FLAG_STRUCT) {
|
|
*p_format = SPV_REFLECT_FORMAT_UNDEFINED;
|
|
result = SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
static SpvReflectResult ParseInterfaceVariable(SpvReflectPrvParser* p_parser,
|
|
const SpvReflectPrvDecorations* p_var_node_decorations,
|
|
const SpvReflectPrvDecorations* p_type_node_decorations,
|
|
SpvReflectShaderModule* p_module, SpvReflectTypeDescription* p_type,
|
|
SpvReflectInterfaceVariable* p_var, bool* p_has_built_in) {
|
|
SpvReflectPrvNode* p_type_node = FindNode(p_parser, p_type->id);
|
|
if (IsNull(p_type_node)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
|
|
if (p_type->member_count > 0) {
|
|
p_var->member_count = p_type->member_count;
|
|
p_var->members = (SpvReflectInterfaceVariable*)calloc(p_var->member_count, sizeof(*p_var->members));
|
|
if (IsNull(p_var->members)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
|
|
for (uint32_t member_index = 0; member_index < p_type_node->member_count; ++member_index) {
|
|
SpvReflectPrvDecorations* p_member_decorations = &p_type_node->member_decorations[member_index];
|
|
SpvReflectTypeDescription* p_member_type = &p_type->members[member_index];
|
|
SpvReflectInterfaceVariable* p_member_var = &p_var->members[member_index];
|
|
|
|
// Storage class is the same throughout the whole struct
|
|
p_member_var->storage_class = p_var->storage_class;
|
|
|
|
SpvReflectResult result =
|
|
ParseInterfaceVariable(p_parser, NULL, p_member_decorations, p_module, p_member_type, p_member_var, p_has_built_in);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
SPV_REFLECT_ASSERT(false);
|
|
return result;
|
|
}
|
|
}
|
|
}
|
|
|
|
p_var->name = p_type_node->name;
|
|
p_var->decoration_flags = ApplyDecorations(p_type_node_decorations);
|
|
if (p_var_node_decorations != NULL) {
|
|
p_var->decoration_flags |= ApplyDecorations(p_var_node_decorations);
|
|
} else {
|
|
// Apply member decoration values to struct members
|
|
p_var->location = p_type_node_decorations->location.value;
|
|
p_var->component = p_type_node_decorations->component.value;
|
|
}
|
|
|
|
p_var->built_in = p_type_node_decorations->built_in;
|
|
ApplyNumericTraits(p_type, &p_var->numeric);
|
|
if (p_type->op == SpvOpTypeArray) {
|
|
ApplyArrayTraits(p_type, &p_var->array);
|
|
}
|
|
|
|
p_var->type_description = p_type;
|
|
|
|
*p_has_built_in |= p_type_node_decorations->is_built_in;
|
|
|
|
// Only parse format for interface variables that are input or output
|
|
if ((p_var->storage_class == SpvStorageClassInput) || (p_var->storage_class == SpvStorageClassOutput)) {
|
|
SpvReflectResult result = ParseFormat(p_var->type_description, &p_var->format);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
SPV_REFLECT_ASSERT(false);
|
|
return result;
|
|
}
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseInterfaceVariables(SpvReflectPrvParser* p_parser, SpvReflectShaderModule* p_module,
|
|
SpvReflectEntryPoint* p_entry, uint32_t interface_variable_count,
|
|
uint32_t* p_interface_variable_ids) {
|
|
if (interface_variable_count == 0) {
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
p_entry->interface_variable_count = interface_variable_count;
|
|
p_entry->input_variable_count = 0;
|
|
p_entry->output_variable_count = 0;
|
|
for (size_t i = 0; i < interface_variable_count; ++i) {
|
|
uint32_t var_result_id = *(p_interface_variable_ids + i);
|
|
SpvReflectPrvNode* p_node = FindNode(p_parser, var_result_id);
|
|
if (IsNull(p_node)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
|
|
if (p_node->storage_class == SpvStorageClassInput) {
|
|
p_entry->input_variable_count += 1;
|
|
} else if (p_node->storage_class == SpvStorageClassOutput) {
|
|
p_entry->output_variable_count += 1;
|
|
}
|
|
}
|
|
|
|
if (p_entry->input_variable_count > 0) {
|
|
p_entry->input_variables =
|
|
(SpvReflectInterfaceVariable**)calloc(p_entry->input_variable_count, sizeof(*(p_entry->input_variables)));
|
|
if (IsNull(p_entry->input_variables)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
}
|
|
|
|
if (p_entry->output_variable_count > 0) {
|
|
p_entry->output_variables =
|
|
(SpvReflectInterfaceVariable**)calloc(p_entry->output_variable_count, sizeof(*(p_entry->output_variables)));
|
|
if (IsNull(p_entry->output_variables)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
}
|
|
|
|
if (p_entry->interface_variable_count > 0) {
|
|
p_entry->interface_variables =
|
|
(SpvReflectInterfaceVariable*)calloc(p_entry->interface_variable_count, sizeof(*(p_entry->interface_variables)));
|
|
if (IsNull(p_entry->interface_variables)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
}
|
|
|
|
size_t input_index = 0;
|
|
size_t output_index = 0;
|
|
for (size_t i = 0; i < interface_variable_count; ++i) {
|
|
uint32_t var_result_id = *(p_interface_variable_ids + i);
|
|
SpvReflectPrvNode* p_node = FindNode(p_parser, var_result_id);
|
|
if (IsNull(p_node)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
|
|
SpvReflectTypeDescription* p_type = FindType(p_module, p_node->type_id);
|
|
if (IsNull(p_node) || IsNull(p_type)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
// If the type is a pointer, resolve it
|
|
if (p_type->op == SpvOpTypePointer) {
|
|
// Find the type's node
|
|
SpvReflectPrvNode* p_type_node = FindNode(p_parser, p_type->id);
|
|
if (IsNull(p_type_node)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
// Should be the resolved type
|
|
p_type = FindType(p_module, p_type_node->type_id);
|
|
if (IsNull(p_type)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
}
|
|
|
|
SpvReflectPrvNode* p_type_node = FindNode(p_parser, p_type->id);
|
|
if (IsNull(p_type_node)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
|
|
SpvReflectInterfaceVariable* p_var = &(p_entry->interface_variables[i]);
|
|
p_var->storage_class = p_node->storage_class;
|
|
|
|
bool has_built_in = p_node->decorations.is_built_in;
|
|
SpvReflectResult result =
|
|
ParseInterfaceVariable(p_parser, &p_node->decorations, &p_type_node->decorations, p_module, p_type, p_var, &has_built_in);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
SPV_REFLECT_ASSERT(false);
|
|
return result;
|
|
}
|
|
|
|
// Input and output variables
|
|
if (p_var->storage_class == SpvStorageClassInput) {
|
|
p_entry->input_variables[input_index] = p_var;
|
|
++input_index;
|
|
} else if (p_node->storage_class == SpvStorageClassOutput) {
|
|
p_entry->output_variables[output_index] = p_var;
|
|
++output_index;
|
|
}
|
|
|
|
// SPIR-V result id
|
|
p_var->spirv_id = p_node->result_id;
|
|
// Name
|
|
p_var->name = p_node->name;
|
|
// Semantic
|
|
p_var->semantic = p_node->decorations.semantic.value;
|
|
|
|
// Decorate with built-in if any member is built-in
|
|
if (has_built_in) {
|
|
p_var->decoration_flags |= SPV_REFLECT_DECORATION_BUILT_IN;
|
|
}
|
|
|
|
// Location is decorated on OpVariable node, not the type node.
|
|
p_var->location = p_node->decorations.location.value;
|
|
p_var->component = p_node->decorations.component.value;
|
|
p_var->word_offset.location = p_node->decorations.location.word_offset;
|
|
|
|
// Built in
|
|
if (p_node->decorations.is_built_in) {
|
|
p_var->built_in = p_node->decorations.built_in;
|
|
}
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult EnumerateAllPushConstants(SpvReflectShaderModule* p_module, size_t* p_push_constant_count,
|
|
uint32_t** p_push_constants) {
|
|
*p_push_constant_count = p_module->push_constant_block_count;
|
|
if (*p_push_constant_count == 0) {
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
*p_push_constants = (uint32_t*)calloc(*p_push_constant_count, sizeof(**p_push_constants));
|
|
|
|
if (IsNull(*p_push_constants)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
|
|
for (size_t i = 0; i < *p_push_constant_count; ++i) {
|
|
(*p_push_constants)[i] = p_module->push_constant_blocks[i].spirv_id;
|
|
}
|
|
qsort(*p_push_constants, *p_push_constant_count, sizeof(**p_push_constants), SortCompareUint32);
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult TraverseCallGraph(SpvReflectPrvParser* p_parser, SpvReflectPrvFunction* p_func, size_t* p_func_count,
|
|
uint32_t* p_func_ids, uint32_t depth) {
|
|
if (depth > p_parser->function_count) {
|
|
// Vulkan does not permit recursion (Vulkan spec Appendix A):
|
|
// "Recursion: The static function-call graph for an entry point must not
|
|
// contain cycles."
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_RECURSION;
|
|
}
|
|
if (IsNotNull(p_func_ids)) {
|
|
p_func_ids[(*p_func_count)++] = p_func->id;
|
|
} else {
|
|
++*p_func_count;
|
|
}
|
|
for (size_t i = 0; i < p_func->callee_count; ++i) {
|
|
SpvReflectResult result = TraverseCallGraph(p_parser, p_func->callee_ptrs[i], p_func_count, p_func_ids, depth + 1);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
}
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static uint32_t GetUint32Constant(SpvReflectPrvParser* p_parser, uint32_t id) {
|
|
uint32_t result = (uint32_t)INVALID_VALUE;
|
|
SpvReflectPrvNode* p_node = FindNode(p_parser, id);
|
|
if (p_node && p_node->op == SpvOpConstant) {
|
|
UNCHECKED_READU32(p_parser, p_node->word_offset + 3, result);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
static bool HasByteAddressBufferOffset(SpvReflectPrvNode* p_node, SpvReflectDescriptorBinding* p_binding) {
|
|
return IsNotNull(p_node) && IsNotNull(p_binding) && p_node->op == SpvOpAccessChain && p_node->word_count == 6 &&
|
|
(p_binding->user_type == SPV_REFLECT_USER_TYPE_BYTE_ADDRESS_BUFFER ||
|
|
p_binding->user_type == SPV_REFLECT_USER_TYPE_RW_BYTE_ADDRESS_BUFFER);
|
|
}
|
|
|
|
static SpvReflectResult ParseByteAddressBuffer(SpvReflectPrvParser* p_parser, SpvReflectPrvNode* p_node,
|
|
SpvReflectDescriptorBinding* p_binding) {
|
|
const SpvReflectResult not_found = SPV_REFLECT_RESULT_SUCCESS;
|
|
if (!HasByteAddressBufferOffset(p_node, p_binding)) {
|
|
return not_found;
|
|
}
|
|
|
|
uint32_t offset = 0; // starting offset
|
|
|
|
uint32_t base_id = 0;
|
|
// expect first index of 2D access is zero
|
|
UNCHECKED_READU32(p_parser, p_node->word_offset + 4, base_id);
|
|
if (GetUint32Constant(p_parser, base_id) != 0) {
|
|
return not_found;
|
|
}
|
|
UNCHECKED_READU32(p_parser, p_node->word_offset + 5, base_id);
|
|
SpvReflectPrvNode* p_next_node = FindNode(p_parser, base_id);
|
|
if (IsNull(p_next_node)) {
|
|
return not_found;
|
|
} else if (p_next_node->op == SpvOpConstant) {
|
|
// The access chain might just be a constant right to the offset
|
|
offset = GetUint32Constant(p_parser, base_id);
|
|
p_binding->byte_address_buffer_offsets[p_binding->byte_address_buffer_offset_count] = offset;
|
|
p_binding->byte_address_buffer_offset_count++;
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
// there is usually 2 (sometimes 3) instrucitons that make up the arithmetic logic to calculate the offset
|
|
SpvReflectPrvNode* arithmetic_node_stack[8];
|
|
uint32_t arithmetic_count = 0;
|
|
|
|
while (IsNotNull(p_next_node)) {
|
|
if (p_next_node->op == SpvOpLoad || p_next_node->op == SpvOpBitcast || p_next_node->op == SpvOpConstant) {
|
|
break; // arithmetic starts here
|
|
}
|
|
arithmetic_node_stack[arithmetic_count++] = p_next_node;
|
|
if (arithmetic_count >= 8) {
|
|
return not_found;
|
|
}
|
|
|
|
UNCHECKED_READU32(p_parser, p_next_node->word_offset + 3, base_id);
|
|
p_next_node = FindNode(p_parser, base_id);
|
|
}
|
|
|
|
const uint32_t count = arithmetic_count;
|
|
for (uint32_t i = 0; i < count; i++) {
|
|
p_next_node = arithmetic_node_stack[--arithmetic_count];
|
|
// All arithmetic ops takes 2 operands, assumption is the 2nd operand has the constant
|
|
UNCHECKED_READU32(p_parser, p_next_node->word_offset + 4, base_id);
|
|
uint32_t value = GetUint32Constant(p_parser, base_id);
|
|
if (value == INVALID_VALUE) {
|
|
return not_found;
|
|
}
|
|
|
|
switch (p_next_node->op) {
|
|
case SpvOpShiftRightLogical:
|
|
offset >>= value;
|
|
break;
|
|
case SpvOpIAdd:
|
|
offset += value;
|
|
break;
|
|
case SpvOpISub:
|
|
offset -= value;
|
|
break;
|
|
case SpvOpIMul:
|
|
offset *= value;
|
|
break;
|
|
case SpvOpUDiv:
|
|
offset /= value;
|
|
break;
|
|
case SpvOpSDiv:
|
|
// OpConstant might be signed, but value should never be negative
|
|
assert((int32_t)value > 0);
|
|
offset /= value;
|
|
break;
|
|
default:
|
|
return not_found;
|
|
}
|
|
}
|
|
|
|
p_binding->byte_address_buffer_offsets[p_binding->byte_address_buffer_offset_count] = offset;
|
|
p_binding->byte_address_buffer_offset_count++;
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseStaticallyUsedResources(SpvReflectPrvParser* p_parser, SpvReflectShaderModule* p_module,
|
|
SpvReflectEntryPoint* p_entry, size_t uniform_count, uint32_t* uniforms,
|
|
size_t push_constant_count, uint32_t* push_constants) {
|
|
// Find function with the right id
|
|
SpvReflectPrvFunction* p_func = NULL;
|
|
for (size_t i = 0; i < p_parser->function_count; ++i) {
|
|
if (p_parser->functions[i].id == p_entry->id) {
|
|
p_func = &(p_parser->functions[i]);
|
|
break;
|
|
}
|
|
}
|
|
if (p_func == NULL) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
|
|
size_t called_function_count = 0;
|
|
SpvReflectResult result = TraverseCallGraph(p_parser, p_func, &called_function_count, NULL, 0);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
|
|
uint32_t* p_called_functions = NULL;
|
|
if (called_function_count > 0) {
|
|
p_called_functions = (uint32_t*)calloc(called_function_count, sizeof(*p_called_functions));
|
|
if (IsNull(p_called_functions)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
}
|
|
|
|
called_function_count = 0;
|
|
result = TraverseCallGraph(p_parser, p_func, &called_function_count, p_called_functions, 0);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
SafeFree(p_called_functions);
|
|
return result;
|
|
}
|
|
|
|
if (called_function_count > 0) {
|
|
qsort(p_called_functions, called_function_count, sizeof(*p_called_functions), SortCompareUint32);
|
|
}
|
|
called_function_count = DedupSortedUint32(p_called_functions, called_function_count);
|
|
|
|
uint32_t used_acessed_count = 0;
|
|
for (size_t i = 0, j = 0; i < called_function_count; ++i) {
|
|
// No need to bounds check j because a missing ID issue would have been
|
|
// found during TraverseCallGraph
|
|
while (p_parser->functions[j].id != p_called_functions[i]) {
|
|
++j;
|
|
}
|
|
used_acessed_count += p_parser->functions[j].accessed_variable_count;
|
|
}
|
|
SpvReflectPrvAccessedVariable* p_used_accesses = NULL;
|
|
if (used_acessed_count > 0) {
|
|
p_used_accesses = (SpvReflectPrvAccessedVariable*)calloc(used_acessed_count, sizeof(SpvReflectPrvAccessedVariable));
|
|
if (IsNull(p_used_accesses)) {
|
|
SafeFree(p_called_functions);
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
}
|
|
used_acessed_count = 0;
|
|
for (size_t i = 0, j = 0; i < called_function_count; ++i) {
|
|
while (p_parser->functions[j].id != p_called_functions[i]) {
|
|
++j;
|
|
}
|
|
|
|
memcpy(&p_used_accesses[used_acessed_count], p_parser->functions[j].accessed_variables,
|
|
p_parser->functions[j].accessed_variable_count * sizeof(SpvReflectPrvAccessedVariable));
|
|
used_acessed_count += p_parser->functions[j].accessed_variable_count;
|
|
}
|
|
SafeFree(p_called_functions);
|
|
|
|
if (used_acessed_count > 0) {
|
|
qsort(p_used_accesses, used_acessed_count, sizeof(*p_used_accesses), SortCompareAccessedVariable);
|
|
}
|
|
|
|
// Do set intersection to find the used uniform and push constants
|
|
size_t used_uniform_count = 0;
|
|
result = IntersectSortedAccessedVariable(p_used_accesses, used_acessed_count, uniforms, uniform_count, &p_entry->used_uniforms,
|
|
&used_uniform_count);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
SafeFree(p_used_accesses);
|
|
return result;
|
|
}
|
|
|
|
size_t used_push_constant_count = 0;
|
|
result = IntersectSortedAccessedVariable(p_used_accesses, used_acessed_count, push_constants, push_constant_count,
|
|
&p_entry->used_push_constants, &used_push_constant_count);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
SafeFree(p_used_accesses);
|
|
return result;
|
|
}
|
|
|
|
for (uint32_t i = 0; i < p_module->descriptor_binding_count; ++i) {
|
|
SpvReflectDescriptorBinding* p_binding = &p_module->descriptor_bindings[i];
|
|
uint32_t byte_address_buffer_offset_count = 0;
|
|
|
|
for (uint32_t j = 0; j < used_acessed_count; j++) {
|
|
if (p_used_accesses[j].variable_ptr == p_binding->spirv_id) {
|
|
p_binding->accessed = 1;
|
|
|
|
if (HasByteAddressBufferOffset(p_used_accesses[j].p_node, p_binding)) {
|
|
byte_address_buffer_offset_count++;
|
|
}
|
|
}
|
|
}
|
|
|
|
// only if SPIR-V has ByteAddressBuffer user type
|
|
if (byte_address_buffer_offset_count > 0) {
|
|
bool multi_entrypoint = p_binding->byte_address_buffer_offset_count > 0;
|
|
if (multi_entrypoint) {
|
|
// If there is a 2nd entrypoint, we can have multiple entry points, in this case we want to just combine the accessed
|
|
// offsets and then de-duplicate it
|
|
uint32_t* prev_byte_address_buffer_offsets = p_binding->byte_address_buffer_offsets;
|
|
p_binding->byte_address_buffer_offsets =
|
|
(uint32_t*)calloc(byte_address_buffer_offset_count + p_binding->byte_address_buffer_offset_count, sizeof(uint32_t));
|
|
memcpy(p_binding->byte_address_buffer_offsets, prev_byte_address_buffer_offsets,
|
|
sizeof(uint32_t) * p_binding->byte_address_buffer_offset_count);
|
|
SafeFree(prev_byte_address_buffer_offsets);
|
|
} else {
|
|
// possible not all allocated offset slots are used, but this will be a max per binding
|
|
p_binding->byte_address_buffer_offsets = (uint32_t*)calloc(byte_address_buffer_offset_count, sizeof(uint32_t));
|
|
}
|
|
|
|
if (IsNull(p_binding->byte_address_buffer_offsets)) {
|
|
SafeFree(p_used_accesses);
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
|
|
for (uint32_t j = 0; j < used_acessed_count; j++) {
|
|
if (p_used_accesses[j].variable_ptr == p_binding->spirv_id) {
|
|
result = ParseByteAddressBuffer(p_parser, p_used_accesses[j].p_node, p_binding);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
SafeFree(p_used_accesses);
|
|
return result;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (multi_entrypoint) {
|
|
qsort(p_binding->byte_address_buffer_offsets, p_binding->byte_address_buffer_offset_count,
|
|
sizeof(*(p_binding->byte_address_buffer_offsets)), SortCompareUint32);
|
|
p_binding->byte_address_buffer_offset_count =
|
|
(uint32_t)DedupSortedUint32(p_binding->byte_address_buffer_offsets, p_binding->byte_address_buffer_offset_count);
|
|
}
|
|
}
|
|
}
|
|
|
|
SafeFree(p_used_accesses);
|
|
|
|
p_entry->used_uniform_count = (uint32_t)used_uniform_count;
|
|
p_entry->used_push_constant_count = (uint32_t)used_push_constant_count;
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseEntryPoints(SpvReflectPrvParser* p_parser, SpvReflectShaderModule* p_module) {
|
|
if (p_parser->entry_point_count == 0) {
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
p_module->entry_point_count = p_parser->entry_point_count;
|
|
p_module->entry_points = (SpvReflectEntryPoint*)calloc(p_module->entry_point_count, sizeof(*(p_module->entry_points)));
|
|
if (IsNull(p_module->entry_points)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
|
|
SpvReflectResult result;
|
|
size_t uniform_count = 0;
|
|
uint32_t* uniforms = NULL;
|
|
if ((result = EnumerateAllUniforms(p_module, &uniform_count, &uniforms)) != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
size_t push_constant_count = 0;
|
|
uint32_t* push_constants = NULL;
|
|
if ((result = EnumerateAllPushConstants(p_module, &push_constant_count, &push_constants)) != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
|
|
size_t entry_point_index = 0;
|
|
for (size_t i = 0; entry_point_index < p_parser->entry_point_count && i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
if (p_node->op != SpvOpEntryPoint) {
|
|
continue;
|
|
}
|
|
|
|
SpvReflectEntryPoint* p_entry_point = &(p_module->entry_points[entry_point_index]);
|
|
CHECKED_READU32_CAST(p_parser, p_node->word_offset + 1, SpvExecutionModel, p_entry_point->spirv_execution_model);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, p_entry_point->id);
|
|
|
|
switch (p_entry_point->spirv_execution_model) {
|
|
default:
|
|
break;
|
|
case SpvExecutionModelVertex:
|
|
p_entry_point->shader_stage = SPV_REFLECT_SHADER_STAGE_VERTEX_BIT;
|
|
break;
|
|
case SpvExecutionModelTessellationControl:
|
|
p_entry_point->shader_stage = SPV_REFLECT_SHADER_STAGE_TESSELLATION_CONTROL_BIT;
|
|
break;
|
|
case SpvExecutionModelTessellationEvaluation:
|
|
p_entry_point->shader_stage = SPV_REFLECT_SHADER_STAGE_TESSELLATION_EVALUATION_BIT;
|
|
break;
|
|
case SpvExecutionModelGeometry:
|
|
p_entry_point->shader_stage = SPV_REFLECT_SHADER_STAGE_GEOMETRY_BIT;
|
|
break;
|
|
case SpvExecutionModelFragment:
|
|
p_entry_point->shader_stage = SPV_REFLECT_SHADER_STAGE_FRAGMENT_BIT;
|
|
break;
|
|
case SpvExecutionModelGLCompute:
|
|
p_entry_point->shader_stage = SPV_REFLECT_SHADER_STAGE_COMPUTE_BIT;
|
|
break;
|
|
case SpvExecutionModelTaskNV:
|
|
p_entry_point->shader_stage = SPV_REFLECT_SHADER_STAGE_TASK_BIT_NV;
|
|
break;
|
|
case SpvExecutionModelTaskEXT:
|
|
p_entry_point->shader_stage = SPV_REFLECT_SHADER_STAGE_TASK_BIT_EXT;
|
|
break;
|
|
case SpvExecutionModelMeshNV:
|
|
p_entry_point->shader_stage = SPV_REFLECT_SHADER_STAGE_MESH_BIT_NV;
|
|
break;
|
|
case SpvExecutionModelMeshEXT:
|
|
p_entry_point->shader_stage = SPV_REFLECT_SHADER_STAGE_MESH_BIT_EXT;
|
|
break;
|
|
case SpvExecutionModelRayGenerationKHR:
|
|
p_entry_point->shader_stage = SPV_REFLECT_SHADER_STAGE_RAYGEN_BIT_KHR;
|
|
break;
|
|
case SpvExecutionModelIntersectionKHR:
|
|
p_entry_point->shader_stage = SPV_REFLECT_SHADER_STAGE_INTERSECTION_BIT_KHR;
|
|
break;
|
|
case SpvExecutionModelAnyHitKHR:
|
|
p_entry_point->shader_stage = SPV_REFLECT_SHADER_STAGE_ANY_HIT_BIT_KHR;
|
|
break;
|
|
case SpvExecutionModelClosestHitKHR:
|
|
p_entry_point->shader_stage = SPV_REFLECT_SHADER_STAGE_CLOSEST_HIT_BIT_KHR;
|
|
break;
|
|
case SpvExecutionModelMissKHR:
|
|
p_entry_point->shader_stage = SPV_REFLECT_SHADER_STAGE_MISS_BIT_KHR;
|
|
break;
|
|
case SpvExecutionModelCallableKHR:
|
|
p_entry_point->shader_stage = SPV_REFLECT_SHADER_STAGE_CALLABLE_BIT_KHR;
|
|
break;
|
|
}
|
|
|
|
++entry_point_index;
|
|
|
|
// Name length is required to calculate next operand
|
|
uint32_t name_start_word_offset = 3;
|
|
uint32_t name_length_with_terminator = 0;
|
|
result =
|
|
ReadStr(p_parser, p_node->word_offset + name_start_word_offset, 0, p_node->word_count, &name_length_with_terminator, NULL);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
p_entry_point->name = (const char*)(p_parser->spirv_code + p_node->word_offset + name_start_word_offset);
|
|
|
|
uint32_t name_word_count = RoundUp(name_length_with_terminator, SPIRV_WORD_SIZE) / SPIRV_WORD_SIZE;
|
|
uint32_t interface_variable_count = (p_node->word_count - (name_start_word_offset + name_word_count));
|
|
uint32_t* p_interface_variables = NULL;
|
|
if (interface_variable_count > 0) {
|
|
p_interface_variables = (uint32_t*)calloc(interface_variable_count, sizeof(*(p_interface_variables)));
|
|
if (IsNull(p_interface_variables)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
}
|
|
|
|
for (uint32_t var_index = 0; var_index < interface_variable_count; ++var_index) {
|
|
uint32_t var_result_id = (uint32_t)INVALID_VALUE;
|
|
uint32_t offset = name_start_word_offset + name_word_count + var_index;
|
|
CHECKED_READU32(p_parser, p_node->word_offset + offset, var_result_id);
|
|
p_interface_variables[var_index] = var_result_id;
|
|
}
|
|
|
|
result = ParseInterfaceVariables(p_parser, p_module, p_entry_point, interface_variable_count, p_interface_variables);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
SafeFree(p_interface_variables);
|
|
|
|
result = ParseStaticallyUsedResources(p_parser, p_module, p_entry_point, uniform_count, uniforms, push_constant_count,
|
|
push_constants);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
}
|
|
|
|
SafeFree(uniforms);
|
|
SafeFree(push_constants);
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseExecutionModes(SpvReflectPrvParser* p_parser, SpvReflectShaderModule* p_module) {
|
|
assert(IsNotNull(p_parser));
|
|
assert(IsNotNull(p_parser->nodes));
|
|
assert(IsNotNull(p_module));
|
|
|
|
if (IsNotNull(p_parser) && IsNotNull(p_parser->spirv_code) && IsNotNull(p_parser->nodes)) {
|
|
for (size_t node_idx = 0; node_idx < p_parser->node_count; ++node_idx) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[node_idx]);
|
|
if (p_node->op != SpvOpExecutionMode && p_node->op != SpvOpExecutionModeId) {
|
|
continue;
|
|
}
|
|
|
|
// Read entry point id
|
|
uint32_t entry_point_id = 0;
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, entry_point_id);
|
|
|
|
// Find entry point
|
|
SpvReflectEntryPoint* p_entry_point = NULL;
|
|
for (size_t entry_point_idx = 0; entry_point_idx < p_module->entry_point_count; ++entry_point_idx) {
|
|
if (p_module->entry_points[entry_point_idx].id == entry_point_id) {
|
|
p_entry_point = &p_module->entry_points[entry_point_idx];
|
|
break;
|
|
}
|
|
}
|
|
// Bail if entry point is null
|
|
if (IsNull(p_entry_point)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ENTRY_POINT;
|
|
}
|
|
|
|
// Read execution mode
|
|
uint32_t execution_mode = (uint32_t)INVALID_VALUE;
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, execution_mode);
|
|
|
|
// Parse execution mode
|
|
switch (execution_mode) {
|
|
case SpvExecutionModeInvocations: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 3, p_entry_point->invocations);
|
|
} break;
|
|
|
|
case SpvExecutionModeLocalSize: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 3, p_entry_point->local_size.x);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 4, p_entry_point->local_size.y);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 5, p_entry_point->local_size.z);
|
|
} break;
|
|
case SpvExecutionModeLocalSizeId: {
|
|
uint32_t local_size_x_id = 0;
|
|
uint32_t local_size_y_id = 0;
|
|
uint32_t local_size_z_id = 0;
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 3, local_size_x_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 4, local_size_y_id);
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 5, local_size_z_id);
|
|
|
|
SpvReflectPrvNode* x_node = FindNode(p_parser, local_size_x_id);
|
|
SpvReflectPrvNode* y_node = FindNode(p_parser, local_size_y_id);
|
|
SpvReflectPrvNode* z_node = FindNode(p_parser, local_size_z_id);
|
|
if (IsNotNull(x_node) && IsNotNull(y_node) && IsNotNull(z_node)) {
|
|
if (IsSpecConstant(x_node)) {
|
|
p_entry_point->local_size.x = (uint32_t)SPV_REFLECT_EXECUTION_MODE_SPEC_CONSTANT;
|
|
} else {
|
|
CHECKED_READU32(p_parser, x_node->word_offset + 3, p_entry_point->local_size.x);
|
|
}
|
|
|
|
if (IsSpecConstant(y_node)) {
|
|
p_entry_point->local_size.y = (uint32_t)SPV_REFLECT_EXECUTION_MODE_SPEC_CONSTANT;
|
|
} else {
|
|
CHECKED_READU32(p_parser, y_node->word_offset + 3, p_entry_point->local_size.y);
|
|
}
|
|
|
|
if (IsSpecConstant(z_node)) {
|
|
p_entry_point->local_size.z = (uint32_t)SPV_REFLECT_EXECUTION_MODE_SPEC_CONSTANT;
|
|
} else {
|
|
CHECKED_READU32(p_parser, z_node->word_offset + 3, p_entry_point->local_size.z);
|
|
}
|
|
}
|
|
} break;
|
|
|
|
case SpvExecutionModeOutputVertices: {
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 3, p_entry_point->output_vertices);
|
|
} break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
p_entry_point->execution_mode_count++;
|
|
}
|
|
uint32_t* indices = (uint32_t*)calloc(p_module->entry_point_count, sizeof(indices));
|
|
if (IsNull(indices)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
for (size_t entry_point_idx = 0; entry_point_idx < p_module->entry_point_count; ++entry_point_idx) {
|
|
SpvReflectEntryPoint* p_entry_point = &p_module->entry_points[entry_point_idx];
|
|
if (p_entry_point->execution_mode_count > 0) {
|
|
p_entry_point->execution_modes =
|
|
(SpvExecutionMode*)calloc(p_entry_point->execution_mode_count, sizeof(*p_entry_point->execution_modes));
|
|
if (IsNull(p_entry_point->execution_modes)) {
|
|
SafeFree(indices);
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
}
|
|
}
|
|
|
|
for (size_t node_idx = 0; node_idx < p_parser->node_count; ++node_idx) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[node_idx]);
|
|
if (p_node->op != SpvOpExecutionMode) {
|
|
continue;
|
|
}
|
|
|
|
// Read entry point id
|
|
uint32_t entry_point_id = 0;
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 1, entry_point_id);
|
|
|
|
// Find entry point
|
|
SpvReflectEntryPoint* p_entry_point = NULL;
|
|
uint32_t* idx = NULL;
|
|
for (size_t entry_point_idx = 0; entry_point_idx < p_module->entry_point_count; ++entry_point_idx) {
|
|
if (p_module->entry_points[entry_point_idx].id == entry_point_id) {
|
|
p_entry_point = &p_module->entry_points[entry_point_idx];
|
|
idx = &indices[entry_point_idx];
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Read execution mode
|
|
uint32_t execution_mode = (uint32_t)INVALID_VALUE;
|
|
CHECKED_READU32(p_parser, p_node->word_offset + 2, execution_mode);
|
|
p_entry_point->execution_modes[(*idx)++] = (SpvExecutionMode)execution_mode;
|
|
}
|
|
SafeFree(indices);
|
|
}
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParsePushConstantBlocks(SpvReflectPrvParser* p_parser, SpvReflectShaderModule* p_module) {
|
|
for (size_t i = 0; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
if ((p_node->op != SpvOpVariable) || (p_node->storage_class != SpvStorageClassPushConstant)) {
|
|
continue;
|
|
}
|
|
|
|
p_module->push_constant_block_count += 1;
|
|
}
|
|
|
|
if (p_module->push_constant_block_count == 0) {
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
p_module->push_constant_blocks =
|
|
(SpvReflectBlockVariable*)calloc(p_module->push_constant_block_count, sizeof(*p_module->push_constant_blocks));
|
|
if (IsNull(p_module->push_constant_blocks)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
|
|
p_parser->physical_pointer_struct_count = 0;
|
|
uint32_t push_constant_index = 0;
|
|
for (size_t i = 0; i < p_parser->node_count; ++i) {
|
|
SpvReflectPrvNode* p_node = &(p_parser->nodes[i]);
|
|
if ((p_node->op != SpvOpVariable) || (p_node->storage_class != SpvStorageClassPushConstant)) {
|
|
continue;
|
|
}
|
|
|
|
SpvReflectTypeDescription* p_type = FindType(p_module, p_node->type_id);
|
|
if (IsNull(p_node) || IsNull(p_type)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
// If the type is a pointer, resolve it
|
|
if (p_type->op == SpvOpTypePointer) {
|
|
// Find the type's node
|
|
SpvReflectPrvNode* p_type_node = FindNode(p_parser, p_type->id);
|
|
if (IsNull(p_type_node)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
// Should be the resolved type
|
|
p_type = FindType(p_module, p_type_node->type_id);
|
|
if (IsNull(p_type)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
}
|
|
|
|
SpvReflectPrvNode* p_type_node = FindNode(p_parser, p_type->id);
|
|
if (IsNull(p_type_node)) {
|
|
return SPV_REFLECT_RESULT_ERROR_SPIRV_INVALID_ID_REFERENCE;
|
|
}
|
|
|
|
SpvReflectBlockVariable* p_push_constant = &p_module->push_constant_blocks[push_constant_index];
|
|
p_push_constant->spirv_id = p_node->result_id;
|
|
p_parser->physical_pointer_count = 0;
|
|
SpvReflectResult result = ParseDescriptorBlockVariable(p_parser, p_module, p_type, p_push_constant);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
|
|
for (uint32_t access_chain_index = 0; access_chain_index < p_parser->access_chain_count; ++access_chain_index) {
|
|
SpvReflectPrvAccessChain* p_access_chain = &(p_parser->access_chains[access_chain_index]);
|
|
// Skip any access chains that aren't touching this push constant block
|
|
if (p_push_constant->spirv_id != FindAccessChainBaseVariable(p_parser, p_access_chain)) {
|
|
continue;
|
|
}
|
|
SpvReflectBlockVariable* p_var =
|
|
(p_access_chain->base_id == p_push_constant->spirv_id) ? p_push_constant : GetRefBlkVar(p_parser, p_access_chain);
|
|
result = ParseDescriptorBlockVariableUsage(p_parser, p_module, p_access_chain, 0, (SpvOp)INVALID_VALUE, p_var);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
}
|
|
|
|
p_push_constant->name = p_node->name;
|
|
result = ParseDescriptorBlockVariableSizes(p_parser, p_module, true, false, false, p_push_constant);
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
return result;
|
|
}
|
|
|
|
// Get minimum offset for whole Push Constant block
|
|
// It is not valid SPIR-V to have an empty Push Constant Block
|
|
p_push_constant->offset = UINT32_MAX;
|
|
for (uint32_t k = 0; k < p_push_constant->member_count; ++k) {
|
|
const uint32_t member_offset = p_push_constant->members[k].offset;
|
|
p_push_constant->offset = Min(p_push_constant->offset, member_offset);
|
|
}
|
|
|
|
++push_constant_index;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static int SortCompareDescriptorSet(const void* a, const void* b) {
|
|
const SpvReflectDescriptorSet* p_elem_a = (const SpvReflectDescriptorSet*)a;
|
|
const SpvReflectDescriptorSet* p_elem_b = (const SpvReflectDescriptorSet*)b;
|
|
int value = (int)(p_elem_a->set) - (int)(p_elem_b->set);
|
|
// We should never see duplicate descriptor set numbers in a shader; if so, a tiebreaker
|
|
// would be needed here.
|
|
assert(value != 0);
|
|
return value;
|
|
}
|
|
|
|
static SpvReflectResult ParseEntrypointDescriptorSets(SpvReflectShaderModule* p_module) {
|
|
// Update the entry point's sets
|
|
for (uint32_t i = 0; i < p_module->entry_point_count; ++i) {
|
|
SpvReflectEntryPoint* p_entry = &p_module->entry_points[i];
|
|
for (uint32_t j = 0; j < p_entry->descriptor_set_count; ++j) {
|
|
SafeFree(p_entry->descriptor_sets[j].bindings);
|
|
}
|
|
SafeFree(p_entry->descriptor_sets);
|
|
p_entry->descriptor_set_count = 0;
|
|
for (uint32_t j = 0; j < p_module->descriptor_set_count; ++j) {
|
|
const SpvReflectDescriptorSet* p_set = &p_module->descriptor_sets[j];
|
|
for (uint32_t k = 0; k < p_set->binding_count; ++k) {
|
|
bool found = SearchSortedUint32(p_entry->used_uniforms, p_entry->used_uniform_count, p_set->bindings[k]->spirv_id);
|
|
if (found) {
|
|
++p_entry->descriptor_set_count;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
p_entry->descriptor_sets = NULL;
|
|
if (p_entry->descriptor_set_count > 0) {
|
|
p_entry->descriptor_sets = (SpvReflectDescriptorSet*)calloc(p_entry->descriptor_set_count, sizeof(*p_entry->descriptor_sets));
|
|
if (IsNull(p_entry->descriptor_sets)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
}
|
|
p_entry->descriptor_set_count = 0;
|
|
for (uint32_t j = 0; j < p_module->descriptor_set_count; ++j) {
|
|
const SpvReflectDescriptorSet* p_set = &p_module->descriptor_sets[j];
|
|
uint32_t count = 0;
|
|
for (uint32_t k = 0; k < p_set->binding_count; ++k) {
|
|
bool found = SearchSortedUint32(p_entry->used_uniforms, p_entry->used_uniform_count, p_set->bindings[k]->spirv_id);
|
|
if (found) {
|
|
++count;
|
|
}
|
|
}
|
|
if (count == 0) {
|
|
continue;
|
|
}
|
|
SpvReflectDescriptorSet* p_entry_set = &p_entry->descriptor_sets[p_entry->descriptor_set_count++];
|
|
p_entry_set->set = p_set->set;
|
|
p_entry_set->bindings = (SpvReflectDescriptorBinding**)calloc(count, sizeof(*p_entry_set->bindings));
|
|
if (IsNull(p_entry_set->bindings)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
for (uint32_t k = 0; k < p_set->binding_count; ++k) {
|
|
bool found = SearchSortedUint32(p_entry->used_uniforms, p_entry->used_uniform_count, p_set->bindings[k]->spirv_id);
|
|
if (found) {
|
|
p_entry_set->bindings[p_entry_set->binding_count++] = p_set->bindings[k];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult ParseDescriptorSets(SpvReflectShaderModule* p_module) {
|
|
// Count the descriptors in each set
|
|
for (uint32_t i = 0; i < p_module->descriptor_binding_count; ++i) {
|
|
SpvReflectDescriptorBinding* p_descriptor = &(p_module->descriptor_bindings[i]);
|
|
|
|
// Look for a target set using the descriptor's set number
|
|
SpvReflectDescriptorSet* p_target_set = NULL;
|
|
for (uint32_t j = 0; j < SPV_REFLECT_MAX_DESCRIPTOR_SETS; ++j) {
|
|
SpvReflectDescriptorSet* p_set = &p_module->descriptor_sets[j];
|
|
if (p_set->set == p_descriptor->set) {
|
|
p_target_set = p_set;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// If a target set isn't found, find the first available one.
|
|
if (IsNull(p_target_set)) {
|
|
for (uint32_t j = 0; j < SPV_REFLECT_MAX_DESCRIPTOR_SETS; ++j) {
|
|
SpvReflectDescriptorSet* p_set = &p_module->descriptor_sets[j];
|
|
if (p_set->set == (uint32_t)INVALID_VALUE) {
|
|
p_target_set = p_set;
|
|
p_target_set->set = p_descriptor->set;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (IsNull(p_target_set)) {
|
|
return SPV_REFLECT_RESULT_ERROR_INTERNAL_ERROR;
|
|
}
|
|
|
|
p_target_set->binding_count += 1;
|
|
}
|
|
|
|
// Count the descriptor sets
|
|
for (uint32_t i = 0; i < SPV_REFLECT_MAX_DESCRIPTOR_SETS; ++i) {
|
|
const SpvReflectDescriptorSet* p_set = &p_module->descriptor_sets[i];
|
|
if (p_set->set != (uint32_t)INVALID_VALUE) {
|
|
p_module->descriptor_set_count += 1;
|
|
}
|
|
}
|
|
|
|
// Sort the descriptor sets based on numbers
|
|
if (p_module->descriptor_set_count > 0) {
|
|
qsort(p_module->descriptor_sets, p_module->descriptor_set_count, sizeof(*(p_module->descriptor_sets)),
|
|
SortCompareDescriptorSet);
|
|
}
|
|
|
|
// Build descriptor pointer array
|
|
for (uint32_t i = 0; i < p_module->descriptor_set_count; ++i) {
|
|
SpvReflectDescriptorSet* p_set = &(p_module->descriptor_sets[i]);
|
|
p_set->bindings = (SpvReflectDescriptorBinding**)calloc(p_set->binding_count, sizeof(*(p_set->bindings)));
|
|
|
|
uint32_t descriptor_index = 0;
|
|
for (uint32_t j = 0; j < p_module->descriptor_binding_count; ++j) {
|
|
SpvReflectDescriptorBinding* p_descriptor = &(p_module->descriptor_bindings[j]);
|
|
if (p_descriptor->set == p_set->set) {
|
|
assert(descriptor_index < p_set->binding_count);
|
|
p_set->bindings[descriptor_index] = p_descriptor;
|
|
++descriptor_index;
|
|
}
|
|
}
|
|
}
|
|
|
|
return ParseEntrypointDescriptorSets(p_module);
|
|
}
|
|
|
|
static SpvReflectResult DisambiguateStorageBufferSrvUav(SpvReflectShaderModule* p_module) {
|
|
if (p_module->descriptor_binding_count == 0) {
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
for (uint32_t descriptor_index = 0; descriptor_index < p_module->descriptor_binding_count; ++descriptor_index) {
|
|
SpvReflectDescriptorBinding* p_descriptor = &(p_module->descriptor_bindings[descriptor_index]);
|
|
// Skip everything that isn't a STORAGE_BUFFER descriptor
|
|
if (p_descriptor->descriptor_type != SPV_REFLECT_DESCRIPTOR_TYPE_STORAGE_BUFFER) {
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// Vulkan doesn't disambiguate between SRVs and UAVs so they
|
|
// come back as STORAGE_BUFFER. The block parsing process will
|
|
// mark a block as non-writable should any member of the block
|
|
// or its descendants are non-writable.
|
|
//
|
|
if (p_descriptor->block.decoration_flags & SPV_REFLECT_DECORATION_NON_WRITABLE) {
|
|
p_descriptor->resource_type = SPV_REFLECT_RESOURCE_FLAG_SRV;
|
|
}
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
static SpvReflectResult SynchronizeDescriptorSets(SpvReflectShaderModule* p_module) {
|
|
// Free and reset all descriptor set numbers
|
|
for (uint32_t i = 0; i < SPV_REFLECT_MAX_DESCRIPTOR_SETS; ++i) {
|
|
SpvReflectDescriptorSet* p_set = &p_module->descriptor_sets[i];
|
|
SafeFree(p_set->bindings);
|
|
p_set->binding_count = 0;
|
|
p_set->set = (uint32_t)INVALID_VALUE;
|
|
}
|
|
// Set descriptor set count to zero
|
|
p_module->descriptor_set_count = 0;
|
|
|
|
SpvReflectResult result = ParseDescriptorSets(p_module);
|
|
return result;
|
|
}
|
|
|
|
static SpvReflectResult CreateShaderModule(uint32_t flags, size_t size, const void* p_code, SpvReflectShaderModule* p_module) {
|
|
// Initialize all module fields to zero
|
|
memset(p_module, 0, sizeof(*p_module));
|
|
|
|
// Allocate module internals
|
|
#ifdef __cplusplus
|
|
p_module->_internal = (SpvReflectShaderModule::Internal*)calloc(1, sizeof(*(p_module->_internal)));
|
|
#else
|
|
p_module->_internal = calloc(1, sizeof(*(p_module->_internal)));
|
|
#endif
|
|
if (IsNull(p_module->_internal)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
// Copy flags
|
|
p_module->_internal->module_flags = flags;
|
|
// Figure out if we need to copy the SPIR-V code or not
|
|
if (flags & SPV_REFLECT_MODULE_FLAG_NO_COPY) {
|
|
// Set internal size and pointer to args passed in
|
|
p_module->_internal->spirv_size = size;
|
|
#if defined(__cplusplus)
|
|
p_module->_internal->spirv_code = const_cast<uint32_t*>(static_cast<const uint32_t*>(p_code)); // cast that const away
|
|
#else
|
|
p_module->_internal->spirv_code = (void*)p_code; // cast that const away
|
|
#endif
|
|
p_module->_internal->spirv_word_count = (uint32_t)(size / SPIRV_WORD_SIZE);
|
|
} else {
|
|
// Allocate SPIR-V code storage
|
|
p_module->_internal->spirv_size = size;
|
|
p_module->_internal->spirv_code = (uint32_t*)calloc(1, p_module->_internal->spirv_size);
|
|
p_module->_internal->spirv_word_count = (uint32_t)(size / SPIRV_WORD_SIZE);
|
|
if (IsNull(p_module->_internal->spirv_code)) {
|
|
SafeFree(p_module->_internal);
|
|
return SPV_REFLECT_RESULT_ERROR_ALLOC_FAILED;
|
|
}
|
|
// Copy SPIR-V to code storage
|
|
memcpy(p_module->_internal->spirv_code, p_code, size);
|
|
}
|
|
|
|
// Initialize everything to zero
|
|
SpvReflectPrvParser parser;
|
|
memset(&parser, 0, sizeof(SpvReflectPrvParser));
|
|
|
|
// Create parser
|
|
SpvReflectResult result = CreateParser(p_module->_internal->spirv_size, p_module->_internal->spirv_code, &parser);
|
|
|
|
// Generator
|
|
{
|
|
const uint32_t* p_ptr = (const uint32_t*)p_module->_internal->spirv_code;
|
|
p_module->generator = (SpvReflectGenerator)((*(p_ptr + 2) & 0xFFFF0000) >> 16);
|
|
}
|
|
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = ParseNodes(&parser);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = ParseStrings(&parser);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = ParseSource(&parser, p_module);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = ParseFunctions(&parser);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = ParseMemberCounts(&parser);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = ParseNames(&parser);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = ParseDecorations(&parser, p_module);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
|
|
// Start of reflection data parsing
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
p_module->source_language = parser.source_language;
|
|
p_module->source_language_version = parser.source_language_version;
|
|
|
|
// Zero out descriptor set data
|
|
p_module->descriptor_set_count = 0;
|
|
memset(p_module->descriptor_sets, 0, SPV_REFLECT_MAX_DESCRIPTOR_SETS * sizeof(*p_module->descriptor_sets));
|
|
// Initialize descriptor set numbers
|
|
for (uint32_t set_number = 0; set_number < SPV_REFLECT_MAX_DESCRIPTOR_SETS; ++set_number) {
|
|
p_module->descriptor_sets[set_number].set = (uint32_t)INVALID_VALUE;
|
|
}
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = ParseTypes(&parser, p_module);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = ParseDescriptorBindings(&parser, p_module);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = ParseDescriptorType(p_module);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = ParseUAVCounterBindings(p_module);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = ParseDescriptorBlocks(&parser, p_module);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = ParsePushConstantBlocks(&parser, p_module);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = ParseEntryPoints(&parser, p_module);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = ParseCapabilities(&parser, p_module);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS && p_module->entry_point_count > 0) {
|
|
SpvReflectEntryPoint* p_entry = &(p_module->entry_points[0]);
|
|
p_module->entry_point_name = p_entry->name;
|
|
p_module->entry_point_id = p_entry->id;
|
|
p_module->spirv_execution_model = p_entry->spirv_execution_model;
|
|
p_module->shader_stage = p_entry->shader_stage;
|
|
p_module->input_variable_count = p_entry->input_variable_count;
|
|
p_module->input_variables = p_entry->input_variables;
|
|
p_module->output_variable_count = p_entry->output_variable_count;
|
|
p_module->output_variables = p_entry->output_variables;
|
|
p_module->interface_variable_count = p_entry->interface_variable_count;
|
|
p_module->interface_variables = p_entry->interface_variables;
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = DisambiguateStorageBufferSrvUav(p_module);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = SynchronizeDescriptorSets(p_module);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
if (result == SPV_REFLECT_RESULT_SUCCESS) {
|
|
result = ParseExecutionModes(&parser, p_module);
|
|
SPV_REFLECT_ASSERT(result == SPV_REFLECT_RESULT_SUCCESS);
|
|
}
|
|
|
|
// Destroy module if parse was not successful
|
|
if (result != SPV_REFLECT_RESULT_SUCCESS) {
|
|
spvReflectDestroyShaderModule(p_module);
|
|
}
|
|
|
|
DestroyParser(&parser);
|
|
|
|
return result;
|
|
}
|
|
|
|
SpvReflectResult spvReflectCreateShaderModule(size_t size, const void* p_code, SpvReflectShaderModule* p_module) {
|
|
return CreateShaderModule(0, size, p_code, p_module);
|
|
}
|
|
|
|
SpvReflectResult spvReflectCreateShaderModule2(uint32_t flags, size_t size, const void* p_code, SpvReflectShaderModule* p_module) {
|
|
return CreateShaderModule(flags, size, p_code, p_module);
|
|
}
|
|
|
|
SpvReflectResult spvReflectGetShaderModule(size_t size, const void* p_code, SpvReflectShaderModule* p_module) {
|
|
return spvReflectCreateShaderModule(size, p_code, p_module);
|
|
}
|
|
|
|
static void SafeFreeTypes(SpvReflectTypeDescription* p_type) {
|
|
if (IsNull(p_type) || p_type->copied) {
|
|
return;
|
|
}
|
|
|
|
if (IsNotNull(p_type->members)) {
|
|
for (size_t i = 0; i < p_type->member_count; ++i) {
|
|
SpvReflectTypeDescription* p_member = &p_type->members[i];
|
|
SafeFreeTypes(p_member);
|
|
}
|
|
|
|
SafeFree(p_type->members);
|
|
p_type->members = NULL;
|
|
}
|
|
}
|
|
|
|
static void SafeFreeBlockVariables(SpvReflectBlockVariable* p_block) {
|
|
if (IsNull(p_block)) {
|
|
return;
|
|
}
|
|
|
|
// We share pointers to Physical Pointer structs and don't want to double free
|
|
if (p_block->flags & SPV_REFLECT_VARIABLE_FLAGS_PHYSICAL_POINTER_COPY) {
|
|
return;
|
|
}
|
|
|
|
if (IsNotNull(p_block->members)) {
|
|
for (size_t i = 0; i < p_block->member_count; ++i) {
|
|
SpvReflectBlockVariable* p_member = &p_block->members[i];
|
|
SafeFreeBlockVariables(p_member);
|
|
}
|
|
|
|
SafeFree(p_block->members);
|
|
p_block->members = NULL;
|
|
}
|
|
}
|
|
|
|
static void SafeFreeInterfaceVariable(SpvReflectInterfaceVariable* p_interface) {
|
|
if (IsNull(p_interface)) {
|
|
return;
|
|
}
|
|
|
|
if (IsNotNull(p_interface->members)) {
|
|
for (size_t i = 0; i < p_interface->member_count; ++i) {
|
|
SpvReflectInterfaceVariable* p_member = &p_interface->members[i];
|
|
SafeFreeInterfaceVariable(p_member);
|
|
}
|
|
|
|
SafeFree(p_interface->members);
|
|
p_interface->members = NULL;
|
|
}
|
|
}
|
|
|
|
void spvReflectDestroyShaderModule(SpvReflectShaderModule* p_module) {
|
|
if (IsNull(p_module->_internal)) {
|
|
return;
|
|
}
|
|
|
|
SafeFree(p_module->source_source);
|
|
|
|
// Descriptor set bindings
|
|
for (size_t i = 0; i < p_module->descriptor_set_count; ++i) {
|
|
SpvReflectDescriptorSet* p_set = &p_module->descriptor_sets[i];
|
|
free(p_set->bindings);
|
|
}
|
|
|
|
// Descriptor binding blocks
|
|
for (size_t i = 0; i < p_module->descriptor_binding_count; ++i) {
|
|
SpvReflectDescriptorBinding* p_descriptor = &p_module->descriptor_bindings[i];
|
|
if (IsNotNull(p_descriptor->byte_address_buffer_offsets)) {
|
|
SafeFree(p_descriptor->byte_address_buffer_offsets);
|
|
}
|
|
SafeFreeBlockVariables(&p_descriptor->block);
|
|
}
|
|
SafeFree(p_module->descriptor_bindings);
|
|
|
|
// Entry points
|
|
for (size_t i = 0; i < p_module->entry_point_count; ++i) {
|
|
SpvReflectEntryPoint* p_entry = &p_module->entry_points[i];
|
|
for (size_t j = 0; j < p_entry->interface_variable_count; j++) {
|
|
SafeFreeInterfaceVariable(&p_entry->interface_variables[j]);
|
|
}
|
|
for (uint32_t j = 0; j < p_entry->descriptor_set_count; ++j) {
|
|
SafeFree(p_entry->descriptor_sets[j].bindings);
|
|
}
|
|
SafeFree(p_entry->descriptor_sets);
|
|
SafeFree(p_entry->input_variables);
|
|
SafeFree(p_entry->output_variables);
|
|
SafeFree(p_entry->interface_variables);
|
|
SafeFree(p_entry->used_uniforms);
|
|
SafeFree(p_entry->used_push_constants);
|
|
SafeFree(p_entry->execution_modes);
|
|
}
|
|
SafeFree(p_module->capabilities);
|
|
SafeFree(p_module->entry_points);
|
|
SafeFree(p_module->spec_constants);
|
|
|
|
// Push constants
|
|
for (size_t i = 0; i < p_module->push_constant_block_count; ++i) {
|
|
SafeFreeBlockVariables(&p_module->push_constant_blocks[i]);
|
|
}
|
|
SafeFree(p_module->push_constant_blocks);
|
|
|
|
// Type infos
|
|
for (size_t i = 0; i < p_module->_internal->type_description_count; ++i) {
|
|
SpvReflectTypeDescription* p_type = &p_module->_internal->type_descriptions[i];
|
|
if (IsNotNull(p_type->members)) {
|
|
SafeFreeTypes(p_type);
|
|
}
|
|
SafeFree(p_type->members);
|
|
}
|
|
SafeFree(p_module->_internal->type_descriptions);
|
|
|
|
// Free SPIR-V code if there was a copy
|
|
if ((p_module->_internal->module_flags & SPV_REFLECT_MODULE_FLAG_NO_COPY) == 0) {
|
|
SafeFree(p_module->_internal->spirv_code);
|
|
}
|
|
// Free internal
|
|
SafeFree(p_module->_internal);
|
|
}
|
|
|
|
uint32_t spvReflectGetCodeSize(const SpvReflectShaderModule* p_module) {
|
|
if (IsNull(p_module)) {
|
|
return 0;
|
|
}
|
|
|
|
return (uint32_t)(p_module->_internal->spirv_size);
|
|
}
|
|
|
|
const uint32_t* spvReflectGetCode(const SpvReflectShaderModule* p_module) {
|
|
if (IsNull(p_module)) {
|
|
return NULL;
|
|
}
|
|
|
|
return p_module->_internal->spirv_code;
|
|
}
|
|
|
|
const SpvReflectEntryPoint* spvReflectGetEntryPoint(const SpvReflectShaderModule* p_module, const char* entry_point) {
|
|
if (IsNull(p_module) || IsNull(entry_point)) {
|
|
return NULL;
|
|
}
|
|
|
|
for (uint32_t i = 0; i < p_module->entry_point_count; ++i) {
|
|
if (strcmp(p_module->entry_points[i].name, entry_point) == 0) {
|
|
return &p_module->entry_points[i];
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
SpvReflectResult spvReflectEnumerateDescriptorBindings(const SpvReflectShaderModule* p_module, uint32_t* p_count,
|
|
SpvReflectDescriptorBinding** pp_bindings) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_count)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
if (IsNotNull(pp_bindings)) {
|
|
if (*p_count != p_module->descriptor_binding_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_COUNT_MISMATCH;
|
|
}
|
|
|
|
for (uint32_t index = 0; index < *p_count; ++index) {
|
|
SpvReflectDescriptorBinding* p_bindings = (SpvReflectDescriptorBinding*)&p_module->descriptor_bindings[index];
|
|
pp_bindings[index] = p_bindings;
|
|
}
|
|
} else {
|
|
*p_count = p_module->descriptor_binding_count;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
SpvReflectResult spvReflectEnumerateEntryPointDescriptorBindings(const SpvReflectShaderModule* p_module, const char* entry_point,
|
|
uint32_t* p_count, SpvReflectDescriptorBinding** pp_bindings) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_count)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
const SpvReflectEntryPoint* p_entry = spvReflectGetEntryPoint(p_module, entry_point);
|
|
if (IsNull(p_entry)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
|
|
uint32_t count = 0;
|
|
for (uint32_t i = 0; i < p_module->descriptor_binding_count; ++i) {
|
|
bool found = SearchSortedUint32(p_entry->used_uniforms, p_entry->used_uniform_count, p_module->descriptor_bindings[i].spirv_id);
|
|
if (found) {
|
|
if (IsNotNull(pp_bindings)) {
|
|
if (count >= *p_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_COUNT_MISMATCH;
|
|
}
|
|
pp_bindings[count++] = (SpvReflectDescriptorBinding*)&p_module->descriptor_bindings[i];
|
|
} else {
|
|
++count;
|
|
}
|
|
}
|
|
}
|
|
if (IsNotNull(pp_bindings)) {
|
|
if (count != *p_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_COUNT_MISMATCH;
|
|
}
|
|
} else {
|
|
*p_count = count;
|
|
}
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
SpvReflectResult spvReflectEnumerateDescriptorSets(const SpvReflectShaderModule* p_module, uint32_t* p_count,
|
|
SpvReflectDescriptorSet** pp_sets) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_count)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
if (IsNotNull(pp_sets)) {
|
|
if (*p_count != p_module->descriptor_set_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_COUNT_MISMATCH;
|
|
}
|
|
|
|
for (uint32_t index = 0; index < *p_count; ++index) {
|
|
SpvReflectDescriptorSet* p_set = (SpvReflectDescriptorSet*)&p_module->descriptor_sets[index];
|
|
pp_sets[index] = p_set;
|
|
}
|
|
} else {
|
|
*p_count = p_module->descriptor_set_count;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
SpvReflectResult spvReflectEnumerateEntryPointDescriptorSets(const SpvReflectShaderModule* p_module, const char* entry_point,
|
|
uint32_t* p_count, SpvReflectDescriptorSet** pp_sets) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_count)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
const SpvReflectEntryPoint* p_entry = spvReflectGetEntryPoint(p_module, entry_point);
|
|
if (IsNull(p_entry)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
|
|
if (IsNotNull(pp_sets)) {
|
|
if (*p_count != p_entry->descriptor_set_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_COUNT_MISMATCH;
|
|
}
|
|
|
|
for (uint32_t index = 0; index < *p_count; ++index) {
|
|
SpvReflectDescriptorSet* p_set = (SpvReflectDescriptorSet*)&p_entry->descriptor_sets[index];
|
|
pp_sets[index] = p_set;
|
|
}
|
|
} else {
|
|
*p_count = p_entry->descriptor_set_count;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
SpvReflectResult spvReflectEnumerateInterfaceVariables(const SpvReflectShaderModule* p_module, uint32_t* p_count,
|
|
SpvReflectInterfaceVariable** pp_variables) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_count)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
if (IsNotNull(pp_variables)) {
|
|
if (*p_count != p_module->interface_variable_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_COUNT_MISMATCH;
|
|
}
|
|
|
|
for (uint32_t index = 0; index < *p_count; ++index) {
|
|
SpvReflectInterfaceVariable* p_var = &p_module->interface_variables[index];
|
|
pp_variables[index] = p_var;
|
|
}
|
|
} else {
|
|
*p_count = p_module->interface_variable_count;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
SpvReflectResult spvReflectEnumerateEntryPointInterfaceVariables(const SpvReflectShaderModule* p_module, const char* entry_point,
|
|
uint32_t* p_count, SpvReflectInterfaceVariable** pp_variables) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_count)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
const SpvReflectEntryPoint* p_entry = spvReflectGetEntryPoint(p_module, entry_point);
|
|
if (IsNull(p_entry)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
|
|
if (IsNotNull(pp_variables)) {
|
|
if (*p_count != p_entry->interface_variable_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_COUNT_MISMATCH;
|
|
}
|
|
|
|
for (uint32_t index = 0; index < *p_count; ++index) {
|
|
SpvReflectInterfaceVariable* p_var = &p_entry->interface_variables[index];
|
|
pp_variables[index] = p_var;
|
|
}
|
|
} else {
|
|
*p_count = p_entry->interface_variable_count;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
SpvReflectResult spvReflectEnumerateInputVariables(const SpvReflectShaderModule* p_module, uint32_t* p_count,
|
|
SpvReflectInterfaceVariable** pp_variables) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_count)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
if (IsNotNull(pp_variables)) {
|
|
if (*p_count != p_module->input_variable_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_COUNT_MISMATCH;
|
|
}
|
|
|
|
for (uint32_t index = 0; index < *p_count; ++index) {
|
|
SpvReflectInterfaceVariable* p_var = p_module->input_variables[index];
|
|
pp_variables[index] = p_var;
|
|
}
|
|
} else {
|
|
*p_count = p_module->input_variable_count;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
SpvReflectResult spvReflectEnumerateEntryPointInputVariables(const SpvReflectShaderModule* p_module, const char* entry_point,
|
|
uint32_t* p_count, SpvReflectInterfaceVariable** pp_variables) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_count)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
const SpvReflectEntryPoint* p_entry = spvReflectGetEntryPoint(p_module, entry_point);
|
|
if (IsNull(p_entry)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
|
|
if (IsNotNull(pp_variables)) {
|
|
if (*p_count != p_entry->input_variable_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_COUNT_MISMATCH;
|
|
}
|
|
|
|
for (uint32_t index = 0; index < *p_count; ++index) {
|
|
SpvReflectInterfaceVariable* p_var = p_entry->input_variables[index];
|
|
pp_variables[index] = p_var;
|
|
}
|
|
} else {
|
|
*p_count = p_entry->input_variable_count;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
SpvReflectResult spvReflectEnumerateOutputVariables(const SpvReflectShaderModule* p_module, uint32_t* p_count,
|
|
SpvReflectInterfaceVariable** pp_variables) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_count)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
if (IsNotNull(pp_variables)) {
|
|
if (*p_count != p_module->output_variable_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_COUNT_MISMATCH;
|
|
}
|
|
|
|
for (uint32_t index = 0; index < *p_count; ++index) {
|
|
SpvReflectInterfaceVariable* p_var = p_module->output_variables[index];
|
|
pp_variables[index] = p_var;
|
|
}
|
|
} else {
|
|
*p_count = p_module->output_variable_count;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
SpvReflectResult spvReflectEnumerateEntryPointOutputVariables(const SpvReflectShaderModule* p_module, const char* entry_point,
|
|
uint32_t* p_count, SpvReflectInterfaceVariable** pp_variables) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_count)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
const SpvReflectEntryPoint* p_entry = spvReflectGetEntryPoint(p_module, entry_point);
|
|
if (IsNull(p_entry)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
|
|
if (IsNotNull(pp_variables)) {
|
|
if (*p_count != p_entry->output_variable_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_COUNT_MISMATCH;
|
|
}
|
|
|
|
for (uint32_t index = 0; index < *p_count; ++index) {
|
|
SpvReflectInterfaceVariable* p_var = p_entry->output_variables[index];
|
|
pp_variables[index] = p_var;
|
|
}
|
|
} else {
|
|
*p_count = p_entry->output_variable_count;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
SpvReflectResult spvReflectEnumeratePushConstantBlocks(const SpvReflectShaderModule* p_module, uint32_t* p_count,
|
|
SpvReflectBlockVariable** pp_blocks) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_count)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
if (pp_blocks != NULL) {
|
|
if (*p_count != p_module->push_constant_block_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_COUNT_MISMATCH;
|
|
}
|
|
|
|
for (uint32_t index = 0; index < *p_count; ++index) {
|
|
SpvReflectBlockVariable* p_push_constant_blocks = (SpvReflectBlockVariable*)&p_module->push_constant_blocks[index];
|
|
pp_blocks[index] = p_push_constant_blocks;
|
|
}
|
|
} else {
|
|
*p_count = p_module->push_constant_block_count;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
SpvReflectResult spvReflectEnumeratePushConstants(const SpvReflectShaderModule* p_module, uint32_t* p_count,
|
|
SpvReflectBlockVariable** pp_blocks) {
|
|
return spvReflectEnumeratePushConstantBlocks(p_module, p_count, pp_blocks);
|
|
}
|
|
|
|
SpvReflectResult spvReflectEnumerateEntryPointPushConstantBlocks(const SpvReflectShaderModule* p_module, const char* entry_point,
|
|
uint32_t* p_count, SpvReflectBlockVariable** pp_blocks) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_count)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
const SpvReflectEntryPoint* p_entry = spvReflectGetEntryPoint(p_module, entry_point);
|
|
if (IsNull(p_entry)) {
|
|
return SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
|
|
uint32_t count = 0;
|
|
for (uint32_t i = 0; i < p_module->push_constant_block_count; ++i) {
|
|
bool found = SearchSortedUint32(p_entry->used_push_constants, p_entry->used_push_constant_count,
|
|
p_module->push_constant_blocks[i].spirv_id);
|
|
if (found) {
|
|
if (IsNotNull(pp_blocks)) {
|
|
if (count >= *p_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_COUNT_MISMATCH;
|
|
}
|
|
pp_blocks[count++] = (SpvReflectBlockVariable*)&p_module->push_constant_blocks[i];
|
|
} else {
|
|
++count;
|
|
}
|
|
}
|
|
}
|
|
if (IsNotNull(pp_blocks)) {
|
|
if (count != *p_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_COUNT_MISMATCH;
|
|
}
|
|
} else {
|
|
*p_count = count;
|
|
}
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
SpvReflectResult spvReflectEnumerateSpecializationConstants(const SpvReflectShaderModule* p_module, uint32_t* p_count,
|
|
SpvReflectSpecializationConstant** pp_constants) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_count)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
if (IsNotNull(pp_constants)) {
|
|
if (*p_count != p_module->spec_constant_count) {
|
|
return SPV_REFLECT_RESULT_ERROR_COUNT_MISMATCH;
|
|
}
|
|
|
|
for (uint32_t index = 0; index < *p_count; ++index) {
|
|
SpvReflectSpecializationConstant* p_constant = (SpvReflectSpecializationConstant*)&p_module->spec_constants[index];
|
|
pp_constants[index] = p_constant;
|
|
}
|
|
} else {
|
|
*p_count = p_module->spec_constant_count;
|
|
}
|
|
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
const SpvReflectDescriptorBinding* spvReflectGetDescriptorBinding(const SpvReflectShaderModule* p_module, uint32_t binding_number,
|
|
uint32_t set_number, SpvReflectResult* p_result) {
|
|
const SpvReflectDescriptorBinding* p_descriptor = NULL;
|
|
if (IsNotNull(p_module)) {
|
|
for (uint32_t index = 0; index < p_module->descriptor_binding_count; ++index) {
|
|
const SpvReflectDescriptorBinding* p_potential = &p_module->descriptor_bindings[index];
|
|
if ((p_potential->binding == binding_number) && (p_potential->set == set_number)) {
|
|
p_descriptor = p_potential;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = IsNotNull(p_descriptor)
|
|
? SPV_REFLECT_RESULT_SUCCESS
|
|
: (IsNull(p_module) ? SPV_REFLECT_RESULT_ERROR_NULL_POINTER : SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND);
|
|
}
|
|
return p_descriptor;
|
|
}
|
|
|
|
const SpvReflectDescriptorBinding* spvReflectGetEntryPointDescriptorBinding(const SpvReflectShaderModule* p_module,
|
|
const char* entry_point, uint32_t binding_number,
|
|
uint32_t set_number, SpvReflectResult* p_result) {
|
|
const SpvReflectEntryPoint* p_entry = spvReflectGetEntryPoint(p_module, entry_point);
|
|
if (IsNull(p_entry)) {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
return NULL;
|
|
}
|
|
const SpvReflectDescriptorBinding* p_descriptor = NULL;
|
|
if (IsNotNull(p_module)) {
|
|
for (uint32_t index = 0; index < p_module->descriptor_binding_count; ++index) {
|
|
const SpvReflectDescriptorBinding* p_potential = &p_module->descriptor_bindings[index];
|
|
bool found = SearchSortedUint32(p_entry->used_uniforms, p_entry->used_uniform_count, p_potential->spirv_id);
|
|
if ((p_potential->binding == binding_number) && (p_potential->set == set_number) && found) {
|
|
p_descriptor = p_potential;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = IsNotNull(p_descriptor)
|
|
? SPV_REFLECT_RESULT_SUCCESS
|
|
: (IsNull(p_module) ? SPV_REFLECT_RESULT_ERROR_NULL_POINTER : SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND);
|
|
}
|
|
return p_descriptor;
|
|
}
|
|
|
|
const SpvReflectDescriptorSet* spvReflectGetDescriptorSet(const SpvReflectShaderModule* p_module, uint32_t set_number,
|
|
SpvReflectResult* p_result) {
|
|
const SpvReflectDescriptorSet* p_set = NULL;
|
|
if (IsNotNull(p_module)) {
|
|
for (uint32_t index = 0; index < p_module->descriptor_set_count; ++index) {
|
|
const SpvReflectDescriptorSet* p_potential = &p_module->descriptor_sets[index];
|
|
if (p_potential->set == set_number) {
|
|
p_set = p_potential;
|
|
}
|
|
}
|
|
}
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = IsNotNull(p_set)
|
|
? SPV_REFLECT_RESULT_SUCCESS
|
|
: (IsNull(p_module) ? SPV_REFLECT_RESULT_ERROR_NULL_POINTER : SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND);
|
|
}
|
|
return p_set;
|
|
}
|
|
|
|
const SpvReflectDescriptorSet* spvReflectGetEntryPointDescriptorSet(const SpvReflectShaderModule* p_module, const char* entry_point,
|
|
uint32_t set_number, SpvReflectResult* p_result) {
|
|
const SpvReflectDescriptorSet* p_set = NULL;
|
|
if (IsNotNull(p_module)) {
|
|
const SpvReflectEntryPoint* p_entry = spvReflectGetEntryPoint(p_module, entry_point);
|
|
if (IsNull(p_entry)) {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
return NULL;
|
|
}
|
|
for (uint32_t index = 0; index < p_entry->descriptor_set_count; ++index) {
|
|
const SpvReflectDescriptorSet* p_potential = &p_entry->descriptor_sets[index];
|
|
if (p_potential->set == set_number) {
|
|
p_set = p_potential;
|
|
}
|
|
}
|
|
}
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = IsNotNull(p_set)
|
|
? SPV_REFLECT_RESULT_SUCCESS
|
|
: (IsNull(p_module) ? SPV_REFLECT_RESULT_ERROR_NULL_POINTER : SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND);
|
|
}
|
|
return p_set;
|
|
}
|
|
|
|
const SpvReflectInterfaceVariable* spvReflectGetInputVariableByLocation(const SpvReflectShaderModule* p_module, uint32_t location,
|
|
SpvReflectResult* p_result) {
|
|
if (location == INVALID_VALUE) {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
return NULL;
|
|
}
|
|
const SpvReflectInterfaceVariable* p_var = NULL;
|
|
if (IsNotNull(p_module)) {
|
|
for (uint32_t index = 0; index < p_module->input_variable_count; ++index) {
|
|
const SpvReflectInterfaceVariable* p_potential = p_module->input_variables[index];
|
|
if (p_potential->location == location) {
|
|
p_var = p_potential;
|
|
}
|
|
}
|
|
}
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = IsNotNull(p_var)
|
|
? SPV_REFLECT_RESULT_SUCCESS
|
|
: (IsNull(p_module) ? SPV_REFLECT_RESULT_ERROR_NULL_POINTER : SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND);
|
|
}
|
|
return p_var;
|
|
}
|
|
const SpvReflectInterfaceVariable* spvReflectGetInputVariable(const SpvReflectShaderModule* p_module, uint32_t location,
|
|
SpvReflectResult* p_result) {
|
|
return spvReflectGetInputVariableByLocation(p_module, location, p_result);
|
|
}
|
|
|
|
const SpvReflectInterfaceVariable* spvReflectGetEntryPointInputVariableByLocation(const SpvReflectShaderModule* p_module,
|
|
const char* entry_point, uint32_t location,
|
|
SpvReflectResult* p_result) {
|
|
if (location == INVALID_VALUE) {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
const SpvReflectInterfaceVariable* p_var = NULL;
|
|
if (IsNotNull(p_module)) {
|
|
const SpvReflectEntryPoint* p_entry = spvReflectGetEntryPoint(p_module, entry_point);
|
|
if (IsNull(p_entry)) {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
return NULL;
|
|
}
|
|
for (uint32_t index = 0; index < p_entry->input_variable_count; ++index) {
|
|
const SpvReflectInterfaceVariable* p_potential = p_entry->input_variables[index];
|
|
if (p_potential->location == location) {
|
|
p_var = p_potential;
|
|
}
|
|
}
|
|
}
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = IsNotNull(p_var)
|
|
? SPV_REFLECT_RESULT_SUCCESS
|
|
: (IsNull(p_module) ? SPV_REFLECT_RESULT_ERROR_NULL_POINTER : SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND);
|
|
}
|
|
return p_var;
|
|
}
|
|
|
|
const SpvReflectInterfaceVariable* spvReflectGetInputVariableBySemantic(const SpvReflectShaderModule* p_module,
|
|
const char* semantic, SpvReflectResult* p_result) {
|
|
if (IsNull(semantic)) {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
return NULL;
|
|
}
|
|
if (semantic[0] == '\0') {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
return NULL;
|
|
}
|
|
const SpvReflectInterfaceVariable* p_var = NULL;
|
|
if (IsNotNull(p_module)) {
|
|
for (uint32_t index = 0; index < p_module->input_variable_count; ++index) {
|
|
const SpvReflectInterfaceVariable* p_potential = p_module->input_variables[index];
|
|
if (p_potential->semantic != NULL && strcmp(p_potential->semantic, semantic) == 0) {
|
|
p_var = p_potential;
|
|
}
|
|
}
|
|
}
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = IsNotNull(p_var)
|
|
? SPV_REFLECT_RESULT_SUCCESS
|
|
: (IsNull(p_module) ? SPV_REFLECT_RESULT_ERROR_NULL_POINTER : SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND);
|
|
}
|
|
return p_var;
|
|
}
|
|
|
|
const SpvReflectInterfaceVariable* spvReflectGetEntryPointInputVariableBySemantic(const SpvReflectShaderModule* p_module,
|
|
const char* entry_point, const char* semantic,
|
|
SpvReflectResult* p_result) {
|
|
if (IsNull(semantic)) {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
return NULL;
|
|
}
|
|
if (semantic[0] == '\0') {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
return NULL;
|
|
}
|
|
const SpvReflectInterfaceVariable* p_var = NULL;
|
|
if (IsNotNull(p_module)) {
|
|
const SpvReflectEntryPoint* p_entry = spvReflectGetEntryPoint(p_module, entry_point);
|
|
if (IsNull(p_entry)) {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
return NULL;
|
|
}
|
|
for (uint32_t index = 0; index < p_entry->input_variable_count; ++index) {
|
|
const SpvReflectInterfaceVariable* p_potential = p_entry->input_variables[index];
|
|
if (p_potential->semantic != NULL && strcmp(p_potential->semantic, semantic) == 0) {
|
|
p_var = p_potential;
|
|
}
|
|
}
|
|
}
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = IsNotNull(p_var)
|
|
? SPV_REFLECT_RESULT_SUCCESS
|
|
: (IsNull(p_module) ? SPV_REFLECT_RESULT_ERROR_NULL_POINTER : SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND);
|
|
}
|
|
return p_var;
|
|
}
|
|
|
|
const SpvReflectInterfaceVariable* spvReflectGetOutputVariableByLocation(const SpvReflectShaderModule* p_module, uint32_t location,
|
|
SpvReflectResult* p_result) {
|
|
if (location == INVALID_VALUE) {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
return NULL;
|
|
}
|
|
const SpvReflectInterfaceVariable* p_var = NULL;
|
|
if (IsNotNull(p_module)) {
|
|
for (uint32_t index = 0; index < p_module->output_variable_count; ++index) {
|
|
const SpvReflectInterfaceVariable* p_potential = p_module->output_variables[index];
|
|
if (p_potential->location == location) {
|
|
p_var = p_potential;
|
|
}
|
|
}
|
|
}
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = IsNotNull(p_var)
|
|
? SPV_REFLECT_RESULT_SUCCESS
|
|
: (IsNull(p_module) ? SPV_REFLECT_RESULT_ERROR_NULL_POINTER : SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND);
|
|
}
|
|
return p_var;
|
|
}
|
|
const SpvReflectInterfaceVariable* spvReflectGetOutputVariable(const SpvReflectShaderModule* p_module, uint32_t location,
|
|
SpvReflectResult* p_result) {
|
|
return spvReflectGetOutputVariableByLocation(p_module, location, p_result);
|
|
}
|
|
|
|
const SpvReflectInterfaceVariable* spvReflectGetEntryPointOutputVariableByLocation(const SpvReflectShaderModule* p_module,
|
|
const char* entry_point, uint32_t location,
|
|
SpvReflectResult* p_result) {
|
|
if (location == INVALID_VALUE) {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
const SpvReflectInterfaceVariable* p_var = NULL;
|
|
if (IsNotNull(p_module)) {
|
|
const SpvReflectEntryPoint* p_entry = spvReflectGetEntryPoint(p_module, entry_point);
|
|
if (IsNull(p_entry)) {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
return NULL;
|
|
}
|
|
for (uint32_t index = 0; index < p_entry->output_variable_count; ++index) {
|
|
const SpvReflectInterfaceVariable* p_potential = p_entry->output_variables[index];
|
|
if (p_potential->location == location) {
|
|
p_var = p_potential;
|
|
}
|
|
}
|
|
}
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = IsNotNull(p_var)
|
|
? SPV_REFLECT_RESULT_SUCCESS
|
|
: (IsNull(p_module) ? SPV_REFLECT_RESULT_ERROR_NULL_POINTER : SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND);
|
|
}
|
|
return p_var;
|
|
}
|
|
|
|
const SpvReflectInterfaceVariable* spvReflectGetOutputVariableBySemantic(const SpvReflectShaderModule* p_module,
|
|
const char* semantic, SpvReflectResult* p_result) {
|
|
if (IsNull(semantic)) {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
return NULL;
|
|
}
|
|
if (semantic[0] == '\0') {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
return NULL;
|
|
}
|
|
const SpvReflectInterfaceVariable* p_var = NULL;
|
|
if (IsNotNull(p_module)) {
|
|
for (uint32_t index = 0; index < p_module->output_variable_count; ++index) {
|
|
const SpvReflectInterfaceVariable* p_potential = p_module->output_variables[index];
|
|
if (p_potential->semantic != NULL && strcmp(p_potential->semantic, semantic) == 0) {
|
|
p_var = p_potential;
|
|
}
|
|
}
|
|
}
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = IsNotNull(p_var)
|
|
? SPV_REFLECT_RESULT_SUCCESS
|
|
: (IsNull(p_module) ? SPV_REFLECT_RESULT_ERROR_NULL_POINTER : SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND);
|
|
}
|
|
return p_var;
|
|
}
|
|
|
|
const SpvReflectInterfaceVariable* spvReflectGetEntryPointOutputVariableBySemantic(const SpvReflectShaderModule* p_module,
|
|
const char* entry_point, const char* semantic,
|
|
SpvReflectResult* p_result) {
|
|
if (IsNull(semantic)) {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
return NULL;
|
|
}
|
|
if (semantic[0] == '\0') {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
return NULL;
|
|
}
|
|
const SpvReflectInterfaceVariable* p_var = NULL;
|
|
if (IsNotNull(p_module)) {
|
|
const SpvReflectEntryPoint* p_entry = spvReflectGetEntryPoint(p_module, entry_point);
|
|
if (IsNull(p_entry)) {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
return NULL;
|
|
}
|
|
for (uint32_t index = 0; index < p_entry->output_variable_count; ++index) {
|
|
const SpvReflectInterfaceVariable* p_potential = p_entry->output_variables[index];
|
|
if (p_potential->semantic != NULL && strcmp(p_potential->semantic, semantic) == 0) {
|
|
p_var = p_potential;
|
|
}
|
|
}
|
|
}
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = IsNotNull(p_var)
|
|
? SPV_REFLECT_RESULT_SUCCESS
|
|
: (IsNull(p_module) ? SPV_REFLECT_RESULT_ERROR_NULL_POINTER : SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND);
|
|
}
|
|
return p_var;
|
|
}
|
|
|
|
const SpvReflectBlockVariable* spvReflectGetPushConstantBlock(const SpvReflectShaderModule* p_module, uint32_t index,
|
|
SpvReflectResult* p_result) {
|
|
const SpvReflectBlockVariable* p_push_constant = NULL;
|
|
if (IsNotNull(p_module)) {
|
|
if (index < p_module->push_constant_block_count) {
|
|
p_push_constant = &p_module->push_constant_blocks[index];
|
|
}
|
|
}
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = IsNotNull(p_push_constant)
|
|
? SPV_REFLECT_RESULT_SUCCESS
|
|
: (IsNull(p_module) ? SPV_REFLECT_RESULT_ERROR_NULL_POINTER : SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND);
|
|
}
|
|
return p_push_constant;
|
|
}
|
|
const SpvReflectBlockVariable* spvReflectGetPushConstant(const SpvReflectShaderModule* p_module, uint32_t index,
|
|
SpvReflectResult* p_result) {
|
|
return spvReflectGetPushConstantBlock(p_module, index, p_result);
|
|
}
|
|
|
|
const SpvReflectBlockVariable* spvReflectGetEntryPointPushConstantBlock(const SpvReflectShaderModule* p_module,
|
|
const char* entry_point, SpvReflectResult* p_result) {
|
|
const SpvReflectBlockVariable* p_push_constant = NULL;
|
|
if (IsNotNull(p_module)) {
|
|
const SpvReflectEntryPoint* p_entry = spvReflectGetEntryPoint(p_module, entry_point);
|
|
if (IsNull(p_entry)) {
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
return NULL;
|
|
}
|
|
for (uint32_t i = 0; i < p_module->push_constant_block_count; ++i) {
|
|
bool found = SearchSortedUint32(p_entry->used_push_constants, p_entry->used_push_constant_count,
|
|
p_module->push_constant_blocks[i].spirv_id);
|
|
if (found) {
|
|
p_push_constant = &p_module->push_constant_blocks[i];
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (IsNotNull(p_result)) {
|
|
*p_result = IsNotNull(p_push_constant)
|
|
? SPV_REFLECT_RESULT_SUCCESS
|
|
: (IsNull(p_module) ? SPV_REFLECT_RESULT_ERROR_NULL_POINTER : SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND);
|
|
}
|
|
return p_push_constant;
|
|
}
|
|
|
|
SpvReflectResult spvReflectChangeDescriptorBindingNumbers(SpvReflectShaderModule* p_module,
|
|
const SpvReflectDescriptorBinding* p_binding, uint32_t new_binding_number,
|
|
uint32_t new_set_binding) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_binding)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
|
|
SpvReflectDescriptorBinding* p_target_descriptor = NULL;
|
|
for (uint32_t index = 0; index < p_module->descriptor_binding_count; ++index) {
|
|
if (&p_module->descriptor_bindings[index] == p_binding) {
|
|
p_target_descriptor = &p_module->descriptor_bindings[index];
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (IsNotNull(p_target_descriptor)) {
|
|
if (p_target_descriptor->word_offset.binding > (p_module->_internal->spirv_word_count - 1)) {
|
|
return SPV_REFLECT_RESULT_ERROR_RANGE_EXCEEDED;
|
|
}
|
|
// Binding number
|
|
if (new_binding_number != (uint32_t)SPV_REFLECT_BINDING_NUMBER_DONT_CHANGE) {
|
|
uint32_t* p_code = p_module->_internal->spirv_code + p_target_descriptor->word_offset.binding;
|
|
*p_code = new_binding_number;
|
|
p_target_descriptor->binding = new_binding_number;
|
|
}
|
|
// Set number
|
|
if (new_set_binding != (uint32_t)SPV_REFLECT_SET_NUMBER_DONT_CHANGE) {
|
|
uint32_t* p_code = p_module->_internal->spirv_code + p_target_descriptor->word_offset.set;
|
|
*p_code = new_set_binding;
|
|
p_target_descriptor->set = new_set_binding;
|
|
}
|
|
}
|
|
|
|
SpvReflectResult result = SPV_REFLECT_RESULT_SUCCESS;
|
|
if (new_set_binding != (uint32_t)SPV_REFLECT_SET_NUMBER_DONT_CHANGE) {
|
|
result = SynchronizeDescriptorSets(p_module);
|
|
}
|
|
return result;
|
|
}
|
|
SpvReflectResult spvReflectChangeDescriptorBindingNumber(SpvReflectShaderModule* p_module,
|
|
const SpvReflectDescriptorBinding* p_descriptor_binding,
|
|
uint32_t new_binding_number, uint32_t optional_new_set_number) {
|
|
return spvReflectChangeDescriptorBindingNumbers(p_module, p_descriptor_binding, new_binding_number, optional_new_set_number);
|
|
}
|
|
|
|
SpvReflectResult spvReflectChangeDescriptorSetNumber(SpvReflectShaderModule* p_module, const SpvReflectDescriptorSet* p_set,
|
|
uint32_t new_set_number) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_set)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
SpvReflectDescriptorSet* p_target_set = NULL;
|
|
for (uint32_t index = 0; index < SPV_REFLECT_MAX_DESCRIPTOR_SETS; ++index) {
|
|
// The descriptor sets for specific entry points might not be in this set,
|
|
// so just match on set index.
|
|
if (p_module->descriptor_sets[index].set == p_set->set) {
|
|
p_target_set = (SpvReflectDescriptorSet*)p_set;
|
|
break;
|
|
}
|
|
}
|
|
|
|
SpvReflectResult result = SPV_REFLECT_RESULT_SUCCESS;
|
|
if (IsNotNull(p_target_set) && new_set_number != (uint32_t)SPV_REFLECT_SET_NUMBER_DONT_CHANGE) {
|
|
for (uint32_t index = 0; index < p_target_set->binding_count; ++index) {
|
|
SpvReflectDescriptorBinding* p_descriptor = p_target_set->bindings[index];
|
|
if (p_descriptor->word_offset.set > (p_module->_internal->spirv_word_count - 1)) {
|
|
return SPV_REFLECT_RESULT_ERROR_RANGE_EXCEEDED;
|
|
}
|
|
|
|
uint32_t* p_code = p_module->_internal->spirv_code + p_descriptor->word_offset.set;
|
|
*p_code = new_set_number;
|
|
p_descriptor->set = new_set_number;
|
|
}
|
|
|
|
result = SynchronizeDescriptorSets(p_module);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
static SpvReflectResult ChangeVariableLocation(SpvReflectShaderModule* p_module, SpvReflectInterfaceVariable* p_variable,
|
|
uint32_t new_location) {
|
|
if (p_variable->word_offset.location > (p_module->_internal->spirv_word_count - 1)) {
|
|
return SPV_REFLECT_RESULT_ERROR_RANGE_EXCEEDED;
|
|
}
|
|
uint32_t* p_code = p_module->_internal->spirv_code + p_variable->word_offset.location;
|
|
*p_code = new_location;
|
|
p_variable->location = new_location;
|
|
return SPV_REFLECT_RESULT_SUCCESS;
|
|
}
|
|
|
|
SpvReflectResult spvReflectChangeInputVariableLocation(SpvReflectShaderModule* p_module,
|
|
const SpvReflectInterfaceVariable* p_input_variable, uint32_t new_location) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_input_variable)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
for (uint32_t index = 0; index < p_module->input_variable_count; ++index) {
|
|
if (p_module->input_variables[index] == p_input_variable) {
|
|
return ChangeVariableLocation(p_module, p_module->input_variables[index], new_location);
|
|
}
|
|
}
|
|
return SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
|
|
SpvReflectResult spvReflectChangeOutputVariableLocation(SpvReflectShaderModule* p_module,
|
|
const SpvReflectInterfaceVariable* p_output_variable,
|
|
uint32_t new_location) {
|
|
if (IsNull(p_module)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
if (IsNull(p_output_variable)) {
|
|
return SPV_REFLECT_RESULT_ERROR_NULL_POINTER;
|
|
}
|
|
for (uint32_t index = 0; index < p_module->output_variable_count; ++index) {
|
|
if (p_module->output_variables[index] == p_output_variable) {
|
|
return ChangeVariableLocation(p_module, p_module->output_variables[index], new_location);
|
|
}
|
|
}
|
|
return SPV_REFLECT_RESULT_ERROR_ELEMENT_NOT_FOUND;
|
|
}
|
|
|
|
const char* spvReflectSourceLanguage(SpvSourceLanguage source_lang) {
|
|
switch (source_lang) {
|
|
case SpvSourceLanguageESSL:
|
|
return "ESSL";
|
|
case SpvSourceLanguageGLSL:
|
|
return "GLSL";
|
|
case SpvSourceLanguageOpenCL_C:
|
|
return "OpenCL_C";
|
|
case SpvSourceLanguageOpenCL_CPP:
|
|
return "OpenCL_CPP";
|
|
case SpvSourceLanguageHLSL:
|
|
return "HLSL";
|
|
case SpvSourceLanguageCPP_for_OpenCL:
|
|
return "CPP_for_OpenCL";
|
|
case SpvSourceLanguageSYCL:
|
|
return "SYCL";
|
|
case SpvSourceLanguageHERO_C:
|
|
return "Hero C";
|
|
case SpvSourceLanguageNZSL:
|
|
return "NZSL";
|
|
default:
|
|
break;
|
|
}
|
|
// The source language is SpvSourceLanguageUnknown, SpvSourceLanguageMax, or
|
|
// some other value that does not correspond to a knonwn language.
|
|
return "Unknown";
|
|
}
|
|
|
|
const char* spvReflectBlockVariableTypeName(const SpvReflectBlockVariable* p_var) {
|
|
if (p_var == NULL) {
|
|
return NULL;
|
|
}
|
|
return p_var->type_description->type_name;
|
|
}
|