mirror of
https://github.com/KhronosGroup/SPIRV-Tools
synced 2024-11-25 21:10:04 +00:00
827 lines
28 KiB
C++
827 lines
28 KiB
C++
// Copyright (c) 2019 Valve Corporation
|
|
// Copyright (c) 2019 LunarG 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.
|
|
|
|
// Bindless Check Instrumentation Tests.
|
|
// Tests ending with V2 use version 2 record format.
|
|
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
#include "test/opt/assembly_builder.h"
|
|
#include "test/opt/pass_fixture.h"
|
|
#include "test/opt/pass_utils.h"
|
|
|
|
namespace spvtools {
|
|
namespace opt {
|
|
namespace {
|
|
|
|
using InstBuffAddrTest = PassTest<::testing::Test>;
|
|
|
|
TEST_F(InstBuffAddrTest, InstPhysicalStorageBufferStore) {
|
|
// #version 450
|
|
// #extension GL_EXT_buffer_reference : enable
|
|
//
|
|
// layout(buffer_reference, buffer_reference_align = 16) buffer bufStruct;
|
|
//
|
|
// layout(set = 0, binding = 0) uniform ufoo {
|
|
// bufStruct data;
|
|
// uint offset;
|
|
// } u_info;
|
|
//
|
|
// layout(buffer_reference, std140) buffer bufStruct {
|
|
// layout(offset = 0) int a[2];
|
|
// layout(offset = 32) int b;
|
|
// };
|
|
//
|
|
// void main() {
|
|
// u_info.data.b = 0xca7;
|
|
// }
|
|
|
|
const std::string defs_before =
|
|
R"(OpCapability Shader
|
|
OpCapability PhysicalStorageBufferAddresses
|
|
OpExtension "SPV_EXT_physical_storage_buffer"
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel PhysicalStorageBuffer64 GLSL450
|
|
OpEntryPoint GLCompute %main "main"
|
|
OpExecutionMode %main LocalSize 1 1 1
|
|
OpSource GLSL 450
|
|
OpSourceExtension "GL_EXT_buffer_reference"
|
|
OpName %main "main"
|
|
OpName %ufoo "ufoo"
|
|
OpMemberName %ufoo 0 "data"
|
|
OpMemberName %ufoo 1 "offset"
|
|
OpName %bufStruct "bufStruct"
|
|
OpMemberName %bufStruct 0 "a"
|
|
OpMemberName %bufStruct 1 "b"
|
|
OpName %u_info "u_info"
|
|
OpMemberDecorate %ufoo 0 Offset 0
|
|
OpMemberDecorate %ufoo 1 Offset 8
|
|
OpDecorate %ufoo Block
|
|
OpDecorate %_arr_int_uint_2 ArrayStride 16
|
|
OpMemberDecorate %bufStruct 0 Offset 0
|
|
OpMemberDecorate %bufStruct 1 Offset 32
|
|
OpDecorate %bufStruct Block
|
|
OpDecorate %u_info DescriptorSet 0
|
|
OpDecorate %u_info Binding 0
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
OpTypeForwardPointer %_ptr_PhysicalStorageBuffer_bufStruct PhysicalStorageBuffer
|
|
%uint = OpTypeInt 32 0
|
|
%ufoo = OpTypeStruct %_ptr_PhysicalStorageBuffer_bufStruct %uint
|
|
%int = OpTypeInt 32 1
|
|
%uint_2 = OpConstant %uint 2
|
|
%_arr_int_uint_2 = OpTypeArray %int %uint_2
|
|
%bufStruct = OpTypeStruct %_arr_int_uint_2 %int
|
|
%_ptr_PhysicalStorageBuffer_bufStruct = OpTypePointer PhysicalStorageBuffer %bufStruct
|
|
%_ptr_Uniform_ufoo = OpTypePointer Uniform %ufoo
|
|
%u_info = OpVariable %_ptr_Uniform_ufoo Uniform
|
|
%int_0 = OpConstant %int 0
|
|
%_ptr_Uniform__ptr_PhysicalStorageBuffer_bufStruct = OpTypePointer Uniform %_ptr_PhysicalStorageBuffer_bufStruct
|
|
%int_1 = OpConstant %int 1
|
|
%int_3239 = OpConstant %int 3239
|
|
%_ptr_PhysicalStorageBuffer_int = OpTypePointer PhysicalStorageBuffer %int
|
|
)";
|
|
|
|
const std::string defs_after =
|
|
R"(OpCapability Shader
|
|
OpCapability PhysicalStorageBufferAddresses
|
|
OpCapability Int64
|
|
OpExtension "SPV_EXT_physical_storage_buffer"
|
|
OpExtension "SPV_KHR_storage_buffer_storage_class"
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel PhysicalStorageBuffer64 GLSL450
|
|
OpEntryPoint GLCompute %main "main" %gl_GlobalInvocationID
|
|
OpExecutionMode %main LocalSize 1 1 1
|
|
OpSource GLSL 450
|
|
OpSourceExtension "GL_EXT_buffer_reference"
|
|
OpName %main "main"
|
|
OpName %ufoo "ufoo"
|
|
OpMemberName %ufoo 0 "data"
|
|
OpMemberName %ufoo 1 "offset"
|
|
OpName %bufStruct "bufStruct"
|
|
OpMemberName %bufStruct 0 "a"
|
|
OpMemberName %bufStruct 1 "b"
|
|
OpName %u_info "u_info"
|
|
OpMemberDecorate %ufoo 0 Offset 0
|
|
OpMemberDecorate %ufoo 1 Offset 8
|
|
OpDecorate %ufoo Block
|
|
OpDecorate %_arr_int_uint_2 ArrayStride 16
|
|
OpMemberDecorate %bufStruct 0 Offset 0
|
|
OpMemberDecorate %bufStruct 1 Offset 32
|
|
OpDecorate %bufStruct Block
|
|
OpDecorate %u_info DescriptorSet 0
|
|
OpDecorate %u_info Binding 0
|
|
OpDecorate %_runtimearr_ulong ArrayStride 8
|
|
OpDecorate %_struct_39 Block
|
|
OpMemberDecorate %_struct_39 0 Offset 0
|
|
OpDecorate %41 DescriptorSet 7
|
|
OpDecorate %41 Binding 2
|
|
OpDecorate %_runtimearr_uint ArrayStride 4
|
|
OpDecorate %_struct_77 Block
|
|
OpMemberDecorate %_struct_77 0 Offset 0
|
|
OpMemberDecorate %_struct_77 1 Offset 4
|
|
OpDecorate %79 DescriptorSet 7
|
|
OpDecorate %79 Binding 0
|
|
OpDecorate %gl_GlobalInvocationID BuiltIn GlobalInvocationId
|
|
%void = OpTypeVoid
|
|
%8 = OpTypeFunction %void
|
|
OpTypeForwardPointer %_ptr_PhysicalStorageBuffer_bufStruct PhysicalStorageBuffer
|
|
%uint = OpTypeInt 32 0
|
|
%ufoo = OpTypeStruct %_ptr_PhysicalStorageBuffer_bufStruct %uint
|
|
%int = OpTypeInt 32 1
|
|
%uint_2 = OpConstant %uint 2
|
|
%_arr_int_uint_2 = OpTypeArray %int %uint_2
|
|
%bufStruct = OpTypeStruct %_arr_int_uint_2 %int
|
|
%_ptr_PhysicalStorageBuffer_bufStruct = OpTypePointer PhysicalStorageBuffer %bufStruct
|
|
%_ptr_Uniform_ufoo = OpTypePointer Uniform %ufoo
|
|
%u_info = OpVariable %_ptr_Uniform_ufoo Uniform
|
|
%int_0 = OpConstant %int 0
|
|
%_ptr_Uniform__ptr_PhysicalStorageBuffer_bufStruct = OpTypePointer Uniform %_ptr_PhysicalStorageBuffer_bufStruct
|
|
%int_1 = OpConstant %int 1
|
|
%int_3239 = OpConstant %int 3239
|
|
%_ptr_PhysicalStorageBuffer_int = OpTypePointer PhysicalStorageBuffer %int
|
|
%ulong = OpTypeInt 64 0
|
|
%uint_4 = OpConstant %uint 4
|
|
%bool = OpTypeBool
|
|
%28 = OpTypeFunction %bool %ulong %uint
|
|
%uint_1 = OpConstant %uint 1
|
|
%_runtimearr_ulong = OpTypeRuntimeArray %ulong
|
|
%_struct_39 = OpTypeStruct %_runtimearr_ulong
|
|
%_ptr_StorageBuffer__struct_39 = OpTypePointer StorageBuffer %_struct_39
|
|
%41 = OpVariable %_ptr_StorageBuffer__struct_39 StorageBuffer
|
|
%_ptr_StorageBuffer_ulong = OpTypePointer StorageBuffer %ulong
|
|
%uint_0 = OpConstant %uint 0
|
|
%uint_32 = OpConstant %uint 32
|
|
%70 = OpTypeFunction %void %uint %uint %uint %uint
|
|
%_runtimearr_uint = OpTypeRuntimeArray %uint
|
|
%_struct_77 = OpTypeStruct %uint %_runtimearr_uint
|
|
%_ptr_StorageBuffer__struct_77 = OpTypePointer StorageBuffer %_struct_77
|
|
%79 = OpVariable %_ptr_StorageBuffer__struct_77 StorageBuffer
|
|
%_ptr_StorageBuffer_uint = OpTypePointer StorageBuffer %uint
|
|
%uint_10 = OpConstant %uint 10
|
|
%uint_23 = OpConstant %uint 23
|
|
%uint_5 = OpConstant %uint 5
|
|
%uint_3 = OpConstant %uint 3
|
|
%v3uint = OpTypeVector %uint 3
|
|
%_ptr_Input_v3uint = OpTypePointer Input %v3uint
|
|
%gl_GlobalInvocationID = OpVariable %_ptr_Input_v3uint Input
|
|
%uint_6 = OpConstant %uint 6
|
|
%uint_7 = OpConstant %uint 7
|
|
%uint_8 = OpConstant %uint 8
|
|
%uint_9 = OpConstant %uint 9
|
|
%uint_48 = OpConstant %uint 48
|
|
)";
|
|
|
|
const std::string func_before =
|
|
R"(%main = OpFunction %void None %3
|
|
%5 = OpLabel
|
|
%17 = OpAccessChain %_ptr_Uniform__ptr_PhysicalStorageBuffer_bufStruct %u_info %int_0
|
|
%18 = OpLoad %_ptr_PhysicalStorageBuffer_bufStruct %17
|
|
%22 = OpAccessChain %_ptr_PhysicalStorageBuffer_int %18 %int_1
|
|
OpStore %22 %int_3239 Aligned 16
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const std::string func_after =
|
|
R"(%main = OpFunction %void None %8
|
|
%19 = OpLabel
|
|
%20 = OpAccessChain %_ptr_Uniform__ptr_PhysicalStorageBuffer_bufStruct %u_info %int_0
|
|
%21 = OpLoad %_ptr_PhysicalStorageBuffer_bufStruct %20
|
|
%22 = OpAccessChain %_ptr_PhysicalStorageBuffer_int %21 %int_1
|
|
%24 = OpConvertPtrToU %ulong %22
|
|
%61 = OpFunctionCall %bool %26 %24 %uint_4
|
|
OpSelectionMerge %62 None
|
|
OpBranchConditional %61 %63 %64
|
|
%63 = OpLabel
|
|
OpStore %22 %int_3239 Aligned 16
|
|
OpBranch %62
|
|
%64 = OpLabel
|
|
%65 = OpUConvert %uint %24
|
|
%67 = OpShiftRightLogical %ulong %24 %uint_32
|
|
%68 = OpUConvert %uint %67
|
|
%124 = OpFunctionCall %void %69 %uint_48 %uint_2 %65 %68
|
|
OpBranch %62
|
|
%62 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const std::string new_funcs =
|
|
R"(%26 = OpFunction %bool None %28
|
|
%29 = OpFunctionParameter %ulong
|
|
%30 = OpFunctionParameter %uint
|
|
%31 = OpLabel
|
|
OpBranch %32
|
|
%32 = OpLabel
|
|
%34 = OpPhi %uint %uint_1 %31 %35 %33
|
|
OpLoopMerge %37 %33 None
|
|
OpBranch %33
|
|
%33 = OpLabel
|
|
%35 = OpIAdd %uint %34 %uint_1
|
|
%44 = OpAccessChain %_ptr_StorageBuffer_ulong %41 %uint_0 %35
|
|
%45 = OpLoad %ulong %44
|
|
%46 = OpUGreaterThan %bool %45 %29
|
|
OpBranchConditional %46 %37 %32
|
|
%37 = OpLabel
|
|
%47 = OpISub %uint %35 %uint_1
|
|
%48 = OpAccessChain %_ptr_StorageBuffer_ulong %41 %uint_0 %47
|
|
%49 = OpLoad %ulong %48
|
|
%50 = OpISub %ulong %29 %49
|
|
%51 = OpUConvert %ulong %30
|
|
%52 = OpIAdd %ulong %50 %51
|
|
%53 = OpAccessChain %_ptr_StorageBuffer_ulong %41 %uint_0 %uint_0
|
|
%54 = OpLoad %ulong %53
|
|
%55 = OpUConvert %uint %54
|
|
%56 = OpISub %uint %47 %uint_1
|
|
%57 = OpIAdd %uint %56 %55
|
|
%58 = OpAccessChain %_ptr_StorageBuffer_ulong %41 %uint_0 %57
|
|
%59 = OpLoad %ulong %58
|
|
%60 = OpULessThanEqual %bool %52 %59
|
|
OpReturnValue %60
|
|
OpFunctionEnd
|
|
%69 = OpFunction %void None %70
|
|
%71 = OpFunctionParameter %uint
|
|
%72 = OpFunctionParameter %uint
|
|
%73 = OpFunctionParameter %uint
|
|
%74 = OpFunctionParameter %uint
|
|
%75 = OpLabel
|
|
%81 = OpAccessChain %_ptr_StorageBuffer_uint %79 %uint_0
|
|
%83 = OpAtomicIAdd %uint %81 %uint_4 %uint_0 %uint_10
|
|
%84 = OpIAdd %uint %83 %uint_10
|
|
%85 = OpArrayLength %uint %79 1
|
|
%86 = OpULessThanEqual %bool %84 %85
|
|
OpSelectionMerge %87 None
|
|
OpBranchConditional %86 %88 %87
|
|
%88 = OpLabel
|
|
%89 = OpIAdd %uint %83 %uint_0
|
|
%90 = OpAccessChain %_ptr_StorageBuffer_uint %79 %uint_1 %89
|
|
OpStore %90 %uint_10
|
|
%92 = OpIAdd %uint %83 %uint_1
|
|
%93 = OpAccessChain %_ptr_StorageBuffer_uint %79 %uint_1 %92
|
|
OpStore %93 %uint_23
|
|
%94 = OpIAdd %uint %83 %uint_2
|
|
%95 = OpAccessChain %_ptr_StorageBuffer_uint %79 %uint_1 %94
|
|
OpStore %95 %71
|
|
%98 = OpIAdd %uint %83 %uint_3
|
|
%99 = OpAccessChain %_ptr_StorageBuffer_uint %79 %uint_1 %98
|
|
OpStore %99 %uint_5
|
|
%103 = OpLoad %v3uint %gl_GlobalInvocationID
|
|
%104 = OpCompositeExtract %uint %103 0
|
|
%105 = OpCompositeExtract %uint %103 1
|
|
%106 = OpCompositeExtract %uint %103 2
|
|
%107 = OpIAdd %uint %83 %uint_4
|
|
%108 = OpAccessChain %_ptr_StorageBuffer_uint %79 %uint_1 %107
|
|
OpStore %108 %104
|
|
%109 = OpIAdd %uint %83 %uint_5
|
|
%110 = OpAccessChain %_ptr_StorageBuffer_uint %79 %uint_1 %109
|
|
OpStore %110 %105
|
|
%112 = OpIAdd %uint %83 %uint_6
|
|
%113 = OpAccessChain %_ptr_StorageBuffer_uint %79 %uint_1 %112
|
|
OpStore %113 %106
|
|
%115 = OpIAdd %uint %83 %uint_7
|
|
%116 = OpAccessChain %_ptr_StorageBuffer_uint %79 %uint_1 %115
|
|
OpStore %116 %72
|
|
%118 = OpIAdd %uint %83 %uint_8
|
|
%119 = OpAccessChain %_ptr_StorageBuffer_uint %79 %uint_1 %118
|
|
OpStore %119 %73
|
|
%121 = OpIAdd %uint %83 %uint_9
|
|
%122 = OpAccessChain %_ptr_StorageBuffer_uint %79 %uint_1 %121
|
|
OpStore %122 %74
|
|
OpBranch %87
|
|
%87 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
// SetAssembleOptions(SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
|
|
SinglePassRunAndCheck<InstBuffAddrCheckPass>(
|
|
defs_before + func_before, defs_after + func_after + new_funcs, true,
|
|
true, 7u, 23u);
|
|
}
|
|
|
|
TEST_F(InstBuffAddrTest, InstPhysicalStorageBufferLoadAndStore) {
|
|
// #version 450
|
|
// #extension GL_EXT_buffer_reference : enable
|
|
|
|
// // forward reference
|
|
// layout(buffer_reference) buffer blockType;
|
|
|
|
// layout(buffer_reference, std430, buffer_reference_align = 16) buffer
|
|
// blockType {
|
|
// int x;
|
|
// blockType next;
|
|
// };
|
|
|
|
// layout(std430) buffer rootBlock {
|
|
// blockType root;
|
|
// } r;
|
|
|
|
// void main()
|
|
// {
|
|
// blockType b = r.root;
|
|
// b = b.next;
|
|
// b.x = 531;
|
|
// }
|
|
|
|
const std::string defs_before =
|
|
R"(OpCapability Shader
|
|
OpCapability PhysicalStorageBufferAddresses
|
|
OpExtension "SPV_EXT_physical_storage_buffer"
|
|
OpExtension "SPV_KHR_storage_buffer_storage_class"
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel PhysicalStorageBuffer64 GLSL450
|
|
OpEntryPoint GLCompute %main "main"
|
|
OpExecutionMode %main LocalSize 1 1 1
|
|
OpSource GLSL 450
|
|
OpSourceExtension "GL_EXT_buffer_reference"
|
|
OpName %main "main"
|
|
OpName %blockType "blockType"
|
|
OpMemberName %blockType 0 "x"
|
|
OpMemberName %blockType 1 "next"
|
|
OpName %rootBlock "rootBlock"
|
|
OpMemberName %rootBlock 0 "root"
|
|
OpName %r "r"
|
|
OpMemberDecorate %blockType 0 Offset 0
|
|
OpMemberDecorate %blockType 1 Offset 8
|
|
OpDecorate %blockType Block
|
|
OpMemberDecorate %rootBlock 0 Offset 0
|
|
OpDecorate %rootBlock Block
|
|
OpDecorate %r DescriptorSet 0
|
|
OpDecorate %r Binding 0
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
OpTypeForwardPointer %_ptr_PhysicalStorageBuffer_blockType PhysicalStorageBuffer
|
|
%int = OpTypeInt 32 1
|
|
%blockType = OpTypeStruct %int %_ptr_PhysicalStorageBuffer_blockType
|
|
%_ptr_PhysicalStorageBuffer_blockType = OpTypePointer PhysicalStorageBuffer %blockType
|
|
%rootBlock = OpTypeStruct %_ptr_PhysicalStorageBuffer_blockType
|
|
%_ptr_StorageBuffer_rootBlock = OpTypePointer StorageBuffer %rootBlock
|
|
%r = OpVariable %_ptr_StorageBuffer_rootBlock StorageBuffer
|
|
%int_0 = OpConstant %int 0
|
|
%_ptr_StorageBuffer__ptr_PhysicalStorageBuffer_blockType = OpTypePointer StorageBuffer %_ptr_PhysicalStorageBuffer_blockType
|
|
%int_1 = OpConstant %int 1
|
|
%_ptr_PhysicalStorageBuffer__ptr_PhysicalStorageBuffer_blockType = OpTypePointer PhysicalStorageBuffer %_ptr_PhysicalStorageBuffer_blockType
|
|
%int_531 = OpConstant %int 531
|
|
%_ptr_PhysicalStorageBuffer_int = OpTypePointer PhysicalStorageBuffer %int
|
|
)";
|
|
|
|
const std::string defs_after =
|
|
R"(OpCapability Shader
|
|
OpCapability PhysicalStorageBufferAddresses
|
|
OpCapability Int64
|
|
OpExtension "SPV_EXT_physical_storage_buffer"
|
|
OpExtension "SPV_KHR_storage_buffer_storage_class"
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel PhysicalStorageBuffer64 GLSL450
|
|
OpEntryPoint GLCompute %main "main" %gl_GlobalInvocationID
|
|
OpExecutionMode %main LocalSize 1 1 1
|
|
OpSource GLSL 450
|
|
OpSourceExtension "GL_EXT_buffer_reference"
|
|
OpName %main "main"
|
|
OpName %blockType "blockType"
|
|
OpMemberName %blockType 0 "x"
|
|
OpMemberName %blockType 1 "next"
|
|
OpName %rootBlock "rootBlock"
|
|
OpMemberName %rootBlock 0 "root"
|
|
OpName %r "r"
|
|
OpMemberDecorate %blockType 0 Offset 0
|
|
OpMemberDecorate %blockType 1 Offset 8
|
|
OpDecorate %blockType Block
|
|
OpMemberDecorate %rootBlock 0 Offset 0
|
|
OpDecorate %rootBlock Block
|
|
OpDecorate %r DescriptorSet 0
|
|
OpDecorate %r Binding 0
|
|
OpDecorate %_runtimearr_ulong ArrayStride 8
|
|
OpDecorate %_struct_45 Block
|
|
OpMemberDecorate %_struct_45 0 Offset 0
|
|
OpDecorate %47 DescriptorSet 7
|
|
OpDecorate %47 Binding 2
|
|
OpDecorate %_runtimearr_uint ArrayStride 4
|
|
OpDecorate %_struct_84 Block
|
|
OpMemberDecorate %_struct_84 0 Offset 0
|
|
OpMemberDecorate %_struct_84 1 Offset 4
|
|
OpDecorate %86 DescriptorSet 7
|
|
OpDecorate %86 Binding 0
|
|
OpDecorate %gl_GlobalInvocationID BuiltIn GlobalInvocationId
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
OpTypeForwardPointer %_ptr_PhysicalStorageBuffer_blockType PhysicalStorageBuffer
|
|
%int = OpTypeInt 32 1
|
|
%blockType = OpTypeStruct %int %_ptr_PhysicalStorageBuffer_blockType
|
|
%_ptr_PhysicalStorageBuffer_blockType = OpTypePointer PhysicalStorageBuffer %blockType
|
|
%rootBlock = OpTypeStruct %_ptr_PhysicalStorageBuffer_blockType
|
|
%_ptr_StorageBuffer_rootBlock = OpTypePointer StorageBuffer %rootBlock
|
|
%r = OpVariable %_ptr_StorageBuffer_rootBlock StorageBuffer
|
|
%int_0 = OpConstant %int 0
|
|
%_ptr_StorageBuffer__ptr_PhysicalStorageBuffer_blockType = OpTypePointer StorageBuffer %_ptr_PhysicalStorageBuffer_blockType
|
|
%int_1 = OpConstant %int 1
|
|
%_ptr_PhysicalStorageBuffer__ptr_PhysicalStorageBuffer_blockType = OpTypePointer PhysicalStorageBuffer %_ptr_PhysicalStorageBuffer_blockType
|
|
%int_531 = OpConstant %int 531
|
|
%_ptr_PhysicalStorageBuffer_int = OpTypePointer PhysicalStorageBuffer %int
|
|
%uint = OpTypeInt 32 0
|
|
%uint_2 = OpConstant %uint 2
|
|
%ulong = OpTypeInt 64 0
|
|
%uint_8 = OpConstant %uint 8
|
|
%bool = OpTypeBool
|
|
%34 = OpTypeFunction %bool %ulong %uint
|
|
%uint_1 = OpConstant %uint 1
|
|
%_runtimearr_ulong = OpTypeRuntimeArray %ulong
|
|
%_struct_45 = OpTypeStruct %_runtimearr_ulong
|
|
%_ptr_StorageBuffer__struct_45 = OpTypePointer StorageBuffer %_struct_45
|
|
%47 = OpVariable %_ptr_StorageBuffer__struct_45 StorageBuffer
|
|
%_ptr_StorageBuffer_ulong = OpTypePointer StorageBuffer %ulong
|
|
%uint_0 = OpConstant %uint 0
|
|
%uint_32 = OpConstant %uint 32
|
|
%77 = OpTypeFunction %void %uint %uint %uint %uint
|
|
%_runtimearr_uint = OpTypeRuntimeArray %uint
|
|
%_struct_84 = OpTypeStruct %uint %_runtimearr_uint
|
|
%_ptr_StorageBuffer__struct_84 = OpTypePointer StorageBuffer %_struct_84
|
|
%86 = OpVariable %_ptr_StorageBuffer__struct_84 StorageBuffer
|
|
%_ptr_StorageBuffer_uint = OpTypePointer StorageBuffer %uint
|
|
%uint_10 = OpConstant %uint 10
|
|
%uint_4 = OpConstant %uint 4
|
|
%uint_23 = OpConstant %uint 23
|
|
%uint_5 = OpConstant %uint 5
|
|
%uint_3 = OpConstant %uint 3
|
|
%v3uint = OpTypeVector %uint 3
|
|
%_ptr_Input_v3uint = OpTypePointer Input %v3uint
|
|
%gl_GlobalInvocationID = OpVariable %_ptr_Input_v3uint Input
|
|
%uint_6 = OpConstant %uint 6
|
|
%uint_7 = OpConstant %uint 7
|
|
%uint_9 = OpConstant %uint 9
|
|
%uint_44 = OpConstant %uint 44
|
|
%132 = OpConstantNull %ulong
|
|
%uint_46 = OpConstant %uint 46
|
|
)";
|
|
|
|
const std::string func_before =
|
|
R"(%main = OpFunction %void None %3
|
|
%5 = OpLabel
|
|
%16 = OpAccessChain %_ptr_StorageBuffer__ptr_PhysicalStorageBuffer_blockType %r %int_0
|
|
%17 = OpLoad %_ptr_PhysicalStorageBuffer_blockType %16
|
|
%21 = OpAccessChain %_ptr_PhysicalStorageBuffer__ptr_PhysicalStorageBuffer_blockType %17 %int_1
|
|
%22 = OpLoad %_ptr_PhysicalStorageBuffer_blockType %21 Aligned 8
|
|
%26 = OpAccessChain %_ptr_PhysicalStorageBuffer_int %22 %int_0
|
|
OpStore %26 %int_531 Aligned 16
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const std::string func_after =
|
|
R"(%main = OpFunction %void None %3
|
|
%5 = OpLabel
|
|
%16 = OpAccessChain %_ptr_StorageBuffer__ptr_PhysicalStorageBuffer_blockType %r %int_0
|
|
%17 = OpLoad %_ptr_PhysicalStorageBuffer_blockType %16
|
|
%21 = OpAccessChain %_ptr_PhysicalStorageBuffer__ptr_PhysicalStorageBuffer_blockType %17 %int_1
|
|
%30 = OpConvertPtrToU %ulong %21
|
|
%67 = OpFunctionCall %bool %32 %30 %uint_8
|
|
OpSelectionMerge %68 None
|
|
OpBranchConditional %67 %69 %70
|
|
%69 = OpLabel
|
|
%71 = OpLoad %_ptr_PhysicalStorageBuffer_blockType %21 Aligned 8
|
|
OpBranch %68
|
|
%70 = OpLabel
|
|
%72 = OpUConvert %uint %30
|
|
%74 = OpShiftRightLogical %ulong %30 %uint_32
|
|
%75 = OpUConvert %uint %74
|
|
%131 = OpFunctionCall %void %76 %uint_44 %uint_2 %72 %75
|
|
%133 = OpConvertUToPtr %_ptr_PhysicalStorageBuffer_blockType %132
|
|
OpBranch %68
|
|
%68 = OpLabel
|
|
%134 = OpPhi %_ptr_PhysicalStorageBuffer_blockType %71 %69 %133 %70
|
|
%26 = OpAccessChain %_ptr_PhysicalStorageBuffer_int %134 %int_0
|
|
%135 = OpConvertPtrToU %ulong %26
|
|
%136 = OpFunctionCall %bool %32 %135 %uint_4
|
|
OpSelectionMerge %137 None
|
|
OpBranchConditional %136 %138 %139
|
|
%138 = OpLabel
|
|
OpStore %26 %int_531 Aligned 16
|
|
OpBranch %137
|
|
%139 = OpLabel
|
|
%140 = OpUConvert %uint %135
|
|
%141 = OpShiftRightLogical %ulong %135 %uint_32
|
|
%142 = OpUConvert %uint %141
|
|
%144 = OpFunctionCall %void %76 %uint_46 %uint_2 %140 %142
|
|
OpBranch %137
|
|
%137 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const std::string new_funcs =
|
|
R"(%32 = OpFunction %bool None %34
|
|
%35 = OpFunctionParameter %ulong
|
|
%36 = OpFunctionParameter %uint
|
|
%37 = OpLabel
|
|
OpBranch %38
|
|
%38 = OpLabel
|
|
%40 = OpPhi %uint %uint_1 %37 %41 %39
|
|
OpLoopMerge %43 %39 None
|
|
OpBranch %39
|
|
%39 = OpLabel
|
|
%41 = OpIAdd %uint %40 %uint_1
|
|
%50 = OpAccessChain %_ptr_StorageBuffer_ulong %47 %uint_0 %41
|
|
%51 = OpLoad %ulong %50
|
|
%52 = OpUGreaterThan %bool %51 %35
|
|
OpBranchConditional %52 %43 %38
|
|
%43 = OpLabel
|
|
%53 = OpISub %uint %41 %uint_1
|
|
%54 = OpAccessChain %_ptr_StorageBuffer_ulong %47 %uint_0 %53
|
|
%55 = OpLoad %ulong %54
|
|
%56 = OpISub %ulong %35 %55
|
|
%57 = OpUConvert %ulong %36
|
|
%58 = OpIAdd %ulong %56 %57
|
|
%59 = OpAccessChain %_ptr_StorageBuffer_ulong %47 %uint_0 %uint_0
|
|
%60 = OpLoad %ulong %59
|
|
%61 = OpUConvert %uint %60
|
|
%62 = OpISub %uint %53 %uint_1
|
|
%63 = OpIAdd %uint %62 %61
|
|
%64 = OpAccessChain %_ptr_StorageBuffer_ulong %47 %uint_0 %63
|
|
%65 = OpLoad %ulong %64
|
|
%66 = OpULessThanEqual %bool %58 %65
|
|
OpReturnValue %66
|
|
OpFunctionEnd
|
|
%76 = OpFunction %void None %77
|
|
%78 = OpFunctionParameter %uint
|
|
%79 = OpFunctionParameter %uint
|
|
%80 = OpFunctionParameter %uint
|
|
%81 = OpFunctionParameter %uint
|
|
%82 = OpLabel
|
|
%88 = OpAccessChain %_ptr_StorageBuffer_uint %86 %uint_0
|
|
%91 = OpAtomicIAdd %uint %88 %uint_4 %uint_0 %uint_10
|
|
%92 = OpIAdd %uint %91 %uint_10
|
|
%93 = OpArrayLength %uint %86 1
|
|
%94 = OpULessThanEqual %bool %92 %93
|
|
OpSelectionMerge %95 None
|
|
OpBranchConditional %94 %96 %95
|
|
%96 = OpLabel
|
|
%97 = OpIAdd %uint %91 %uint_0
|
|
%98 = OpAccessChain %_ptr_StorageBuffer_uint %86 %uint_1 %97
|
|
OpStore %98 %uint_10
|
|
%100 = OpIAdd %uint %91 %uint_1
|
|
%101 = OpAccessChain %_ptr_StorageBuffer_uint %86 %uint_1 %100
|
|
OpStore %101 %uint_23
|
|
%102 = OpIAdd %uint %91 %uint_2
|
|
%103 = OpAccessChain %_ptr_StorageBuffer_uint %86 %uint_1 %102
|
|
OpStore %103 %78
|
|
%106 = OpIAdd %uint %91 %uint_3
|
|
%107 = OpAccessChain %_ptr_StorageBuffer_uint %86 %uint_1 %106
|
|
OpStore %107 %uint_5
|
|
%111 = OpLoad %v3uint %gl_GlobalInvocationID
|
|
%112 = OpCompositeExtract %uint %111 0
|
|
%113 = OpCompositeExtract %uint %111 1
|
|
%114 = OpCompositeExtract %uint %111 2
|
|
%115 = OpIAdd %uint %91 %uint_4
|
|
%116 = OpAccessChain %_ptr_StorageBuffer_uint %86 %uint_1 %115
|
|
OpStore %116 %112
|
|
%117 = OpIAdd %uint %91 %uint_5
|
|
%118 = OpAccessChain %_ptr_StorageBuffer_uint %86 %uint_1 %117
|
|
OpStore %118 %113
|
|
%120 = OpIAdd %uint %91 %uint_6
|
|
%121 = OpAccessChain %_ptr_StorageBuffer_uint %86 %uint_1 %120
|
|
OpStore %121 %114
|
|
%123 = OpIAdd %uint %91 %uint_7
|
|
%124 = OpAccessChain %_ptr_StorageBuffer_uint %86 %uint_1 %123
|
|
OpStore %124 %79
|
|
%125 = OpIAdd %uint %91 %uint_8
|
|
%126 = OpAccessChain %_ptr_StorageBuffer_uint %86 %uint_1 %125
|
|
OpStore %126 %80
|
|
%128 = OpIAdd %uint %91 %uint_9
|
|
%129 = OpAccessChain %_ptr_StorageBuffer_uint %86 %uint_1 %128
|
|
OpStore %129 %81
|
|
OpBranch %95
|
|
%95 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SetAssembleOptions(SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
|
|
SinglePassRunAndCheck<InstBuffAddrCheckPass>(
|
|
defs_before + func_before, defs_after + func_after + new_funcs, true,
|
|
true, 7u, 23u);
|
|
}
|
|
|
|
TEST_F(InstBuffAddrTest, StructLoad) {
|
|
// #version 450
|
|
// #extension GL_EXT_buffer_reference : enable
|
|
// #extension GL_ARB_gpu_shader_int64 : enable
|
|
// struct Test {
|
|
// float a;
|
|
// };
|
|
//
|
|
// layout(buffer_reference, std430, buffer_reference_align = 16) buffer
|
|
// TestBuffer { Test test; };
|
|
//
|
|
// Test GetTest(uint64_t ptr) {
|
|
// return TestBuffer(ptr).test;
|
|
// }
|
|
//
|
|
// void main() {
|
|
// GetTest(0xe0000000);
|
|
// }
|
|
|
|
const std::string defs =
|
|
R"(
|
|
OpCapability Shader
|
|
OpCapability Int64
|
|
OpCapability PhysicalStorageBufferAddresses
|
|
; CHECK: OpExtension "SPV_KHR_storage_buffer_storage_class"
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel PhysicalStorageBuffer64 GLSL450
|
|
OpEntryPoint Fragment %main "main"
|
|
; CHECK: OpEntryPoint Fragment %main "main" %60 %99 %gl_FragCoord
|
|
OpExecutionMode %main OriginUpperLeft
|
|
OpSource GLSL 450
|
|
OpSourceExtension "GL_ARB_gpu_shader_int64"
|
|
OpSourceExtension "GL_EXT_buffer_reference"
|
|
OpName %main "main"
|
|
OpName %Test "Test"
|
|
OpMemberName %Test 0 "a"
|
|
OpName %Test_0 "Test"
|
|
OpMemberName %Test_0 0 "a"
|
|
OpName %TestBuffer "TestBuffer"
|
|
OpMemberName %TestBuffer 0 "test"
|
|
)";
|
|
|
|
const std::string decorates =
|
|
R"(
|
|
OpMemberDecorate %Test_0 0 Offset 0
|
|
OpMemberDecorate %TestBuffer 0 Offset 0
|
|
OpDecorate %TestBuffer Block
|
|
; CHECK: OpDecorate %_runtimearr_ulong ArrayStride 8
|
|
; CHECK: OpDecorate %_struct_58 Block
|
|
; CHECK: OpMemberDecorate %_struct_58 0 Offset 0
|
|
; CHECK: OpDecorate %60 DescriptorSet 7
|
|
; CHECK: OpDecorate %60 Binding 2
|
|
; CHECK: OpDecorate %_runtimearr_uint ArrayStride 4
|
|
; CHECK: OpDecorate %_struct_97 Block
|
|
; CHECK: OpMemberDecorate %_struct_97 0 Offset 0
|
|
; CHECK: OpMemberDecorate %_struct_97 1 Offset 4
|
|
; CHECK: OpDecorate %99 DescriptorSet 7
|
|
; CHECK: OpDecorate %99 Binding 0
|
|
; CHECK: OpDecorate %gl_FragCoord BuiltIn FragCoord
|
|
)";
|
|
|
|
const std::string globals =
|
|
R"(
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
%ulong = OpTypeInt 64 0
|
|
%float = OpTypeFloat 32
|
|
%Test = OpTypeStruct %float
|
|
OpTypeForwardPointer %_ptr_PhysicalStorageBuffer_TestBuffer PhysicalStorageBuffer
|
|
%Test_0 = OpTypeStruct %float
|
|
%TestBuffer = OpTypeStruct %Test_0
|
|
%_ptr_PhysicalStorageBuffer_TestBuffer = OpTypePointer PhysicalStorageBuffer %TestBuffer
|
|
%int = OpTypeInt 32 1
|
|
%int_0 = OpConstant %int 0
|
|
%_ptr_PhysicalStorageBuffer_Test_0 = OpTypePointer PhysicalStorageBuffer %Test_0
|
|
%ulong_18446744073172680704 = OpConstant %ulong 18446744073172680704
|
|
; CHECK: %47 = OpTypeFunction %bool %ulong %uint
|
|
; CHECK: %_struct_58 = OpTypeStruct %_runtimearr_ulong
|
|
; CHECK: %60 = OpVariable %_ptr_StorageBuffer__struct_58 StorageBuffer
|
|
; CHECK: %90 = OpTypeFunction %void %uint %uint %uint %uint
|
|
; CHECK: %_struct_97 = OpTypeStruct %uint %_runtimearr_uint
|
|
; CHECK: %99 = OpVariable %_ptr_StorageBuffer__struct_97 StorageBuffer
|
|
; CHECK: %143 = OpConstantNull %Test_0
|
|
)";
|
|
|
|
const std::string main =
|
|
R"(
|
|
%main = OpFunction %void None %3
|
|
%5 = OpLabel
|
|
%37 = OpConvertUToPtr %_ptr_PhysicalStorageBuffer_TestBuffer %ulong_18446744073172680704
|
|
%38 = OpAccessChain %_ptr_PhysicalStorageBuffer_Test_0 %37 %int_0
|
|
%39 = OpLoad %Test_0 %38 Aligned 16
|
|
; CHECK-NOT: %39 = OpLoad %Test_0 %38 Aligned 16
|
|
; CHECK: %43 = OpConvertPtrToU %ulong %38
|
|
; CHECK: %80 = OpFunctionCall %bool %45 %43 %uint_4
|
|
; CHECK: OpSelectionMerge %81 None
|
|
; CHECK: OpBranchConditional %80 %82 %83
|
|
; CHECK: %82 = OpLabel
|
|
; CHECK: %84 = OpLoad %Test_0 %38 Aligned 16
|
|
; CHECK: OpBranch %81
|
|
; CHECK: %83 = OpLabel
|
|
; CHECK: %85 = OpUConvert %uint %43
|
|
; CHECK: %87 = OpShiftRightLogical %ulong %43 %uint_32
|
|
; CHECK: %88 = OpUConvert %uint %87
|
|
; CHECK: %142 = OpFunctionCall %void %89 %uint_37 %uint_2 %85 %88
|
|
; CHECK: OpBranch %81
|
|
; CHECK: %81 = OpLabel
|
|
; CHECK: %144 = OpPhi %Test_0 %84 %82 %143 %83
|
|
%40 = OpCopyLogical %Test %39
|
|
; CHECK-NOT: %40 = OpCopyLogical %Test %39
|
|
; CHECK: %40 = OpCopyLogical %Test %144
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const std::string output_funcs =
|
|
R"(
|
|
; CHECK: %45 = OpFunction %bool None %47
|
|
; CHECK: %48 = OpFunctionParameter %ulong
|
|
; CHECK: %49 = OpFunctionParameter %uint
|
|
; CHECK: %50 = OpLabel
|
|
; CHECK: OpBranch %51
|
|
; CHECK: %51 = OpLabel
|
|
; CHECK: %53 = OpPhi %uint %uint_1 %50 %54 %52
|
|
; CHECK: OpLoopMerge %56 %52 None
|
|
; CHECK: OpBranch %52
|
|
; CHECK: %52 = OpLabel
|
|
; CHECK: %54 = OpIAdd %uint %53 %uint_1
|
|
; CHECK: %63 = OpAccessChain %_ptr_StorageBuffer_ulong %60 %uint_0 %54
|
|
; CHECK: %64 = OpLoad %ulong %63
|
|
; CHECK: %65 = OpUGreaterThan %bool %64 %48
|
|
; CHECK: OpBranchConditional %65 %56 %51
|
|
; CHECK: %56 = OpLabel
|
|
; CHECK: %66 = OpISub %uint %54 %uint_1
|
|
; CHECK: %67 = OpAccessChain %_ptr_StorageBuffer_ulong %60 %uint_0 %66
|
|
; CHECK: %68 = OpLoad %ulong %67
|
|
; CHECK: %69 = OpISub %ulong %48 %68
|
|
; CHECK: %70 = OpUConvert %ulong %49
|
|
; CHECK: %71 = OpIAdd %ulong %69 %70
|
|
; CHECK: %72 = OpAccessChain %_ptr_StorageBuffer_ulong %60 %uint_0 %uint_0
|
|
; CHECK: %73 = OpLoad %ulong %72
|
|
; CHECK: %74 = OpUConvert %uint %73
|
|
; CHECK: %75 = OpISub %uint %66 %uint_1
|
|
; CHECK: %76 = OpIAdd %uint %75 %74
|
|
; CHECK: %77 = OpAccessChain %_ptr_StorageBuffer_ulong %60 %uint_0 %76
|
|
; CHECK: %78 = OpLoad %ulong %77
|
|
; CHECK: %79 = OpULessThanEqual %bool %71 %78
|
|
; CHECK: OpReturnValue %79
|
|
; CHECK: OpFunctionEnd
|
|
; CHECK: %89 = OpFunction %void None %90
|
|
; CHECK: %91 = OpFunctionParameter %uint
|
|
; CHECK: %92 = OpFunctionParameter %uint
|
|
; CHECK: %93 = OpFunctionParameter %uint
|
|
; CHECK: %94 = OpFunctionParameter %uint
|
|
; CHECK: %95 = OpLabel
|
|
; CHECK: %101 = OpAccessChain %_ptr_StorageBuffer_uint %99 %uint_0
|
|
; CHECK: %103 = OpAtomicIAdd %uint %101 %uint_4 %uint_0 %uint_10
|
|
; CHECK: %104 = OpIAdd %uint %103 %uint_10
|
|
; CHECK: %105 = OpArrayLength %uint %99 1
|
|
; CHECK: %106 = OpULessThanEqual %bool %104 %105
|
|
; CHECK: OpSelectionMerge %107 None
|
|
; CHECK: OpBranchConditional %106 %108 %107
|
|
; CHECK: %108 = OpLabel
|
|
; CHECK: %109 = OpIAdd %uint %103 %uint_0
|
|
; CHECK: %110 = OpAccessChain %_ptr_StorageBuffer_uint %99 %uint_1 %109
|
|
; CHECK: OpStore %110 %uint_10
|
|
; CHECK: %112 = OpIAdd %uint %103 %uint_1
|
|
; CHECK: %113 = OpAccessChain %_ptr_StorageBuffer_uint %99 %uint_1 %112
|
|
; CHECK: OpStore %113 %uint_23
|
|
; CHECK: %114 = OpIAdd %uint %103 %uint_2
|
|
; CHECK: %115 = OpAccessChain %_ptr_StorageBuffer_uint %99 %uint_1 %114
|
|
; CHECK: OpStore %115 %91
|
|
; CHECK: %117 = OpIAdd %uint %103 %uint_3
|
|
; CHECK: %118 = OpAccessChain %_ptr_StorageBuffer_uint %99 %uint_1 %117
|
|
; CHECK: OpStore %118 %uint_4
|
|
; CHECK: %122 = OpLoad %v4float %gl_FragCoord
|
|
; CHECK: %124 = OpBitcast %v4uint %122
|
|
; CHECK: %125 = OpCompositeExtract %uint %124 0
|
|
; CHECK: %126 = OpIAdd %uint %103 %uint_4
|
|
; CHECK: %127 = OpAccessChain %_ptr_StorageBuffer_uint %99 %uint_1 %126
|
|
; CHECK: OpStore %127 %125
|
|
; CHECK: %128 = OpCompositeExtract %uint %124 1
|
|
; CHECK: %130 = OpIAdd %uint %103 %uint_5
|
|
; CHECK: %131 = OpAccessChain %_ptr_StorageBuffer_uint %99 %uint_1 %130
|
|
; CHECK: OpStore %131 %128
|
|
; CHECK: %133 = OpIAdd %uint %103 %uint_7
|
|
; CHECK: %134 = OpAccessChain %_ptr_StorageBuffer_uint %99 %uint_1 %133
|
|
; CHECK: OpStore %134 %92
|
|
; CHECK: %136 = OpIAdd %uint %103 %uint_8
|
|
; CHECK: %137 = OpAccessChain %_ptr_StorageBuffer_uint %99 %uint_1 %136
|
|
; CHECK: OpStore %137 %93
|
|
; CHECK: %139 = OpIAdd %uint %103 %uint_9
|
|
; CHECK: %140 = OpAccessChain %_ptr_StorageBuffer_uint %99 %uint_1 %139
|
|
; CHECK: OpStore %140 %94
|
|
; CHECK: OpBranch %107
|
|
; CHECK: %107 = OpLabel
|
|
; CHECK: OpReturn
|
|
; CHECK: OpFunctionEnd
|
|
)";
|
|
|
|
SetTargetEnv(SPV_ENV_VULKAN_1_2);
|
|
SetAssembleOptions(SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
|
|
SinglePassRunAndMatch<InstBuffAddrCheckPass>(
|
|
defs + decorates + globals + main + output_funcs, true);
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace opt
|
|
} // namespace spvtools
|