mirror of
https://github.com/KhronosGroup/SPIRV-Tools
synced 2024-12-27 10:20:14 +00:00
SROA: Do replacement on structs with no partial references.
This commit is contained in:
parent
06cdb96984
commit
ca4457b4b6
@ -592,9 +592,9 @@ bool ScalarReplacementPass::CheckUses(const ir::Instruction* inst) const {
|
||||
VariableStats stats = {0, 0};
|
||||
bool ok = CheckUses(inst, &stats);
|
||||
|
||||
// TODO(alanbaker): Extend this to some meaningful heuristics about when
|
||||
// SRoA is valuable.
|
||||
if (stats.num_partial_accesses == 0) ok = false;
|
||||
// TODO(alanbaker/greg-lunarg): Add some meaningful heuristics about when
|
||||
// SRoA is costly, such as when the structure has many (unaccessed?)
|
||||
// members.
|
||||
|
||||
return ok;
|
||||
}
|
||||
|
@ -455,24 +455,32 @@ TEST_F(ScalarReplacementTest, NonUniformCompositeInitialization) {
|
||||
const std::string text = R"(
|
||||
;
|
||||
; CHECK: [[uint:%\w+]] = OpTypeInt 32 0
|
||||
; CHECK: [[long:%\w+]] = OpTypeInt 64 1
|
||||
; CHECK: [[dvector:%\w+]] = OpTypeVector
|
||||
; CHECK: [[vector:%\w+]] = OpTypeVector
|
||||
; CHECK: [[array:%\w+]] = OpTypeArray
|
||||
; CHECK: [[matrix:%\w+]] = OpTypeMatrix
|
||||
; CHECK: [[struct1:%\w+]] = OpTypeStruct [[uint]]
|
||||
; CHECK: [[struct1:%\w+]] = OpTypeStruct [[uint]] [[vector]]
|
||||
; CHECK: [[struct2:%\w+]] = OpTypeStruct [[struct1]] [[matrix]] [[array]] [[uint]]
|
||||
; CHECK: [[struct1_ptr:%\w+]] = OpTypePointer Function [[struct1]]
|
||||
; CHECK: [[matrix_ptr:%\w+]] = OpTypePointer Function [[matrix]]
|
||||
; CHECK: [[array_ptr:%\w+]] = OpTypePointer Function [[array]]
|
||||
; CHECK: [[uint_ptr:%\w+]] = OpTypePointer Function [[uint]]
|
||||
; CHECK: [[struct2_ptr:%\w+]] = OpTypePointer Function [[struct2]]
|
||||
; CHECK: [[const_uint:%\w+]] = OpConstant [[uint]]
|
||||
; CHECK: [[const_array:%\w+]] = OpConstantComposite [[array]]
|
||||
; CHECK: [[const_matrix:%\w+]] = OpConstantNull [[matrix]]
|
||||
; CHECK: [[const_struct1:%\w+]] = OpConstantComposite [[struct1]]
|
||||
; CHECK: [[vector_ptr:%\w+]] = OpTypePointer Function [[vector]]
|
||||
; CHECK: [[long_ptr:%\w+]] = OpTypePointer Function [[long]]
|
||||
; CHECK-NOT: OpVariable [[struct2_ptr]] Function
|
||||
; CHECK: OpVariable [[uint_ptr]] Function [[const_uint]]
|
||||
; CHECK-NEXT: OpVariable [[array_ptr]] Function [[const_array]]
|
||||
; CHECK: OpVariable [[long_ptr]] Function
|
||||
; CHECK: OpVariable [[long_ptr]] Function
|
||||
; CHECK: OpVariable [[long_ptr]] Function
|
||||
; CHECK: OpVariable [[vector_ptr]] Function
|
||||
; CHECK: OpVariable [[uint_ptr]] Function
|
||||
; CHECK: OpVariable [[uint_ptr]] Function
|
||||
; CHECK-NEXT: OpVariable [[matrix_ptr]] Function [[const_matrix]]
|
||||
; CHECK-NEXT: OpVariable [[struct1_ptr]] Function [[const_struct1]]
|
||||
; CHECK-NOT: OpVariable [[struct1_ptr]] Function [[const_struct1]]
|
||||
; CHECK-NOT: OpVariable [[struct2_ptr]] Function
|
||||
;
|
||||
OpCapability Shader
|
||||
@ -891,10 +899,10 @@ OpFunctionEnd
|
||||
TEST_F(ScalarReplacementTest, NoPartialAccesses) {
|
||||
const std::string text = R"(
|
||||
;
|
||||
; CHECK: [[struct:%\w+]] = OpTypeStruct
|
||||
; CHECK: [[struct_ptr:%\w+]] = OpTypePointer Function [[struct]]
|
||||
; CHECK: [[uint:%\w+]] = OpTypeInt 32 0
|
||||
; CHECK: [[uint_ptr:%\w+]] = OpTypePointer Function [[uint]]
|
||||
; CHECK: OpLabel
|
||||
; CHECK-NEXT: OpVariable [[struct_ptr]]
|
||||
; CHECK-NEXT: OpVariable [[uint_ptr]]
|
||||
; CHECK-NOT: OpVariable
|
||||
;
|
||||
OpCapability Shader
|
||||
@ -1136,6 +1144,130 @@ OpFunctionEnd
|
||||
|
||||
SinglePassRunAndMatch<opt::ScalarReplacementPass>(text, true);
|
||||
}
|
||||
|
||||
TEST_F(ScalarReplacementTest, NoPartialAccesses2) {
|
||||
const std::string text = R"(
|
||||
;
|
||||
; CHECK: [[float:%\w+]] = OpTypeFloat 32
|
||||
; CHECK: [[float_ptr:%\w+]] = OpTypePointer Function [[float]]
|
||||
; CHECK: OpVariable [[float_ptr]] Function
|
||||
; CHECK: OpVariable [[float_ptr]] Function
|
||||
; CHECK: OpVariable [[float_ptr]] Function
|
||||
; CHECK: OpVariable [[float_ptr]] Function
|
||||
; CHECK: OpVariable [[float_ptr]] Function
|
||||
; CHECK: OpVariable [[float_ptr]] Function
|
||||
; CHECK: OpVariable [[float_ptr]] Function
|
||||
; CHECK: OpVariable [[float_ptr]] Function
|
||||
; CHECK-NOT: OpVariable
|
||||
;
|
||||
OpCapability Shader
|
||||
%1 = OpExtInstImport "GLSL.std.450"
|
||||
OpMemoryModel Logical GLSL450
|
||||
OpEntryPoint Fragment %main "main" %fo
|
||||
OpExecutionMode %main OriginUpperLeft
|
||||
OpSource GLSL 430
|
||||
OpName %main "main"
|
||||
OpName %S "S"
|
||||
OpMemberName %S 0 "x"
|
||||
OpMemberName %S 1 "y"
|
||||
OpName %ts1 "ts1"
|
||||
OpName %S_0 "S"
|
||||
OpMemberName %S_0 0 "x"
|
||||
OpMemberName %S_0 1 "y"
|
||||
OpName %U_t "U_t"
|
||||
OpMemberName %U_t 0 "g_s1"
|
||||
OpMemberName %U_t 1 "g_s2"
|
||||
OpMemberName %U_t 2 "g_s3"
|
||||
OpName %_ ""
|
||||
OpName %ts2 "ts2"
|
||||
OpName %_Globals_ "_Globals_"
|
||||
OpMemberName %_Globals_ 0 "g_b"
|
||||
OpName %__0 ""
|
||||
OpName %ts3 "ts3"
|
||||
OpName %ts4 "ts4"
|
||||
OpName %fo "fo"
|
||||
OpMemberDecorate %S_0 0 Offset 0
|
||||
OpMemberDecorate %S_0 1 Offset 4
|
||||
OpMemberDecorate %U_t 0 Offset 0
|
||||
OpMemberDecorate %U_t 1 Offset 8
|
||||
OpMemberDecorate %U_t 2 Offset 16
|
||||
OpDecorate %U_t BufferBlock
|
||||
OpDecorate %_ DescriptorSet 0
|
||||
OpMemberDecorate %_Globals_ 0 Offset 0
|
||||
OpDecorate %_Globals_ Block
|
||||
OpDecorate %__0 DescriptorSet 0
|
||||
OpDecorate %__0 Binding 0
|
||||
OpDecorate %fo Location 0
|
||||
%void = OpTypeVoid
|
||||
%15 = OpTypeFunction %void
|
||||
%float = OpTypeFloat 32
|
||||
%S = OpTypeStruct %float %float
|
||||
%_ptr_Function_S = OpTypePointer Function %S
|
||||
%S_0 = OpTypeStruct %float %float
|
||||
%U_t = OpTypeStruct %S_0 %S_0 %S_0
|
||||
%_ptr_Uniform_U_t = OpTypePointer Uniform %U_t
|
||||
%_ = OpVariable %_ptr_Uniform_U_t Uniform
|
||||
%int = OpTypeInt 32 1
|
||||
%int_0 = OpConstant %int 0
|
||||
%_ptr_Uniform_S_0 = OpTypePointer Uniform %S_0
|
||||
%_ptr_Function_float = OpTypePointer Function %float
|
||||
%int_1 = OpConstant %int 1
|
||||
%uint = OpTypeInt 32 0
|
||||
%_Globals_ = OpTypeStruct %uint
|
||||
%_ptr_Uniform__Globals_ = OpTypePointer Uniform %_Globals_
|
||||
%__0 = OpVariable %_ptr_Uniform__Globals_ Uniform
|
||||
%_ptr_Uniform_uint = OpTypePointer Uniform %uint
|
||||
%bool = OpTypeBool
|
||||
%uint_0 = OpConstant %uint 0
|
||||
%_ptr_Output_float = OpTypePointer Output %float
|
||||
%fo = OpVariable %_ptr_Output_float Output
|
||||
%main = OpFunction %void None %15
|
||||
%30 = OpLabel
|
||||
%ts1 = OpVariable %_ptr_Function_S Function
|
||||
%ts2 = OpVariable %_ptr_Function_S Function
|
||||
%ts3 = OpVariable %_ptr_Function_S Function
|
||||
%ts4 = OpVariable %_ptr_Function_S Function
|
||||
%31 = OpAccessChain %_ptr_Uniform_S_0 %_ %int_0
|
||||
%32 = OpLoad %S_0 %31
|
||||
%33 = OpCompositeExtract %float %32 0
|
||||
%34 = OpAccessChain %_ptr_Function_float %ts1 %int_0
|
||||
OpStore %34 %33
|
||||
%35 = OpCompositeExtract %float %32 1
|
||||
%36 = OpAccessChain %_ptr_Function_float %ts1 %int_1
|
||||
OpStore %36 %35
|
||||
%37 = OpAccessChain %_ptr_Uniform_S_0 %_ %int_1
|
||||
%38 = OpLoad %S_0 %37
|
||||
%39 = OpCompositeExtract %float %38 0
|
||||
%40 = OpAccessChain %_ptr_Function_float %ts2 %int_0
|
||||
OpStore %40 %39
|
||||
%41 = OpCompositeExtract %float %38 1
|
||||
%42 = OpAccessChain %_ptr_Function_float %ts2 %int_1
|
||||
OpStore %42 %41
|
||||
%43 = OpAccessChain %_ptr_Uniform_uint %__0 %int_0
|
||||
%44 = OpLoad %uint %43
|
||||
%45 = OpINotEqual %bool %44 %uint_0
|
||||
OpSelectionMerge %46 None
|
||||
OpBranchConditional %45 %47 %48
|
||||
%47 = OpLabel
|
||||
%49 = OpLoad %S %ts1
|
||||
OpStore %ts3 %49
|
||||
OpBranch %46
|
||||
%48 = OpLabel
|
||||
%50 = OpLoad %S %ts2
|
||||
OpStore %ts3 %50
|
||||
OpBranch %46
|
||||
%46 = OpLabel
|
||||
%51 = OpLoad %S %ts3
|
||||
OpStore %ts4 %51
|
||||
%52 = OpAccessChain %_ptr_Function_float %ts4 %int_1
|
||||
%53 = OpLoad %float %52
|
||||
OpStore %fo %53
|
||||
OpReturn
|
||||
OpFunctionEnd
|
||||
)";
|
||||
|
||||
SinglePassRunAndMatch<opt::ScalarReplacementPass>(text, true);
|
||||
}
|
||||
#endif // SPIRV_EFFCEE
|
||||
|
||||
} // namespace
|
||||
|
Loading…
Reference in New Issue
Block a user