| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201 |
- // 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 analyse multiple independent loops in a function
- 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++) {
- // invariant
- hoist = a + b;
- }
- for (int i = 0; i < 10; i++) {
- // invariant
- hoist = a + b;
- }
- int c = 1;
- int d = 2;
- int hoist2 = 0;
- for (int i = 0; i < 10; i++) {
- // invariant
- hoist2 = c + d;
- }
- }
- */
- TEST_F(PassClassTest, HoistFromIndependentLoops) {
- const std::string before_hoist = 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
- %main = OpFunction %void None %4
- %12 = OpLabel
- OpBranch %13
- %13 = OpLabel
- %14 = OpPhi %int %int_0 %12 %15 %16
- %17 = OpPhi %int %int_0 %12 %18 %16
- OpLoopMerge %19 %16 None
- OpBranch %20
- %20 = OpLabel
- %21 = OpSLessThan %bool %17 %int_10
- OpBranchConditional %21 %22 %19
- %22 = OpLabel
- %15 = OpIAdd %int %int_1 %int_2
- OpBranch %16
- %16 = OpLabel
- %18 = OpIAdd %int %17 %int_1
- OpBranch %13
- %19 = OpLabel
- OpBranch %23
- %23 = OpLabel
- %24 = OpPhi %int %14 %19 %25 %26
- %27 = OpPhi %int %int_0 %19 %28 %26
- OpLoopMerge %29 %26 None
- OpBranch %30
- %30 = OpLabel
- %31 = OpSLessThan %bool %27 %int_10
- OpBranchConditional %31 %32 %29
- %32 = OpLabel
- %25 = OpIAdd %int %int_1 %int_2
- OpBranch %26
- %26 = OpLabel
- %28 = OpIAdd %int %27 %int_1
- OpBranch %23
- %29 = OpLabel
- OpBranch %33
- %33 = OpLabel
- %34 = OpPhi %int %int_0 %29 %35 %36
- %37 = OpPhi %int %int_0 %29 %38 %36
- OpLoopMerge %39 %36 None
- OpBranch %40
- %40 = OpLabel
- %41 = OpSLessThan %bool %37 %int_10
- OpBranchConditional %41 %42 %39
- %42 = OpLabel
- %35 = OpIAdd %int %int_1 %int_2
- OpBranch %36
- %36 = OpLabel
- %38 = OpIAdd %int %37 %int_1
- OpBranch %33
- %39 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- const std::string after_hoist = 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
- %main = OpFunction %void None %4
- %12 = OpLabel
- %15 = OpIAdd %int %int_1 %int_2
- OpBranch %13
- %13 = OpLabel
- %14 = OpPhi %int %int_0 %12 %15 %16
- %17 = OpPhi %int %int_0 %12 %18 %16
- OpLoopMerge %19 %16 None
- OpBranch %20
- %20 = OpLabel
- %21 = OpSLessThan %bool %17 %int_10
- OpBranchConditional %21 %22 %19
- %22 = OpLabel
- OpBranch %16
- %16 = OpLabel
- %18 = OpIAdd %int %17 %int_1
- OpBranch %13
- %19 = OpLabel
- %25 = OpIAdd %int %int_1 %int_2
- OpBranch %23
- %23 = OpLabel
- %24 = OpPhi %int %14 %19 %25 %26
- %27 = OpPhi %int %int_0 %19 %28 %26
- OpLoopMerge %29 %26 None
- OpBranch %30
- %30 = OpLabel
- %31 = OpSLessThan %bool %27 %int_10
- OpBranchConditional %31 %32 %29
- %32 = OpLabel
- OpBranch %26
- %26 = OpLabel
- %28 = OpIAdd %int %27 %int_1
- OpBranch %23
- %29 = OpLabel
- %35 = OpIAdd %int %int_1 %int_2
- OpBranch %33
- %33 = OpLabel
- %34 = OpPhi %int %int_0 %29 %35 %36
- %37 = OpPhi %int %int_0 %29 %38 %36
- OpLoopMerge %39 %36 None
- OpBranch %40
- %40 = OpLabel
- %41 = OpSLessThan %bool %37 %int_10
- OpBranchConditional %41 %42 %39
- %42 = OpLabel
- OpBranch %36
- %36 = OpLabel
- %38 = OpIAdd %int %37 %int_1
- OpBranch %33
- %39 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- SinglePassRunAndCheck<LICMPass>(before_hoist, after_hoist, true);
- }
- } // namespace
- } // namespace opt
- } // namespace spvtools
|