| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197 |
- // 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 "gmock/gmock.h"
- #include "source/opt/licm_pass.h"
- #include "test/opt/pass_fixture.h"
- namespace spvtools {
- namespace opt {
- namespace {
- using ::testing::UnorderedElementsAre;
- using PassClassTest = PassTest<::testing::Test>;
- /*
- Tests that the LICM pass will generate a preheader when one is not present
- Generated from the following GLSL fragment shader
- --eliminate-local-multi-store has also been run on the spv binary
- #version 440 core
- void main(){
- int a = 1;
- int b = 2;
- int hoist = 0;
- for (int i = 0; i < 10; i++) {
- if (i == 5) {
- break;
- }
- }
- for (int i = 0; i < 10; i++) {
- hoist = a + b;
- }
- }
- */
- TEST_F(PassClassTest, HoistWithoutPreheader) {
- const std::string text = R"(OpCapability Shader
- %1 = OpExtInstImport "GLSL.std.450"
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- OpExecutionMode %main OriginUpperLeft
- OpSource GLSL 440
- OpName %main "main"
- %void = OpTypeVoid
- %4 = OpTypeFunction %void
- %int = OpTypeInt 32 1
- %_ptr_Function_int = OpTypePointer Function %int
- %int_1 = OpConstant %int 1
- %int_2 = OpConstant %int 2
- %int_0 = OpConstant %int 0
- %int_10 = OpConstant %int 10
- %bool = OpTypeBool
- %int_5 = OpConstant %int 5
- %main = OpFunction %void None %4
- %13 = OpLabel
- OpBranch %14
- %14 = OpLabel
- %15 = OpPhi %int %int_0 %13 %16 %17
- ; CHECK: OpLoopMerge [[preheader:%\w+]]
- OpLoopMerge %25 %17 None
- OpBranch %19
- %19 = OpLabel
- %20 = OpSLessThan %bool %15 %int_10
- OpBranchConditional %20 %21 %25
- %21 = OpLabel
- %22 = OpIEqual %bool %15 %int_5
- OpSelectionMerge %23 None
- OpBranchConditional %22 %24 %23
- %24 = OpLabel
- OpBranch %25
- %23 = OpLabel
- OpBranch %17
- %17 = OpLabel
- %16 = OpIAdd %int %15 %int_1
- OpBranch %14
- ; Check that we hoisted the code to the preheader
- ; CHECK: [[preheader]] = OpLabel
- ; CHECK-NEXT: OpPhi
- ; CHECK-NEXT: OpPhi
- ; CHECK-NEXT: OpIAdd
- ; CHECK-NEXT: OpBranch [[header:%\w+]]
- ; CHECK: [[header]] = OpLabel
- ; CHECK-NEXT: OpPhi
- ; CHECK-NEXT: OpPhi
- ; CHECK: OpLoopMerge
- %25 = OpLabel
- %26 = OpPhi %int %int_0 %24 %int_0 %19 %27 %28
- %29 = OpPhi %int %int_0 %24 %int_0 %19 %30 %28
- OpLoopMerge %31 %28 None
- OpBranch %32
- %32 = OpLabel
- %33 = OpSLessThan %bool %29 %int_10
- OpBranchConditional %33 %34 %31
- %34 = OpLabel
- %27 = OpIAdd %int %int_1 %int_2
- OpBranch %28
- %28 = OpLabel
- %30 = OpIAdd %int %29 %int_1
- OpBranch %25
- %31 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- SinglePassRunAndMatch<LICMPass>(text, false);
- }
- TEST_F(PassClassTest, HoistWithoutPreheaderAtIdBound) {
- const std::string text = R"(OpCapability Shader
- %1 = OpExtInstImport "GLSL.std.450"
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- OpExecutionMode %main OriginUpperLeft
- OpSource GLSL 440
- OpName %main "main"
- %void = OpTypeVoid
- %4 = OpTypeFunction %void
- %int = OpTypeInt 32 1
- %_ptr_Function_int = OpTypePointer Function %int
- %int_1 = OpConstant %int 1
- %int_2 = OpConstant %int 2
- %int_0 = OpConstant %int 0
- %int_10 = OpConstant %int 10
- %bool = OpTypeBool
- %int_5 = OpConstant %int 5
- %main = OpFunction %void None %4
- %13 = OpLabel
- OpBranch %14
- %14 = OpLabel
- %15 = OpPhi %int %int_0 %13 %16 %17
- OpLoopMerge %25 %17 None
- OpBranch %19
- %19 = OpLabel
- %20 = OpSLessThan %bool %15 %int_10
- OpBranchConditional %20 %21 %25
- %21 = OpLabel
- %22 = OpIEqual %bool %15 %int_5
- OpSelectionMerge %23 None
- OpBranchConditional %22 %24 %23
- %24 = OpLabel
- OpBranch %25
- %23 = OpLabel
- OpBranch %17
- %17 = OpLabel
- %16 = OpIAdd %int %15 %int_1
- OpBranch %14
- %25 = OpLabel
- %26 = OpPhi %int %int_0 %24 %int_0 %19 %27 %28
- %29 = OpPhi %int %int_0 %24 %int_0 %19 %30 %28
- OpLoopMerge %31 %28 None
- OpBranch %32
- %32 = OpLabel
- %33 = OpSLessThan %bool %29 %int_10
- OpBranchConditional %33 %34 %31
- %34 = OpLabel
- %27 = OpIAdd %int %int_1 %int_2
- OpBranch %28
- %28 = OpLabel
- %30 = OpIAdd %int %29 %int_1
- OpBranch %25
- %31 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- uint32_t current_bound = context->module()->id_bound();
- context->set_max_id_bound(current_bound);
- auto pass = MakeUnique<LICMPass>();
- auto result = pass->Run(context.get());
- EXPECT_EQ(result, Pass::Status::Failure);
- std::vector<uint32_t> binary;
- context->module()->ToBinary(&binary, false);
- std::string optimized_asm;
- SpirvTools tools_(SPV_ENV_UNIVERSAL_1_1);
- tools_.Disassemble(binary, &optimized_asm);
- std::cout << optimized_asm << std::endl;
- }
- } // namespace
- } // namespace opt
- } // namespace spvtools
|