SPIRV-Tools/test/opt/dominator_tree/nested_loops.cpp
Stephen McGroarty 8ba68fa9b9 Dominator Tree Analysis (#3)
Support for dominator and post dominator analysis on ir::Functions. This patch contains a DominatorTree class for building the tree and DominatorAnalysis and DominatorAnalysisPass classes for interfacing and caching the built trees.
2017-12-05 22:59:43 -05:00

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