// Copyright (c) 2017 Google 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. #include #include #include #include "gmock/gmock.h" #include "source/opt/dominator_analysis.h" #include "source/opt/pass.h" #include "test/opt/assembly_builder.h" #include "test/opt/function_utils.h" #include "test/opt/pass_fixture.h" #include "test/opt/pass_utils.h" namespace spvtools { namespace opt { namespace { using ::testing::UnorderedElementsAre; using PassClassTest = PassTest<::testing::Test>; /* Generated from the following GLSL #version 440 core layout(location = 0) out vec4 v; layout(location = 1) in vec4 in_val; void main() { for (int i = 0; i < in_val.x; ++i) { for (int j = 0; j < in_val.y; j++) { } } for (int i = 0; i < in_val.x; ++i) { for (int j = 0; j < in_val.y; j++) { } if (in_val.z == in_val.w) { break; } } int i = 0; while (i < in_val.x) { ++i; for (int j = 0; j < 1; j++) { for (int k = 0; k < 1; k++) { } } } i = 0; while (i < in_val.x) { ++i; if (in_val.z == in_val.w) { continue; } for (int j = 0; j < 1; j++) { for (int k = 0; k < 1; k++) { } if (in_val.z == in_val.w) { break; } } } v = vec4(1,1,1,1); } */ TEST_F(PassClassTest, BasicVisitFromEntryPoint) { const std::string text = R"( OpCapability Shader %1 = OpExtInstImport "GLSL.std.450" OpMemoryModel Logical GLSL450 OpEntryPoint Fragment %4 "main" %20 %163 OpExecutionMode %4 OriginUpperLeft OpSource GLSL 440 OpName %4 "main" OpName %8 "i" OpName %20 "in_val" OpName %28 "j" OpName %45 "i" OpName %56 "j" OpName %81 "i" OpName %94 "j" OpName %102 "k" OpName %134 "j" OpName %142 "k" OpName %163 "v" OpDecorate %20 Location 1 OpDecorate %163 Location 0 %2 = OpTypeVoid %3 = OpTypeFunction %2 %6 = OpTypeInt 32 1 %7 = OpTypePointer Function %6 %9 = OpConstant %6 0 %16 = OpTypeFloat 32 %18 = OpTypeVector %16 4 %19 = OpTypePointer Input %18 %20 = OpVariable %19 Input %21 = OpTypeInt 32 0 %22 = OpConstant %21 0 %23 = OpTypePointer Input %16 %26 = OpTypeBool %36 = OpConstant %21 1 %41 = OpConstant %6 1 %69 = OpConstant %21 2 %72 = OpConstant %21 3 %162 = OpTypePointer Output %18 %163 = OpVariable %162 Output %164 = OpConstant %16 1 %165 = OpConstantComposite %18 %164 %164 %164 %164 %4 = OpFunction %2 None %3 %5 = OpLabel %8 = OpVariable %7 Function %28 = OpVariable %7 Function %45 = OpVariable %7 Function %56 = OpVariable %7 Function %81 = OpVariable %7 Function %94 = OpVariable %7 Function %102 = OpVariable %7 Function %134 = OpVariable %7 Function %142 = OpVariable %7 Function OpStore %8 %9 OpBranch %10 %10 = OpLabel OpLoopMerge %12 %13 None OpBranch %14 %14 = OpLabel %15 = OpLoad %6 %8 %17 = OpConvertSToF %16 %15 %24 = OpAccessChain %23 %20 %22 %25 = OpLoad %16 %24 %27 = OpFOrdLessThan %26 %17 %25 OpBranchConditional %27 %11 %12 %11 = OpLabel OpStore %28 %9 OpBranch %29 %29 = OpLabel OpLoopMerge %31 %32 None OpBranch %33 %33 = OpLabel %34 = OpLoad %6 %28 %35 = OpConvertSToF %16 %34 %37 = OpAccessChain %23 %20 %36 %38 = OpLoad %16 %37 %39 = OpFOrdLessThan %26 %35 %38 OpBranchConditional %39 %30 %31 %30 = OpLabel OpBranch %32 %32 = OpLabel %40 = OpLoad %6 %28 %42 = OpIAdd %6 %40 %41 OpStore %28 %42 OpBranch %29 %31 = OpLabel OpBranch %13 %13 = OpLabel %43 = OpLoad %6 %8 %44 = OpIAdd %6 %43 %41 OpStore %8 %44 OpBranch %10 %12 = OpLabel OpStore %45 %9 OpBranch %46 %46 = OpLabel OpLoopMerge %48 %49 None OpBranch %50 %50 = OpLabel %51 = OpLoad %6 %45 %52 = OpConvertSToF %16 %51 %53 = OpAccessChain %23 %20 %22 %54 = OpLoad %16 %53 %55 = OpFOrdLessThan %26 %52 %54 OpBranchConditional %55 %47 %48 %47 = OpLabel OpStore %56 %9 OpBranch %57 %57 = OpLabel OpLoopMerge %59 %60 None OpBranch %61 %61 = OpLabel %62 = OpLoad %6 %56 %63 = OpConvertSToF %16 %62 %64 = OpAccessChain %23 %20 %36 %65 = OpLoad %16 %64 %66 = OpFOrdLessThan %26 %63 %65 OpBranchConditional %66 %58 %59 %58 = OpLabel OpBranch %60 %60 = OpLabel %67 = OpLoad %6 %56 %68 = OpIAdd %6 %67 %41 OpStore %56 %68 OpBranch %57 %59 = OpLabel %70 = OpAccessChain %23 %20 %69 %71 = OpLoad %16 %70 %73 = OpAccessChain %23 %20 %72 %74 = OpLoad %16 %73 %75 = OpFOrdEqual %26 %71 %74 OpSelectionMerge %77 None OpBranchConditional %75 %76 %77 %76 = OpLabel OpBranch %48 %77 = OpLabel OpBranch %49 %49 = OpLabel %79 = OpLoad %6 %45 %80 = OpIAdd %6 %79 %41 OpStore %45 %80 OpBranch %46 %48 = OpLabel OpStore %81 %9 OpBranch %82 %82 = OpLabel OpLoopMerge %84 %85 None OpBranch %86 %86 = OpLabel %87 = OpLoad %6 %81 %88 = OpConvertSToF %16 %87 %89 = OpAccessChain %23 %20 %22 %90 = OpLoad %16 %89 %91 = OpFOrdLessThan %26 %88 %90 OpBranchConditional %91 %83 %84 %83 = OpLabel %92 = OpLoad %6 %81 %93 = OpIAdd %6 %92 %41 OpStore %81 %93 OpStore %94 %9 OpBranch %95 %95 = OpLabel OpLoopMerge %97 %98 None OpBranch %99 %99 = OpLabel %100 = OpLoad %6 %94 %101 = OpSLessThan %26 %100 %41 OpBranchConditional %101 %96 %97 %96 = OpLabel OpStore %102 %9 OpBranch %103 %103 = OpLabel OpLoopMerge %105 %106 None OpBranch %107 %107 = OpLabel %108 = OpLoad %6 %102 %109 = OpSLessThan %26 %108 %41 OpBranchConditional %109 %104 %105 %104 = OpLabel OpBranch %106 %106 = OpLabel %110 = OpLoad %6 %102 %111 = OpIAdd %6 %110 %41 OpStore %102 %111 OpBranch %103 %105 = OpLabel OpBranch %98 %98 = OpLabel %112 = OpLoad %6 %94 %113 = OpIAdd %6 %112 %41 OpStore %94 %113 OpBranch %95 %97 = OpLabel OpBranch %85 %85 = OpLabel OpBranch %82 %84 = OpLabel OpStore %81 %9 OpBranch %114 %114 = OpLabel OpLoopMerge %116 %117 None OpBranch %118 %118 = OpLabel %119 = OpLoad %6 %81 %120 = OpConvertSToF %16 %119 %121 = OpAccessChain %23 %20 %22 %122 = OpLoad %16 %121 %123 = OpFOrdLessThan %26 %120 %122 OpBranchConditional %123 %115 %116 %115 = OpLabel %124 = OpLoad %6 %81 %125 = OpIAdd %6 %124 %41 OpStore %81 %125 %126 = OpAccessChain %23 %20 %69 %127 = OpLoad %16 %126 %128 = OpAccessChain %23 %20 %72 %129 = OpLoad %16 %128 %130 = OpFOrdEqual %26 %127 %129 OpSelectionMerge %132 None OpBranchConditional %130 %131 %132 %131 = OpLabel OpBranch %117 %132 = OpLabel OpStore %134 %9 OpBranch %135 %135 = OpLabel OpLoopMerge %137 %138 None OpBranch %139 %139 = OpLabel %140 = OpLoad %6 %134 %141 = OpSLessThan %26 %140 %41 OpBranchConditional %141 %136 %137 %136 = OpLabel OpStore %142 %9 OpBranch %143 %143 = OpLabel OpLoopMerge %145 %146 None OpBranch %147 %147 = OpLabel %148 = OpLoad %6 %142 %149 = OpSLessThan %26 %148 %41 OpBranchConditional %149 %144 %145 %144 = OpLabel OpBranch %146 %146 = OpLabel %150 = OpLoad %6 %142 %151 = OpIAdd %6 %150 %41 OpStore %142 %151 OpBranch %143 %145 = OpLabel %152 = OpAccessChain %23 %20 %69 %153 = OpLoad %16 %152 %154 = OpAccessChain %23 %20 %72 %155 = OpLoad %16 %154 %156 = OpFOrdEqual %26 %153 %155 OpSelectionMerge %158 None OpBranchConditional %156 %157 %158 %157 = OpLabel OpBranch %137 %158 = OpLabel OpBranch %138 %138 = OpLabel %160 = OpLoad %6 %134 %161 = OpIAdd %6 %160 %41 OpStore %134 %161 OpBranch %135 %137 = OpLabel OpBranch %117 %117 = OpLabel OpBranch %114 %116 = OpLabel OpStore %163 %165 OpReturn OpFunctionEnd )"; // clang-format on std::unique_ptr context = BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text, SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS); Module* module = context->module(); EXPECT_NE(nullptr, module) << "Assembling failed for shader:\n" << text << std::endl; const Function* f = spvtest::GetFunction(module, 4); DominatorAnalysis* analysis = context->GetDominatorAnalysis(f); EXPECT_TRUE(analysis->Dominates(5, 10)); EXPECT_TRUE(analysis->Dominates(5, 46)); EXPECT_TRUE(analysis->Dominates(5, 82)); EXPECT_TRUE(analysis->Dominates(5, 114)); EXPECT_TRUE(analysis->Dominates(5, 116)); EXPECT_TRUE(analysis->Dominates(10, 14)); EXPECT_TRUE(analysis->Dominates(10, 11)); EXPECT_TRUE(analysis->Dominates(10, 29)); EXPECT_TRUE(analysis->Dominates(10, 33)); EXPECT_TRUE(analysis->Dominates(10, 30)); EXPECT_TRUE(analysis->Dominates(10, 32)); EXPECT_TRUE(analysis->Dominates(10, 31)); EXPECT_TRUE(analysis->Dominates(10, 13)); EXPECT_TRUE(analysis->Dominates(10, 12)); EXPECT_TRUE(analysis->Dominates(12, 46)); EXPECT_TRUE(analysis->Dominates(46, 50)); EXPECT_TRUE(analysis->Dominates(46, 47)); EXPECT_TRUE(analysis->Dominates(46, 57)); EXPECT_TRUE(analysis->Dominates(46, 61)); EXPECT_TRUE(analysis->Dominates(46, 58)); EXPECT_TRUE(analysis->Dominates(46, 60)); EXPECT_TRUE(analysis->Dominates(46, 59)); EXPECT_TRUE(analysis->Dominates(46, 77)); EXPECT_TRUE(analysis->Dominates(46, 49)); EXPECT_TRUE(analysis->Dominates(46, 76)); EXPECT_TRUE(analysis->Dominates(46, 48)); EXPECT_TRUE(analysis->Dominates(48, 82)); EXPECT_TRUE(analysis->Dominates(82, 86)); EXPECT_TRUE(analysis->Dominates(82, 83)); EXPECT_TRUE(analysis->Dominates(82, 95)); EXPECT_TRUE(analysis->Dominates(82, 99)); EXPECT_TRUE(analysis->Dominates(82, 96)); EXPECT_TRUE(analysis->Dominates(82, 103)); EXPECT_TRUE(analysis->Dominates(82, 107)); EXPECT_TRUE(analysis->Dominates(82, 104)); EXPECT_TRUE(analysis->Dominates(82, 106)); EXPECT_TRUE(analysis->Dominates(82, 105)); EXPECT_TRUE(analysis->Dominates(82, 98)); EXPECT_TRUE(analysis->Dominates(82, 97)); EXPECT_TRUE(analysis->Dominates(82, 85)); EXPECT_TRUE(analysis->Dominates(82, 84)); EXPECT_TRUE(analysis->Dominates(84, 114)); EXPECT_TRUE(analysis->Dominates(114, 118)); EXPECT_TRUE(analysis->Dominates(114, 116)); EXPECT_TRUE(analysis->Dominates(114, 115)); EXPECT_TRUE(analysis->Dominates(114, 132)); EXPECT_TRUE(analysis->Dominates(114, 135)); EXPECT_TRUE(analysis->Dominates(114, 139)); EXPECT_TRUE(analysis->Dominates(114, 136)); EXPECT_TRUE(analysis->Dominates(114, 143)); EXPECT_TRUE(analysis->Dominates(114, 147)); EXPECT_TRUE(analysis->Dominates(114, 144)); EXPECT_TRUE(analysis->Dominates(114, 146)); EXPECT_TRUE(analysis->Dominates(114, 145)); EXPECT_TRUE(analysis->Dominates(114, 158)); EXPECT_TRUE(analysis->Dominates(114, 138)); EXPECT_TRUE(analysis->Dominates(114, 137)); EXPECT_TRUE(analysis->Dominates(114, 131)); EXPECT_TRUE(analysis->Dominates(114, 117)); } } // namespace } // namespace opt } // namespace spvtools