| // Copyright (c) 2018 Google LLC. |
| // |
| // 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 <string> |
| |
| #include "effcee/effcee.h" |
| #include "gmock/gmock.h" |
| #include "test/opt/pass_fixture.h" |
| |
| namespace spvtools { |
| namespace opt { |
| namespace { |
| |
| using UnswitchTest = PassTest<::testing::Test>; |
| |
| /* |
| Generated from the following GLSL + --eliminate-local-multi-store |
| |
| #version 450 core |
| uniform vec4 c; |
| void main() { |
| int i = 0; |
| int j = 0; |
| bool cond = c[0] == 0; |
| for (; i < 10; i++, j++) { |
| if (cond) { |
| i++; |
| } |
| else { |
| j++; |
| } |
| } |
| } |
| */ |
| TEST_F(UnswitchTest, SimpleUnswitch) { |
| const std::string text = R"( |
| ; CHECK: [[cst_cond:%\w+]] = OpFOrdEqual |
| ; CHECK-NEXT: OpSelectionMerge [[if_merge:%\w+]] None |
| ; CHECK-NEXT: OpBranchConditional [[cst_cond]] [[loop_t:%\w+]] [[loop_f:%\w+]] |
| |
| ; Loop specialized for false. |
| ; CHECK: [[loop_f]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[loop:%\w+]] |
| ; CHECK: [[loop]] = OpLabel |
| ; CHECK-NEXT: [[phi_i:%\w+]] = OpPhi %int %int_0 [[loop_f]] [[iv_i:%\w+]] [[continue:%\w+]] |
| ; CHECK-NEXT: [[phi_j:%\w+]] = OpPhi %int %int_0 [[loop_f]] [[iv_j:%\w+]] [[continue]] |
| ; CHECK-NEXT: OpLoopMerge [[merge:%\w+]] [[continue]] None |
| ; CHECK: [[loop_exit:%\w+]] = OpSLessThan {{%\w+}} [[phi_i]] {{%\w+}} |
| ; CHECK-NEXT: OpBranchConditional [[loop_exit]] [[loop_body:%\w+]] [[merge]] |
| ; [[loop_body]] = OpLabel |
| ; CHECK: OpSelectionMerge [[sel_merge:%\w+]] None |
| ; CHECK: OpBranchConditional %false [[bb1:%\w+]] [[bb2:%\w+]] |
| ; CHECK: [[bb2]] = OpLabel |
| ; CHECK-NEXT: [[inc_j:%\w+]] = OpIAdd %int [[phi_j]] %int_1 |
| ; CHECK-NEXT: OpBranch [[sel_merge]] |
| ; CHECK: [[bb1]] = OpLabel |
| ; CHECK-NEXT: [[inc_i:%\w+]] = OpIAdd %int [[phi_i]] %int_1 |
| ; CHECK-NEXT: OpBranch [[sel_merge]] |
| ; CHECK: [[sel_merge]] = OpLabel |
| ; CHECK: OpBranch [[if_merge]] |
| |
| ; Loop specialized for true. |
| ; CHECK: [[loop_t]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[loop:%\w+]] |
| ; CHECK: [[loop]] = OpLabel |
| ; CHECK-NEXT: [[phi_i:%\w+]] = OpPhi %int %int_0 [[loop_t]] [[iv_i:%\w+]] [[continue:%\w+]] |
| ; CHECK-NEXT: [[phi_j:%\w+]] = OpPhi %int %int_0 [[loop_t]] [[iv_j:%\w+]] [[continue]] |
| ; CHECK-NEXT: OpLoopMerge [[merge:%\w+]] [[continue]] None |
| ; CHECK: [[loop_exit:%\w+]] = OpSLessThan {{%\w+}} [[phi_i]] {{%\w+}} |
| ; CHECK-NEXT: OpBranchConditional [[loop_exit]] [[loop_body:%\w+]] [[merge]] |
| ; [[loop_body]] = OpLabel |
| ; CHECK: OpSelectionMerge [[sel_merge:%\w+]] None |
| ; CHECK: OpBranchConditional %true [[bb1:%\w+]] [[bb2:%\w+]] |
| ; CHECK: [[bb1]] = OpLabel |
| ; CHECK-NEXT: [[inc_i:%\w+]] = OpIAdd %int [[phi_i]] %int_1 |
| ; CHECK-NEXT: OpBranch [[sel_merge]] |
| ; CHECK: [[bb2]] = OpLabel |
| ; CHECK-NEXT: [[inc_j:%\w+]] = OpIAdd %int [[phi_j]] %int_1 |
| ; CHECK-NEXT: OpBranch [[sel_merge]] |
| ; CHECK: [[sel_merge]] = OpLabel |
| ; CHECK: OpBranch [[if_merge]] |
| |
| ; CHECK: [[if_merge]] = OpLabel |
| ; CHECK-NEXT: OpReturn |
| |
| OpCapability Shader |
| %1 = OpExtInstImport "GLSL.std.450" |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint Fragment %main "main" |
| OpExecutionMode %main OriginLowerLeft |
| OpSource GLSL 450 |
| OpName %main "main" |
| OpName %c "c" |
| OpDecorate %c Location 0 |
| OpDecorate %c DescriptorSet 0 |
| %void = OpTypeVoid |
| %3 = OpTypeFunction %void |
| %int = OpTypeInt 32 1 |
| %_ptr_Function_int = OpTypePointer Function %int |
| %int_0 = OpConstant %int 0 |
| %bool = OpTypeBool |
| %_ptr_Function_bool = OpTypePointer Function %bool |
| %float = OpTypeFloat 32 |
| %v4float = OpTypeVector %float 4 |
| %_ptr_UniformConstant_v4float = OpTypePointer UniformConstant %v4float |
| %c = OpVariable %_ptr_UniformConstant_v4float UniformConstant |
| %uint = OpTypeInt 32 0 |
| %uint_0 = OpConstant %uint 0 |
| %_ptr_UniformConstant_float = OpTypePointer UniformConstant %float |
| %float_0 = OpConstant %float 0 |
| %int_10 = OpConstant %int 10 |
| %int_1 = OpConstant %int 1 |
| %main = OpFunction %void None %3 |
| %5 = OpLabel |
| %21 = OpAccessChain %_ptr_UniformConstant_float %c %uint_0 |
| %22 = OpLoad %float %21 |
| %24 = OpFOrdEqual %bool %22 %float_0 |
| OpBranch %25 |
| %25 = OpLabel |
| %46 = OpPhi %int %int_0 %5 %43 %28 |
| %47 = OpPhi %int %int_0 %5 %45 %28 |
| OpLoopMerge %27 %28 None |
| OpBranch %29 |
| %29 = OpLabel |
| %32 = OpSLessThan %bool %46 %int_10 |
| OpBranchConditional %32 %26 %27 |
| %26 = OpLabel |
| OpSelectionMerge %35 None |
| OpBranchConditional %24 %34 %39 |
| %34 = OpLabel |
| %38 = OpIAdd %int %46 %int_1 |
| OpBranch %35 |
| %39 = OpLabel |
| %41 = OpIAdd %int %47 %int_1 |
| OpBranch %35 |
| %35 = OpLabel |
| %48 = OpPhi %int %38 %34 %46 %39 |
| %49 = OpPhi %int %47 %34 %41 %39 |
| OpBranch %28 |
| %28 = OpLabel |
| %43 = OpIAdd %int %48 %int_1 |
| %45 = OpIAdd %int %49 %int_1 |
| OpBranch %25 |
| %27 = OpLabel |
| OpReturn |
| OpFunctionEnd |
| )"; |
| |
| SinglePassRunAndMatch<LoopUnswitchPass>(text, true); |
| } |
| |
| /* |
| Generated from the following GLSL + --eliminate-local-multi-store |
| |
| #version 330 core |
| in vec4 c; |
| void main() { |
| int i = 0; |
| bool cond = c[0] == 0; |
| for (; i < 10; i++) { |
| if (cond) { |
| i++; |
| } |
| else { |
| return; |
| } |
| } |
| } |
| */ |
| TEST_F(UnswitchTest, UnswitchExit) { |
| const std::string text = R"( |
| ; CHECK: [[cst_cond:%\w+]] = OpFOrdEqual |
| ; CHECK-NEXT: OpSelectionMerge [[if_merge:%\w+]] None |
| ; CHECK-NEXT: OpBranchConditional [[cst_cond]] [[loop_t:%\w+]] [[loop_f:%\w+]] |
| |
| ; Loop specialized for false. |
| ; CHECK: [[loop_f]] = OpLabel |
| ; CHECK: OpReturn |
| |
| ; Loop specialized for true. |
| ; CHECK: [[loop_t]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[loop:%\w+]] |
| ; CHECK: [[loop]] = OpLabel |
| ; CHECK-NEXT: [[phi_i:%\w+]] = OpPhi %int %int_0 [[loop_t]] [[iv_i:%\w+]] [[continue:%\w+]] |
| ; CHECK-NEXT: OpLoopMerge [[merge:%\w+]] [[continue]] None |
| ; CHECK: [[loop_exit:%\w+]] = OpSLessThan {{%\w+}} [[phi_i]] {{%\w+}} |
| ; CHECK-NEXT: OpBranchConditional [[loop_exit]] {{%\w+}} [[merge]] |
| ; Check that we have i+=2. |
| ; CHECK: [[phi_i:%\w+]] = OpIAdd %int [[phi_i]] %int_1 |
| ; CHECK: [[iv_i]] = OpIAdd %int [[phi_i]] %int_1 |
| ; CHECK: [[merge]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[if_merge]] |
| |
| ; CHECK: [[if_merge]] = OpLabel |
| ; CHECK-NEXT: OpReturn |
| |
| OpCapability Shader |
| %1 = OpExtInstImport "GLSL.std.450" |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint Fragment %main "main" %c |
| OpExecutionMode %main OriginUpperLeft |
| OpSource GLSL 330 |
| OpName %main "main" |
| OpName %c "c" |
| OpDecorate %c Location 0 |
| OpDecorate %23 Uniform |
| %void = OpTypeVoid |
| %3 = OpTypeFunction %void |
| %int = OpTypeInt 32 1 |
| %_ptr_Function_int = OpTypePointer Function %int |
| %int_0 = OpConstant %int 0 |
| %bool = OpTypeBool |
| %_ptr_Function_bool = OpTypePointer Function %bool |
| %float = OpTypeFloat 32 |
| %v4float = OpTypeVector %float 4 |
| %_ptr_Input_v4float = OpTypePointer Input %v4float |
| %c = OpVariable %_ptr_Input_v4float Input |
| %uint = OpTypeInt 32 0 |
| %uint_0 = OpConstant %uint 0 |
| %_ptr_Input_float = OpTypePointer Input %float |
| %float_0 = OpConstant %float 0 |
| %int_10 = OpConstant %int 10 |
| %int_1 = OpConstant %int 1 |
| %main = OpFunction %void None %3 |
| %5 = OpLabel |
| %20 = OpAccessChain %_ptr_Input_float %c %uint_0 |
| %21 = OpLoad %float %20 |
| %23 = OpFOrdEqual %bool %21 %float_0 |
| OpBranch %24 |
| %24 = OpLabel |
| %42 = OpPhi %int %int_0 %5 %41 %27 |
| OpLoopMerge %26 %27 None |
| OpBranch %28 |
| %28 = OpLabel |
| %31 = OpSLessThan %bool %42 %int_10 |
| OpBranchConditional %31 %25 %26 |
| %25 = OpLabel |
| OpSelectionMerge %34 None |
| OpBranchConditional %23 %33 %38 |
| %33 = OpLabel |
| %37 = OpIAdd %int %42 %int_1 |
| OpBranch %34 |
| %38 = OpLabel |
| OpReturn |
| %34 = OpLabel |
| OpBranch %27 |
| %27 = OpLabel |
| %41 = OpIAdd %int %37 %int_1 |
| OpBranch %24 |
| %26 = OpLabel |
| OpReturn |
| OpFunctionEnd |
| )"; |
| |
| SinglePassRunAndMatch<LoopUnswitchPass>(text, true); |
| } |
| |
| /* |
| Generated from the following GLSL + --eliminate-local-multi-store |
| |
| #version 330 core |
| in vec4 c; |
| void main() { |
| int i = 0; |
| bool cond = c[0] == 0; |
| for (; i < 10; i++) { |
| if (cond) { |
| continue; |
| } |
| else { |
| i++; |
| } |
| } |
| } |
| */ |
| TEST_F(UnswitchTest, UnswitchContinue) { |
| const std::string text = R"( |
| ; CHECK: [[cst_cond:%\w+]] = OpFOrdEqual |
| ; CHECK-NEXT: OpSelectionMerge [[if_merge:%\w+]] None |
| ; CHECK-NEXT: OpBranchConditional [[cst_cond]] [[loop_t:%\w+]] [[loop_f:%\w+]] |
| |
| ; Loop specialized for false. |
| ; CHECK: [[loop_f]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[loop:%\w+]] |
| ; CHECK: [[loop]] = OpLabel |
| ; CHECK-NEXT: [[phi_i:%\w+]] = OpPhi %int %int_0 [[loop_f]] [[iv_i:%\w+]] [[continue:%\w+]] |
| ; CHECK-NEXT: OpLoopMerge [[merge:%\w+]] [[continue]] None |
| ; CHECK: [[loop_exit:%\w+]] = OpSLessThan {{%\w+}} [[phi_i]] {{%\w+}} |
| ; CHECK-NEXT: OpBranchConditional [[loop_exit]] [[loop_body:%\w+]] [[merge]] |
| ; CHECK: [[loop_body:%\w+]] = OpLabel |
| ; CHECK-NEXT: OpSelectionMerge |
| ; CHECK-NEXT: OpBranchConditional %false |
| ; CHECK: [[merge]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[if_merge]] |
| |
| ; Loop specialized for true. |
| ; CHECK: [[loop_t]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[loop:%\w+]] |
| ; CHECK: [[loop]] = OpLabel |
| ; CHECK-NEXT: [[phi_i:%\w+]] = OpPhi %int %int_0 [[loop_t]] [[iv_i:%\w+]] [[continue:%\w+]] |
| ; CHECK-NEXT: OpLoopMerge [[merge:%\w+]] [[continue]] None |
| ; CHECK: [[loop_exit:%\w+]] = OpSLessThan {{%\w+}} [[phi_i]] {{%\w+}} |
| ; CHECK-NEXT: OpBranchConditional [[loop_exit]] [[loop_body:%\w+]] [[merge]] |
| ; CHECK: [[loop_body:%\w+]] = OpLabel |
| ; CHECK-NEXT: OpSelectionMerge |
| ; CHECK-NEXT: OpBranchConditional %true |
| ; CHECK: [[merge]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[if_merge]] |
| |
| ; CHECK: [[if_merge]] = OpLabel |
| ; CHECK-NEXT: OpReturn |
| |
| OpCapability Shader |
| %1 = OpExtInstImport "GLSL.std.450" |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint Fragment %main "main" %c |
| OpExecutionMode %main OriginUpperLeft |
| OpSource GLSL 330 |
| OpName %main "main" |
| OpName %c "c" |
| OpDecorate %c Location 0 |
| OpDecorate %23 Uniform |
| %void = OpTypeVoid |
| %3 = OpTypeFunction %void |
| %int = OpTypeInt 32 1 |
| %_ptr_Function_int = OpTypePointer Function %int |
| %int_0 = OpConstant %int 0 |
| %bool = OpTypeBool |
| %_ptr_Function_bool = OpTypePointer Function %bool |
| %float = OpTypeFloat 32 |
| %v4float = OpTypeVector %float 4 |
| %_ptr_Input_v4float = OpTypePointer Input %v4float |
| %c = OpVariable %_ptr_Input_v4float Input |
| %uint = OpTypeInt 32 0 |
| %uint_0 = OpConstant %uint 0 |
| %_ptr_Input_float = OpTypePointer Input %float |
| %float_0 = OpConstant %float 0 |
| %int_10 = OpConstant %int 10 |
| %int_1 = OpConstant %int 1 |
| %main = OpFunction %void None %3 |
| %5 = OpLabel |
| %20 = OpAccessChain %_ptr_Input_float %c %uint_0 |
| %21 = OpLoad %float %20 |
| %23 = OpFOrdEqual %bool %21 %float_0 |
| OpBranch %24 |
| %24 = OpLabel |
| %42 = OpPhi %int %int_0 %5 %41 %27 |
| OpLoopMerge %26 %27 None |
| OpBranch %28 |
| %28 = OpLabel |
| %31 = OpSLessThan %bool %42 %int_10 |
| OpBranchConditional %31 %25 %26 |
| %25 = OpLabel |
| OpSelectionMerge %34 None |
| OpBranchConditional %23 %33 %36 |
| %33 = OpLabel |
| OpBranch %27 |
| %36 = OpLabel |
| %39 = OpIAdd %int %42 %int_1 |
| OpBranch %34 |
| %34 = OpLabel |
| OpBranch %27 |
| %27 = OpLabel |
| %43 = OpPhi %int %42 %33 %39 %34 |
| %41 = OpIAdd %int %43 %int_1 |
| OpBranch %24 |
| %26 = OpLabel |
| OpReturn |
| OpFunctionEnd |
| )"; |
| |
| SinglePassRunAndMatch<LoopUnswitchPass>(text, true); |
| } |
| |
| /* |
| Generated from the following GLSL + --eliminate-local-multi-store |
| |
| #version 330 core |
| in vec4 c; |
| void main() { |
| int i = 0; |
| bool cond = c[0] == 0; |
| for (; i < 10; i++) { |
| if (cond) { |
| i++; |
| } |
| else { |
| break; |
| } |
| } |
| } |
| */ |
| TEST_F(UnswitchTest, UnswitchKillLoop) { |
| const std::string text = R"( |
| ; CHECK: [[cst_cond:%\w+]] = OpFOrdEqual |
| ; CHECK-NEXT: OpSelectionMerge [[if_merge:%\w+]] None |
| ; CHECK-NEXT: OpBranchConditional [[cst_cond]] [[loop_t:%\w+]] [[loop_f:%\w+]] |
| |
| ; Loop specialized for false. |
| ; CHECK: [[loop_f]] = OpLabel |
| ; CHECK: OpBranch [[if_merge]] |
| |
| ; Loop specialized for true. |
| ; CHECK: [[loop_t]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[loop:%\w+]] |
| ; CHECK: [[loop]] = OpLabel |
| ; CHECK-NEXT: [[phi_i:%\w+]] = OpPhi %int %int_0 [[loop_t]] [[iv_i:%\w+]] [[continue:%\w+]] |
| ; CHECK-NEXT: OpLoopMerge [[merge:%\w+]] [[continue]] None |
| ; CHECK: [[loop_exit:%\w+]] = OpSLessThan {{%\w+}} [[phi_i]] {{%\w+}} |
| ; CHECK-NEXT: OpBranchConditional [[loop_exit]] {{%\w+}} [[merge]] |
| ; Check that we have i+=2. |
| ; CHECK: [[phi_i:%\w+]] = OpIAdd %int [[phi_i]] %int_1 |
| ; CHECK: [[iv_i]] = OpIAdd %int [[phi_i]] %int_1 |
| ; CHECK: [[merge]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[if_merge]] |
| |
| ; CHECK: [[if_merge]] = OpLabel |
| ; CHECK-NEXT: OpReturn |
| |
| OpCapability Shader |
| %1 = OpExtInstImport "GLSL.std.450" |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint Fragment %main "main" %c |
| OpExecutionMode %main OriginUpperLeft |
| OpSource GLSL 330 |
| OpName %main "main" |
| OpName %c "c" |
| OpDecorate %c Location 0 |
| OpDecorate %23 Uniform |
| %void = OpTypeVoid |
| %3 = OpTypeFunction %void |
| %int = OpTypeInt 32 1 |
| %_ptr_Function_int = OpTypePointer Function %int |
| %int_0 = OpConstant %int 0 |
| %bool = OpTypeBool |
| %_ptr_Function_bool = OpTypePointer Function %bool |
| %float = OpTypeFloat 32 |
| %v4float = OpTypeVector %float 4 |
| %_ptr_Input_v4float = OpTypePointer Input %v4float |
| %c = OpVariable %_ptr_Input_v4float Input |
| %uint = OpTypeInt 32 0 |
| %uint_0 = OpConstant %uint 0 |
| %_ptr_Input_float = OpTypePointer Input %float |
| %float_0 = OpConstant %float 0 |
| %int_10 = OpConstant %int 10 |
| %int_1 = OpConstant %int 1 |
| %main = OpFunction %void None %3 |
| %5 = OpLabel |
| %20 = OpAccessChain %_ptr_Input_float %c %uint_0 |
| %21 = OpLoad %float %20 |
| %23 = OpFOrdEqual %bool %21 %float_0 |
| OpBranch %24 |
| %24 = OpLabel |
| %42 = OpPhi %int %int_0 %5 %41 %27 |
| OpLoopMerge %26 %27 None |
| OpBranch %28 |
| %28 = OpLabel |
| %31 = OpSLessThan %bool %42 %int_10 |
| OpBranchConditional %31 %25 %26 |
| %25 = OpLabel |
| OpSelectionMerge %34 None |
| OpBranchConditional %23 %33 %38 |
| %33 = OpLabel |
| %37 = OpIAdd %int %42 %int_1 |
| OpBranch %34 |
| %38 = OpLabel |
| OpBranch %26 |
| %34 = OpLabel |
| OpBranch %27 |
| %27 = OpLabel |
| %41 = OpIAdd %int %37 %int_1 |
| OpBranch %24 |
| %26 = OpLabel |
| OpReturn |
| OpFunctionEnd |
| )"; |
| |
| SinglePassRunAndMatch<LoopUnswitchPass>(text, true); |
| } |
| |
| /* |
| Generated from the following GLSL + --eliminate-local-multi-store |
| |
| #version 330 core |
| in vec4 c; |
| void main() { |
| int i = 0; |
| int cond = int(c[0]); |
| for (; i < 10; i++) { |
| switch (cond) { |
| case 0: |
| return; |
| case 1: |
| discard; |
| case 2: |
| break; |
| default: |
| break; |
| } |
| } |
| bool cond2 = i == 9; |
| } |
| */ |
| TEST_F(UnswitchTest, UnswitchSwitch) { |
| const std::string text = R"( |
| ; CHECK: [[cst_cond:%\w+]] = OpConvertFToS |
| ; CHECK-NEXT: OpSelectionMerge [[if_merge:%\w+]] None |
| ; CHECK-NEXT: OpSwitch [[cst_cond]] [[default:%\w+]] 0 [[loop_0:%\w+]] 1 [[loop_1:%\w+]] 2 [[loop_2:%\w+]] |
| |
| ; Loop specialized for 2. |
| ; CHECK: [[loop_2]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[loop:%\w+]] |
| ; CHECK: [[loop]] = OpLabel |
| ; CHECK-NEXT: [[phi_i:%\w+]] = OpPhi %int %int_0 [[loop_2]] [[iv_i:%\w+]] [[continue:%\w+]] |
| ; CHECK-NEXT: OpLoopMerge [[merge:%\w+]] [[continue]] None |
| ; CHECK: [[loop_exit:%\w+]] = OpSLessThan {{%\w+}} [[phi_i]] {{%\w+}} |
| ; CHECK-NEXT: OpBranchConditional [[loop_exit]] [[loop_body:%\w+]] [[merge]] |
| ; CHECK: [[loop_body]] = OpLabel |
| ; CHECK-NEXT: OpSelectionMerge |
| ; CHECK-NEXT: OpSwitch %int_2 |
| ; CHECK: [[merge]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[if_merge]] |
| |
| ; Loop specialized for 1. |
| ; CHECK: [[loop_1]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[loop:%\w+]] |
| ; CHECK: [[loop]] = OpLabel |
| ; CHECK-NEXT: [[phi_i:%\w+]] = OpPhi %int %int_0 [[loop_1]] [[iv_i:%\w+]] [[continue:%\w+]] |
| ; CHECK-NEXT: OpLoopMerge [[merge:%\w+]] [[continue]] None |
| ; CHECK: [[loop_exit:%\w+]] = OpSLessThan {{%\w+}} [[phi_i]] {{%\w+}} |
| ; CHECK-NEXT: OpBranchConditional [[loop_exit]] [[loop_body:%\w+]] [[merge]] |
| ; CHECK: [[loop_body]] = OpLabel |
| ; CHECK-NEXT: OpSelectionMerge |
| ; CHECK-NEXT: OpSwitch %int_1 |
| ; CHECK: [[merge]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[if_merge]] |
| |
| ; Loop specialized for 0. |
| ; CHECK: [[loop_0]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[loop:%\w+]] |
| ; CHECK: [[loop]] = OpLabel |
| ; CHECK-NEXT: [[phi_i:%\w+]] = OpPhi %int %int_0 [[loop_0]] [[iv_i:%\w+]] [[continue:%\w+]] |
| ; CHECK-NEXT: OpLoopMerge [[merge:%\w+]] [[continue]] None |
| ; CHECK: [[loop_exit:%\w+]] = OpSLessThan {{%\w+}} [[phi_i]] {{%\w+}} |
| ; CHECK-NEXT: OpBranchConditional [[loop_exit]] [[loop_body:%\w+]] [[merge]] |
| ; CHECK: [[loop_body]] = OpLabel |
| ; CHECK-NEXT: OpSelectionMerge |
| ; CHECK-NEXT: OpSwitch %int_0 |
| ; CHECK: [[merge]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[if_merge]] |
| |
| ; Loop specialized for the default case. |
| ; CHECK: [[default]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[loop:%\w+]] |
| ; CHECK: [[loop]] = OpLabel |
| ; CHECK-NEXT: [[phi_i:%\w+]] = OpPhi %int %int_0 [[default]] [[iv_i:%\w+]] [[continue:%\w+]] |
| ; CHECK-NEXT: OpLoopMerge [[merge:%\w+]] [[continue]] None |
| ; CHECK: [[loop_exit:%\w+]] = OpSLessThan {{%\w+}} [[phi_i]] {{%\w+}} |
| ; CHECK-NEXT: OpBranchConditional [[loop_exit]] [[loop_body:%\w+]] [[merge]] |
| ; CHECK: [[loop_body]] = OpLabel |
| ; CHECK-NEXT: OpSelectionMerge |
| ; CHECK-NEXT: OpSwitch %uint_3 |
| ; CHECK: [[merge]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[if_merge]] |
| |
| ; CHECK: [[if_merge]] = OpLabel |
| ; CHECK-NEXT: OpReturn |
| OpCapability Shader |
| %1 = OpExtInstImport "GLSL.std.450" |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint Fragment %main "main" %c |
| OpExecutionMode %main OriginUpperLeft |
| OpSource GLSL 330 |
| OpName %main "main" |
| OpName %c "c" |
| OpDecorate %c Location 0 |
| OpDecorate %20 Uniform |
| %void = OpTypeVoid |
| %3 = OpTypeFunction %void |
| %int = OpTypeInt 32 1 |
| %_ptr_Function_int = OpTypePointer Function %int |
| %int_0 = OpConstant %int 0 |
| %float = OpTypeFloat 32 |
| %v4float = OpTypeVector %float 4 |
| %_ptr_Input_v4float = OpTypePointer Input %v4float |
| %c = OpVariable %_ptr_Input_v4float Input |
| %uint = OpTypeInt 32 0 |
| %uint_0 = OpConstant %uint 0 |
| %_ptr_Input_float = OpTypePointer Input %float |
| %int_10 = OpConstant %int 10 |
| %bool = OpTypeBool |
| %int_1 = OpConstant %int 1 |
| %_ptr_Function_bool = OpTypePointer Function %bool |
| %main = OpFunction %void None %3 |
| %5 = OpLabel |
| %18 = OpAccessChain %_ptr_Input_float %c %uint_0 |
| %19 = OpLoad %float %18 |
| %20 = OpConvertFToS %int %19 |
| OpBranch %21 |
| %21 = OpLabel |
| %49 = OpPhi %int %int_0 %5 %43 %24 |
| OpLoopMerge %23 %24 None |
| OpBranch %25 |
| %25 = OpLabel |
| %29 = OpSLessThan %bool %49 %int_10 |
| OpBranchConditional %29 %22 %23 |
| %22 = OpLabel |
| OpSelectionMerge %35 None |
| OpSwitch %20 %34 0 %31 1 %32 2 %33 |
| %34 = OpLabel |
| OpBranch %35 |
| %31 = OpLabel |
| OpReturn |
| %32 = OpLabel |
| OpKill |
| %33 = OpLabel |
| OpBranch %35 |
| %35 = OpLabel |
| OpBranch %24 |
| %24 = OpLabel |
| %43 = OpIAdd %int %49 %int_1 |
| OpBranch %21 |
| %23 = OpLabel |
| OpReturn |
| OpFunctionEnd |
| )"; |
| |
| SinglePassRunAndMatch<LoopUnswitchPass>(text, true); |
| } |
| |
| /* |
| Generated from the following GLSL + --eliminate-local-multi-store |
| |
| #version 440 core |
| layout(location = 0)in vec4 c; |
| void main() { |
| int i = 0; |
| int j = 0; |
| int k = 0; |
| bool cond = c[0] == 0; |
| for (; i < 10; i++) { |
| for (; j < 10; j++) { |
| if (cond) { |
| i++; |
| } else { |
| j++; |
| } |
| } |
| } |
| } |
| */ |
| TEST_F(UnswitchTest, UnSwitchNested) { |
| // Test that an branch can be unswitched out of two nested loops. |
| const std::string text = R"( |
| ; CHECK: [[cst_cond:%\w+]] = OpFOrdEqual |
| ; CHECK-NEXT: OpSelectionMerge [[if_merge:%\w+]] None |
| ; CHECK-NEXT: OpBranchConditional [[cst_cond]] [[loop_t:%\w+]] [[loop_f:%\w+]] |
| |
| ; Loop specialized for false |
| ; CHECK: [[loop_f]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[loop:%\w+]] |
| ; CHECK: [[loop]] = OpLabel |
| ; CHECK-NEXT: {{%\w+}} = OpPhi %int %int_0 [[loop_f]] {{%\w+}} [[continue:%\w+]] |
| ; CHECK-NEXT: {{%\w+}} = OpPhi %int %int_0 [[loop_f]] {{%\w+}} [[continue]] |
| ; CHECK-NEXT: OpLoopMerge [[merge:%\w+]] [[continue]] None |
| ; CHECK-NOT: [[merge]] = OpLabel |
| ; CHECK: OpLoopMerge |
| ; CHECK-NEXT: OpBranch [[bb1:%\w+]] |
| ; CHECK: [[bb1]] = OpLabel |
| ; CHECK-NEXT: OpSLessThan |
| ; CHECK-NEXT: OpBranchConditional {{%\w+}} [[bb2:%\w+]] |
| ; CHECK: [[bb2]] = OpLabel |
| ; CHECK-NEXT: OpSelectionMerge |
| ; CHECK-NEXT: OpBranchConditional %false |
| ; CHECK: [[merge]] = OpLabel |
| |
| ; Loop specialized for true. Same as first loop except the branch condition is true. |
| ; CHECK: [[loop_t]] = OpLabel |
| ; CHECK-NEXT: OpBranch [[loop:%\w+]] |
| ; CHECK: [[loop]] = OpLabel |
| ; CHECK-NEXT: {{%\w+}} = OpPhi %int %int_0 [[loop_t]] {{%\w+}} [[continue:%\w+]] |
| ; CHECK-NEXT: {{%\w+}} = OpPhi %int %int_0 [[loop_t]] {{%\w+}} [[continue]] |
| ; CHECK-NEXT: OpLoopMerge [[merge:%\w+]] [[continue]] None |
| ; CHECK-NOT: [[merge]] = OpLabel |
| ; CHECK: OpLoopMerge |
| ; CHECK-NEXT: OpBranch [[bb1:%\w+]] |
| ; CHECK: [[bb1]] = OpLabel |
| ; CHECK-NEXT: OpSLessThan |
| ; CHECK-NEXT: OpBranchConditional {{%\w+}} [[bb2:%\w+]] |
| ; CHECK: [[bb2]] = OpLabel |
| ; CHECK-NEXT: OpSelectionMerge |
| ; CHECK-NEXT: OpBranchConditional %true |
| ; CHECK: [[merge]] = OpLabel |
| |
| OpCapability Shader |
| %1 = OpExtInstImport "GLSL.std.450" |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint Fragment %main "main" %c |
| OpExecutionMode %main OriginUpperLeft |
| OpSource GLSL 440 |
| OpName %main "main" |
| OpName %c "c" |
| OpDecorate %c Location 0 |
| OpDecorate %25 Uniform |
| %void = OpTypeVoid |
| %3 = OpTypeFunction %void |
| %int = OpTypeInt 32 1 |
| %_ptr_Function_int = OpTypePointer Function %int |
| %int_0 = OpConstant %int 0 |
| %bool = OpTypeBool |
| %_ptr_Function_bool = OpTypePointer Function %bool |
| %float = OpTypeFloat 32 |
| %v4float = OpTypeVector %float 4 |
| %_ptr_Input_v4float = OpTypePointer Input %v4float |
| %c = OpVariable %_ptr_Input_v4float Input |
| %uint = OpTypeInt 32 0 |
| %uint_0 = OpConstant %uint 0 |
| %_ptr_Input_float = OpTypePointer Input %float |
| %float_0 = OpConstant %float 0 |
| %int_10 = OpConstant %int 10 |
| %int_1 = OpConstant %int 1 |
| %main = OpFunction %void None %3 |
| %5 = OpLabel |
| %22 = OpAccessChain %_ptr_Input_float %c %uint_0 |
| %23 = OpLoad %float %22 |
| %25 = OpFOrdEqual %bool %23 %float_0 |
| OpBranch %26 |
| %26 = OpLabel |
| %67 = OpPhi %int %int_0 %5 %52 %29 |
| %68 = OpPhi %int %int_0 %5 %70 %29 |
| OpLoopMerge %28 %29 None |
| OpBranch %30 |
| %30 = OpLabel |
| %33 = OpSLessThan %bool %67 %int_10 |
| OpBranchConditional %33 %27 %28 |
| %27 = OpLabel |
| OpBranch %34 |
| %34 = OpLabel |
| %69 = OpPhi %int %67 %27 %46 %37 |
| %70 = OpPhi %int %68 %27 %50 %37 |
| OpLoopMerge %36 %37 None |
| OpBranch %38 |
| %38 = OpLabel |
| %40 = OpSLessThan %bool %70 %int_10 |
| OpBranchConditional %40 %35 %36 |
| %35 = OpLabel |
| OpSelectionMerge %43 None |
| OpBranchConditional %25 %42 %47 |
| %42 = OpLabel |
| %46 = OpIAdd %int %69 %int_1 |
| OpBranch %43 |
| %47 = OpLabel |
| OpReturn |
| %43 = OpLabel |
| OpBranch %37 |
| %37 = OpLabel |
| %50 = OpIAdd %int %70 %int_1 |
| OpBranch %34 |
| %36 = OpLabel |
| OpBranch %29 |
| %29 = OpLabel |
| %52 = OpIAdd %int %69 %int_1 |
| OpBranch %26 |
| %28 = OpLabel |
| OpReturn |
| OpFunctionEnd |
| )"; |
| |
| SinglePassRunAndMatch<LoopUnswitchPass>(text, true); |
| } |
| |
| /* |
| Generated from the following GLSL + --eliminate-local-multi-store |
| |
| #version 330 core |
| in vec4 c; |
| void main() { |
| bool cond = false; |
| if (c[0] == 0) { |
| cond = c[1] == 0; |
| } else { |
| cond = c[2] == 0; |
| } |
| for (int i = 0; i < 10; i++) { |
| if (cond) { |
| i++; |
| } |
| } |
| } |
| */ |
| TEST_F(UnswitchTest, UnswitchNotUniform) { |
| // Check that the unswitch is not triggered (condition loop invariant but not |
| // uniform) |
| const std::string text = R"( |
| OpCapability Shader |
| %1 = OpExtInstImport "GLSL.std.450" |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint Fragment %main "main" %c |
| OpExecutionMode %main OriginUpperLeft |
| OpSource GLSL 330 |
| OpName %main "main" |
| OpName %c "c" |
| OpDecorate %c Location 0 |
| %void = OpTypeVoid |
| %3 = OpTypeFunction %void |
| %bool = OpTypeBool |
| %_ptr_Function_bool = OpTypePointer Function %bool |
| %float = OpTypeFloat 32 |
| %v4float = OpTypeVector %float 4 |
| %_ptr_Input_v4float = OpTypePointer Input %v4float |
| %c = OpVariable %_ptr_Input_v4float Input |
| %uint = OpTypeInt 32 0 |
| %uint_0 = OpConstant %uint 0 |
| %_ptr_Input_float = OpTypePointer Input %float |
| %float_0 = OpConstant %float 0 |
| %uint_1 = OpConstant %uint 1 |
| %uint_2 = OpConstant %uint 2 |
| %int = OpTypeInt 32 1 |
| %_ptr_Function_int = OpTypePointer Function %int |
| %int_0 = OpConstant %int 0 |
| %int_10 = OpConstant %int 10 |
| %int_1 = OpConstant %int 1 |
| %main = OpFunction %void None %3 |
| %5 = OpLabel |
| %17 = OpAccessChain %_ptr_Input_float %c %uint_0 |
| %18 = OpLoad %float %17 |
| %20 = OpFOrdEqual %bool %18 %float_0 |
| OpSelectionMerge %22 None |
| OpBranchConditional %20 %21 %27 |
| %21 = OpLabel |
| %24 = OpAccessChain %_ptr_Input_float %c %uint_1 |
| %25 = OpLoad %float %24 |
| %26 = OpFOrdEqual %bool %25 %float_0 |
| OpBranch %22 |
| %27 = OpLabel |
| %29 = OpAccessChain %_ptr_Input_float %c %uint_2 |
| %30 = OpLoad %float %29 |
| %31 = OpFOrdEqual %bool %30 %float_0 |
| OpBranch %22 |
| %22 = OpLabel |
| %52 = OpPhi %bool %26 %21 %31 %27 |
| OpBranch %36 |
| %36 = OpLabel |
| %53 = OpPhi %int %int_0 %22 %51 %39 |
| OpLoopMerge %38 %39 None |
| OpBranch %40 |
| %40 = OpLabel |
| %43 = OpSLessThan %bool %53 %int_10 |
| OpBranchConditional %43 %37 %38 |
| %37 = OpLabel |
| OpSelectionMerge %46 None |
| OpBranchConditional %52 %45 %46 |
| %45 = OpLabel |
| %49 = OpIAdd %int %53 %int_1 |
| OpBranch %46 |
| %46 = OpLabel |
| %54 = OpPhi %int %53 %37 %49 %45 |
| OpBranch %39 |
| %39 = OpLabel |
| %51 = OpIAdd %int %54 %int_1 |
| OpBranch %36 |
| %38 = OpLabel |
| OpReturn |
| OpFunctionEnd |
| )"; |
| |
| auto result = |
| SinglePassRunAndDisassemble<LoopUnswitchPass>(text, true, false); |
| |
| EXPECT_EQ(Pass::Status::SuccessWithoutChange, std::get<1>(result)); |
| } |
| |
| TEST_F(UnswitchTest, DontUnswitchLatch) { |
| // Check that the unswitch is not triggered for the latch branch. |
| const std::string text = R"( |
| OpCapability Shader |
| %1 = OpExtInstImport "GLSL.std.450" |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint Fragment %4 "main" |
| OpExecutionMode %4 OriginUpperLeft |
| OpSource ESSL 310 |
| %void = OpTypeVoid |
| %3 = OpTypeFunction %void |
| %bool = OpTypeBool |
| %false = OpConstantFalse %bool |
| %4 = OpFunction %void None %3 |
| %5 = OpLabel |
| OpBranch %6 |
| %6 = OpLabel |
| OpLoopMerge %8 %9 None |
| OpBranch %7 |
| %7 = OpLabel |
| OpBranch %9 |
| %9 = OpLabel |
| OpBranchConditional %false %6 %8 |
| %8 = OpLabel |
| OpReturn |
| OpFunctionEnd |
| )"; |
| |
| auto result = |
| SinglePassRunAndDisassemble<LoopUnswitchPass>(text, true, false); |
| EXPECT_EQ(Pass::Status::SuccessWithoutChange, std::get<1>(result)); |
| } |
| |
| TEST_F(UnswitchTest, DontUnswitchConstantCondition) { |
| const std::string text = R"( |
| OpCapability Shader |
| %1 = OpExtInstImport "GLSL.std.450" |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint Fragment %main "main" |
| OpExecutionMode %main OriginLowerLeft |
| OpSource GLSL 450 |
| OpName %main "main" |
| %void = OpTypeVoid |
| %4 = OpTypeFunction %void |
| %int = OpTypeInt 32 1 |
| %int_0 = OpConstant %int 0 |
| %bool = OpTypeBool |
| %true = OpConstantTrue %bool |
| %int_1 = OpConstant %int 1 |
| %main = OpFunction %void None %4 |
| %10 = OpLabel |
| OpBranch %11 |
| %11 = OpLabel |
| %12 = OpPhi %int %int_0 %10 %13 %14 |
| OpLoopMerge %15 %14 None |
| OpBranch %16 |
| %16 = OpLabel |
| %17 = OpSLessThan %bool %12 %int_1 |
| OpBranchConditional %17 %18 %15 |
| %18 = OpLabel |
| OpSelectionMerge %19 None |
| OpBranchConditional %true %20 %19 |
| %20 = OpLabel |
| %21 = OpIAdd %int %12 %int_1 |
| OpBranch %19 |
| %19 = OpLabel |
| %22 = OpPhi %int %21 %20 %12 %18 |
| OpBranch %14 |
| %14 = OpLabel |
| %13 = OpIAdd %int %22 %int_1 |
| OpBranch %11 |
| %15 = OpLabel |
| OpReturn |
| OpFunctionEnd |
| )"; |
| |
| auto result = |
| SinglePassRunAndDisassemble<LoopUnswitchPass>(text, true, false); |
| EXPECT_EQ(Pass::Status::SuccessWithoutChange, std::get<1>(result)); |
| } |
| |
| } // namespace |
| } // namespace opt |
| } // namespace spvtools |