mirror of
https://github.com/KhronosGroup/SPIRV-Tools
synced 2025-01-15 10:50:05 +00:00
434 lines
12 KiB
C++
434 lines
12 KiB
C++
|
// 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 <gmock/gmock.h>
|
||
|
|
||
|
#include <memory>
|
||
|
#include <string>
|
||
|
#include <vector>
|
||
|
|
||
|
#include "../assembly_builder.h"
|
||
|
#include "../function_utils.h"
|
||
|
#include "../pass_fixture.h"
|
||
|
#include "../pass_utils.h"
|
||
|
#include "opt/dominator_analysis.h"
|
||
|
#include "opt/pass.h"
|
||
|
|
||
|
namespace {
|
||
|
|
||
|
using namespace spvtools;
|
||
|
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<ir::IRContext> context =
|
||
|
BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
|
||
|
SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
|
||
|
ir::Module* module = context->module();
|
||
|
EXPECT_NE(nullptr, module) << "Assembling failed for shader:\n"
|
||
|
<< text << std::endl;
|
||
|
|
||
|
const ir::Function* f = spvtest::GetFunction(module, 4);
|
||
|
ir::CFG cfg(module);
|
||
|
opt::DominatorAnalysis* analysis = context->GetDominatorAnalysis(f, cfg);
|
||
|
|
||
|
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
|