mirror of
https://github.com/KhronosGroup/SPIRV-Tools
synced 2025-01-05 14:31:05 +00:00
3017 lines
103 KiB
C++
3017 lines
103 KiB
C++
// 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 <gmock/gmock.h>
|
|
|
|
#include <memory>
|
|
#include <string>
|
|
#include <unordered_set>
|
|
#include <vector>
|
|
|
|
#include "../assembly_builder.h"
|
|
#include "../function_utils.h"
|
|
#include "../pass_fixture.h"
|
|
#include "../pass_utils.h"
|
|
|
|
#include "opt/iterator.h"
|
|
#include "opt/loop_dependence.h"
|
|
#include "opt/loop_descriptor.h"
|
|
#include "opt/pass.h"
|
|
#include "opt/scalar_analysis.h"
|
|
#include "opt/tree_iterator.h"
|
|
|
|
namespace {
|
|
|
|
using namespace spvtools;
|
|
using DependencyAnalysisHelpers = ::testing::Test;
|
|
|
|
/*
|
|
Generated from the following GLSL fragment shader
|
|
with --eliminate-local-multi-store
|
|
#version 440 core
|
|
void a() {
|
|
int[10][10] arr;
|
|
int i = 0;
|
|
int j = 0;
|
|
for (; i < 10 && j < 10; i++, j++) {
|
|
arr[i][j] = arr[i][j];
|
|
}
|
|
}
|
|
void b() {
|
|
int[10] arr;
|
|
for (int i = 0; i < 10; i+=2) {
|
|
arr[i] = arr[i];
|
|
}
|
|
}
|
|
void main(){
|
|
a();
|
|
b();
|
|
}
|
|
*/
|
|
TEST(DependencyAnalysisHelpers, UnsupportedLoops) {
|
|
const std::string text = R"( OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource GLSL 440
|
|
OpName %4 "main"
|
|
OpName %6 "a("
|
|
OpName %8 "b("
|
|
OpName %12 "i"
|
|
OpName %14 "j"
|
|
OpName %32 "arr"
|
|
OpName %45 "i"
|
|
OpName %54 "arr"
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%10 = OpTypeInt 32 1
|
|
%11 = OpTypePointer Function %10
|
|
%13 = OpConstant %10 0
|
|
%21 = OpConstant %10 10
|
|
%22 = OpTypeBool
|
|
%27 = OpTypeInt 32 0
|
|
%28 = OpConstant %27 10
|
|
%29 = OpTypeArray %10 %28
|
|
%30 = OpTypeArray %29 %28
|
|
%31 = OpTypePointer Function %30
|
|
%41 = OpConstant %10 1
|
|
%53 = OpTypePointer Function %29
|
|
%60 = OpConstant %10 2
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%63 = OpFunctionCall %2 %6
|
|
%64 = OpFunctionCall %2 %8
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%6 = OpFunction %2 None %3
|
|
%7 = OpLabel
|
|
%12 = OpVariable %11 Function
|
|
%14 = OpVariable %11 Function
|
|
%32 = OpVariable %31 Function
|
|
OpStore %12 %13
|
|
OpStore %14 %13
|
|
OpBranch %15
|
|
%15 = OpLabel
|
|
%65 = OpPhi %10 %13 %7 %42 %18
|
|
%66 = OpPhi %10 %13 %7 %44 %18
|
|
OpLoopMerge %17 %18 None
|
|
OpBranch %19
|
|
%19 = OpLabel
|
|
%23 = OpSLessThan %22 %65 %21
|
|
%25 = OpSLessThan %22 %66 %21
|
|
%26 = OpLogicalAnd %22 %23 %25
|
|
OpBranchConditional %26 %16 %17
|
|
%16 = OpLabel
|
|
%37 = OpAccessChain %11 %32 %65 %66
|
|
%38 = OpLoad %10 %37
|
|
%39 = OpAccessChain %11 %32 %65 %66
|
|
OpStore %39 %38
|
|
OpBranch %18
|
|
%18 = OpLabel
|
|
%42 = OpIAdd %10 %65 %41
|
|
OpStore %12 %42
|
|
%44 = OpIAdd %10 %66 %41
|
|
OpStore %14 %44
|
|
OpBranch %15
|
|
%17 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%8 = OpFunction %2 None %3
|
|
%9 = OpLabel
|
|
%45 = OpVariable %11 Function
|
|
%54 = OpVariable %53 Function
|
|
OpStore %45 %13
|
|
OpBranch %46
|
|
%46 = OpLabel
|
|
%67 = OpPhi %10 %13 %9 %62 %49
|
|
OpLoopMerge %48 %49 None
|
|
OpBranch %50
|
|
%50 = OpLabel
|
|
%52 = OpSLessThan %22 %67 %21
|
|
OpBranchConditional %52 %47 %48
|
|
%47 = OpLabel
|
|
%57 = OpAccessChain %11 %54 %67
|
|
%58 = OpLoad %10 %57
|
|
%59 = OpAccessChain %11 %54 %67
|
|
OpStore %59 %58
|
|
OpBranch %49
|
|
%49 = OpLabel
|
|
%62 = OpIAdd %10 %67 %60
|
|
OpStore %45 %62
|
|
OpBranch %46
|
|
%48 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
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;
|
|
{
|
|
// Function a
|
|
const ir::Function* f = spvtest::GetFunction(module, 6);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
const ir::Instruction* store[1] = {nullptr};
|
|
int stores_found = 0;
|
|
for (const ir::Instruction& inst : *spvtest::GetBasicBlock(f, 16)) {
|
|
if (inst.opcode() == SpvOp::SpvOpStore) {
|
|
store[stores_found] = &inst;
|
|
++stores_found;
|
|
}
|
|
}
|
|
// 38 -> 39
|
|
opt::DistanceVector distance_vector{loops.size()};
|
|
EXPECT_FALSE(analysis.IsSupportedLoop(loops[0]));
|
|
EXPECT_FALSE(analysis.GetDependence(context->get_def_use_mgr()->GetDef(38),
|
|
store[0], &distance_vector));
|
|
EXPECT_EQ(distance_vector.GetEntries()[0].dependence_information,
|
|
opt::DistanceEntry::DependenceInformation::UNKNOWN);
|
|
EXPECT_EQ(distance_vector.GetEntries()[0].direction,
|
|
opt::DistanceEntry::Directions::ALL);
|
|
}
|
|
{
|
|
// Function b
|
|
const ir::Function* f = spvtest::GetFunction(module, 8);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
const ir::Instruction* store[1] = {nullptr};
|
|
int stores_found = 0;
|
|
for (const ir::Instruction& inst : *spvtest::GetBasicBlock(f, 47)) {
|
|
if (inst.opcode() == SpvOp::SpvOpStore) {
|
|
store[stores_found] = &inst;
|
|
++stores_found;
|
|
}
|
|
}
|
|
// 58 -> 59
|
|
opt::DistanceVector distance_vector{loops.size()};
|
|
EXPECT_FALSE(analysis.IsSupportedLoop(loops[0]));
|
|
EXPECT_FALSE(analysis.GetDependence(context->get_def_use_mgr()->GetDef(58),
|
|
store[0], &distance_vector));
|
|
EXPECT_EQ(distance_vector.GetEntries()[0].dependence_information,
|
|
opt::DistanceEntry::DependenceInformation::UNKNOWN);
|
|
EXPECT_EQ(distance_vector.GetEntries()[0].direction,
|
|
opt::DistanceEntry::Directions::ALL);
|
|
}
|
|
}
|
|
|
|
/*
|
|
Generated from the following GLSL fragment shader
|
|
with --eliminate-local-multi-store
|
|
#version 440 core
|
|
void a() {
|
|
for (int i = -10; i < 0; i++) {
|
|
|
|
}
|
|
}
|
|
void b() {
|
|
for (int i = -5; i < 5; i++) {
|
|
|
|
}
|
|
}
|
|
void c() {
|
|
for (int i = 0; i < 10; i++) {
|
|
|
|
}
|
|
}
|
|
void d() {
|
|
for (int i = 5; i < 15; i++) {
|
|
|
|
}
|
|
}
|
|
void e() {
|
|
for (int i = -10; i <= 0; i++) {
|
|
|
|
}
|
|
}
|
|
void f() {
|
|
for (int i = -5; i <= 5; i++) {
|
|
|
|
}
|
|
}
|
|
void g() {
|
|
for (int i = 0; i <= 10; i++) {
|
|
|
|
}
|
|
}
|
|
void h() {
|
|
for (int i = 5; i <= 15; i++) {
|
|
|
|
}
|
|
}
|
|
void i() {
|
|
for (int i = 0; i > -10; i--) {
|
|
|
|
}
|
|
}
|
|
void j() {
|
|
for (int i = 5; i > -5; i--) {
|
|
|
|
}
|
|
}
|
|
void k() {
|
|
for (int i = 10; i > 0; i--) {
|
|
|
|
}
|
|
}
|
|
void l() {
|
|
for (int i = 15; i > 5; i--) {
|
|
|
|
}
|
|
}
|
|
void m() {
|
|
for (int i = 0; i >= -10; i--) {
|
|
|
|
}
|
|
}
|
|
void n() {
|
|
for (int i = 5; i >= -5; i--) {
|
|
|
|
}
|
|
}
|
|
void o() {
|
|
for (int i = 10; i >= 0; i--) {
|
|
|
|
}
|
|
}
|
|
void p() {
|
|
for (int i = 15; i >= 5; i--) {
|
|
|
|
}
|
|
}
|
|
void main(){
|
|
a();
|
|
b();
|
|
c();
|
|
d();
|
|
e();
|
|
f();
|
|
g();
|
|
h();
|
|
i();
|
|
j();
|
|
k();
|
|
l();
|
|
m();
|
|
n();
|
|
o();
|
|
p();
|
|
}
|
|
*/
|
|
TEST(DependencyAnalysisHelpers, loop_information) {
|
|
const std::string text = R"( OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource GLSL 440
|
|
OpName %4 "main"
|
|
OpName %6 "a("
|
|
OpName %8 "b("
|
|
OpName %10 "c("
|
|
OpName %12 "d("
|
|
OpName %14 "e("
|
|
OpName %16 "f("
|
|
OpName %18 "g("
|
|
OpName %20 "h("
|
|
OpName %22 "i("
|
|
OpName %24 "j("
|
|
OpName %26 "k("
|
|
OpName %28 "l("
|
|
OpName %30 "m("
|
|
OpName %32 "n("
|
|
OpName %34 "o("
|
|
OpName %36 "p("
|
|
OpName %40 "i"
|
|
OpName %54 "i"
|
|
OpName %66 "i"
|
|
OpName %77 "i"
|
|
OpName %88 "i"
|
|
OpName %98 "i"
|
|
OpName %108 "i"
|
|
OpName %118 "i"
|
|
OpName %128 "i"
|
|
OpName %138 "i"
|
|
OpName %148 "i"
|
|
OpName %158 "i"
|
|
OpName %168 "i"
|
|
OpName %178 "i"
|
|
OpName %188 "i"
|
|
OpName %198 "i"
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%38 = OpTypeInt 32 1
|
|
%39 = OpTypePointer Function %38
|
|
%41 = OpConstant %38 -10
|
|
%48 = OpConstant %38 0
|
|
%49 = OpTypeBool
|
|
%52 = OpConstant %38 1
|
|
%55 = OpConstant %38 -5
|
|
%62 = OpConstant %38 5
|
|
%73 = OpConstant %38 10
|
|
%84 = OpConstant %38 15
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%208 = OpFunctionCall %2 %6
|
|
%209 = OpFunctionCall %2 %8
|
|
%210 = OpFunctionCall %2 %10
|
|
%211 = OpFunctionCall %2 %12
|
|
%212 = OpFunctionCall %2 %14
|
|
%213 = OpFunctionCall %2 %16
|
|
%214 = OpFunctionCall %2 %18
|
|
%215 = OpFunctionCall %2 %20
|
|
%216 = OpFunctionCall %2 %22
|
|
%217 = OpFunctionCall %2 %24
|
|
%218 = OpFunctionCall %2 %26
|
|
%219 = OpFunctionCall %2 %28
|
|
%220 = OpFunctionCall %2 %30
|
|
%221 = OpFunctionCall %2 %32
|
|
%222 = OpFunctionCall %2 %34
|
|
%223 = OpFunctionCall %2 %36
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%6 = OpFunction %2 None %3
|
|
%7 = OpLabel
|
|
%40 = OpVariable %39 Function
|
|
OpStore %40 %41
|
|
OpBranch %42
|
|
%42 = OpLabel
|
|
%224 = OpPhi %38 %41 %7 %53 %45
|
|
OpLoopMerge %44 %45 None
|
|
OpBranch %46
|
|
%46 = OpLabel
|
|
%50 = OpSLessThan %49 %224 %48
|
|
OpBranchConditional %50 %43 %44
|
|
%43 = OpLabel
|
|
OpBranch %45
|
|
%45 = OpLabel
|
|
%53 = OpIAdd %38 %224 %52
|
|
OpStore %40 %53
|
|
OpBranch %42
|
|
%44 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%8 = OpFunction %2 None %3
|
|
%9 = OpLabel
|
|
%54 = OpVariable %39 Function
|
|
OpStore %54 %55
|
|
OpBranch %56
|
|
%56 = OpLabel
|
|
%225 = OpPhi %38 %55 %9 %65 %59
|
|
OpLoopMerge %58 %59 None
|
|
OpBranch %60
|
|
%60 = OpLabel
|
|
%63 = OpSLessThan %49 %225 %62
|
|
OpBranchConditional %63 %57 %58
|
|
%57 = OpLabel
|
|
OpBranch %59
|
|
%59 = OpLabel
|
|
%65 = OpIAdd %38 %225 %52
|
|
OpStore %54 %65
|
|
OpBranch %56
|
|
%58 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%10 = OpFunction %2 None %3
|
|
%11 = OpLabel
|
|
%66 = OpVariable %39 Function
|
|
OpStore %66 %48
|
|
OpBranch %67
|
|
%67 = OpLabel
|
|
%226 = OpPhi %38 %48 %11 %76 %70
|
|
OpLoopMerge %69 %70 None
|
|
OpBranch %71
|
|
%71 = OpLabel
|
|
%74 = OpSLessThan %49 %226 %73
|
|
OpBranchConditional %74 %68 %69
|
|
%68 = OpLabel
|
|
OpBranch %70
|
|
%70 = OpLabel
|
|
%76 = OpIAdd %38 %226 %52
|
|
OpStore %66 %76
|
|
OpBranch %67
|
|
%69 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%12 = OpFunction %2 None %3
|
|
%13 = OpLabel
|
|
%77 = OpVariable %39 Function
|
|
OpStore %77 %62
|
|
OpBranch %78
|
|
%78 = OpLabel
|
|
%227 = OpPhi %38 %62 %13 %87 %81
|
|
OpLoopMerge %80 %81 None
|
|
OpBranch %82
|
|
%82 = OpLabel
|
|
%85 = OpSLessThan %49 %227 %84
|
|
OpBranchConditional %85 %79 %80
|
|
%79 = OpLabel
|
|
OpBranch %81
|
|
%81 = OpLabel
|
|
%87 = OpIAdd %38 %227 %52
|
|
OpStore %77 %87
|
|
OpBranch %78
|
|
%80 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%14 = OpFunction %2 None %3
|
|
%15 = OpLabel
|
|
%88 = OpVariable %39 Function
|
|
OpStore %88 %41
|
|
OpBranch %89
|
|
%89 = OpLabel
|
|
%228 = OpPhi %38 %41 %15 %97 %92
|
|
OpLoopMerge %91 %92 None
|
|
OpBranch %93
|
|
%93 = OpLabel
|
|
%95 = OpSLessThanEqual %49 %228 %48
|
|
OpBranchConditional %95 %90 %91
|
|
%90 = OpLabel
|
|
OpBranch %92
|
|
%92 = OpLabel
|
|
%97 = OpIAdd %38 %228 %52
|
|
OpStore %88 %97
|
|
OpBranch %89
|
|
%91 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%16 = OpFunction %2 None %3
|
|
%17 = OpLabel
|
|
%98 = OpVariable %39 Function
|
|
OpStore %98 %55
|
|
OpBranch %99
|
|
%99 = OpLabel
|
|
%229 = OpPhi %38 %55 %17 %107 %102
|
|
OpLoopMerge %101 %102 None
|
|
OpBranch %103
|
|
%103 = OpLabel
|
|
%105 = OpSLessThanEqual %49 %229 %62
|
|
OpBranchConditional %105 %100 %101
|
|
%100 = OpLabel
|
|
OpBranch %102
|
|
%102 = OpLabel
|
|
%107 = OpIAdd %38 %229 %52
|
|
OpStore %98 %107
|
|
OpBranch %99
|
|
%101 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%18 = OpFunction %2 None %3
|
|
%19 = OpLabel
|
|
%108 = OpVariable %39 Function
|
|
OpStore %108 %48
|
|
OpBranch %109
|
|
%109 = OpLabel
|
|
%230 = OpPhi %38 %48 %19 %117 %112
|
|
OpLoopMerge %111 %112 None
|
|
OpBranch %113
|
|
%113 = OpLabel
|
|
%115 = OpSLessThanEqual %49 %230 %73
|
|
OpBranchConditional %115 %110 %111
|
|
%110 = OpLabel
|
|
OpBranch %112
|
|
%112 = OpLabel
|
|
%117 = OpIAdd %38 %230 %52
|
|
OpStore %108 %117
|
|
OpBranch %109
|
|
%111 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%20 = OpFunction %2 None %3
|
|
%21 = OpLabel
|
|
%118 = OpVariable %39 Function
|
|
OpStore %118 %62
|
|
OpBranch %119
|
|
%119 = OpLabel
|
|
%231 = OpPhi %38 %62 %21 %127 %122
|
|
OpLoopMerge %121 %122 None
|
|
OpBranch %123
|
|
%123 = OpLabel
|
|
%125 = OpSLessThanEqual %49 %231 %84
|
|
OpBranchConditional %125 %120 %121
|
|
%120 = OpLabel
|
|
OpBranch %122
|
|
%122 = OpLabel
|
|
%127 = OpIAdd %38 %231 %52
|
|
OpStore %118 %127
|
|
OpBranch %119
|
|
%121 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%22 = OpFunction %2 None %3
|
|
%23 = OpLabel
|
|
%128 = OpVariable %39 Function
|
|
OpStore %128 %48
|
|
OpBranch %129
|
|
%129 = OpLabel
|
|
%232 = OpPhi %38 %48 %23 %137 %132
|
|
OpLoopMerge %131 %132 None
|
|
OpBranch %133
|
|
%133 = OpLabel
|
|
%135 = OpSGreaterThan %49 %232 %41
|
|
OpBranchConditional %135 %130 %131
|
|
%130 = OpLabel
|
|
OpBranch %132
|
|
%132 = OpLabel
|
|
%137 = OpISub %38 %232 %52
|
|
OpStore %128 %137
|
|
OpBranch %129
|
|
%131 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%24 = OpFunction %2 None %3
|
|
%25 = OpLabel
|
|
%138 = OpVariable %39 Function
|
|
OpStore %138 %62
|
|
OpBranch %139
|
|
%139 = OpLabel
|
|
%233 = OpPhi %38 %62 %25 %147 %142
|
|
OpLoopMerge %141 %142 None
|
|
OpBranch %143
|
|
%143 = OpLabel
|
|
%145 = OpSGreaterThan %49 %233 %55
|
|
OpBranchConditional %145 %140 %141
|
|
%140 = OpLabel
|
|
OpBranch %142
|
|
%142 = OpLabel
|
|
%147 = OpISub %38 %233 %52
|
|
OpStore %138 %147
|
|
OpBranch %139
|
|
%141 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%26 = OpFunction %2 None %3
|
|
%27 = OpLabel
|
|
%148 = OpVariable %39 Function
|
|
OpStore %148 %73
|
|
OpBranch %149
|
|
%149 = OpLabel
|
|
%234 = OpPhi %38 %73 %27 %157 %152
|
|
OpLoopMerge %151 %152 None
|
|
OpBranch %153
|
|
%153 = OpLabel
|
|
%155 = OpSGreaterThan %49 %234 %48
|
|
OpBranchConditional %155 %150 %151
|
|
%150 = OpLabel
|
|
OpBranch %152
|
|
%152 = OpLabel
|
|
%157 = OpISub %38 %234 %52
|
|
OpStore %148 %157
|
|
OpBranch %149
|
|
%151 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%28 = OpFunction %2 None %3
|
|
%29 = OpLabel
|
|
%158 = OpVariable %39 Function
|
|
OpStore %158 %84
|
|
OpBranch %159
|
|
%159 = OpLabel
|
|
%235 = OpPhi %38 %84 %29 %167 %162
|
|
OpLoopMerge %161 %162 None
|
|
OpBranch %163
|
|
%163 = OpLabel
|
|
%165 = OpSGreaterThan %49 %235 %62
|
|
OpBranchConditional %165 %160 %161
|
|
%160 = OpLabel
|
|
OpBranch %162
|
|
%162 = OpLabel
|
|
%167 = OpISub %38 %235 %52
|
|
OpStore %158 %167
|
|
OpBranch %159
|
|
%161 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%30 = OpFunction %2 None %3
|
|
%31 = OpLabel
|
|
%168 = OpVariable %39 Function
|
|
OpStore %168 %48
|
|
OpBranch %169
|
|
%169 = OpLabel
|
|
%236 = OpPhi %38 %48 %31 %177 %172
|
|
OpLoopMerge %171 %172 None
|
|
OpBranch %173
|
|
%173 = OpLabel
|
|
%175 = OpSGreaterThanEqual %49 %236 %41
|
|
OpBranchConditional %175 %170 %171
|
|
%170 = OpLabel
|
|
OpBranch %172
|
|
%172 = OpLabel
|
|
%177 = OpISub %38 %236 %52
|
|
OpStore %168 %177
|
|
OpBranch %169
|
|
%171 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%32 = OpFunction %2 None %3
|
|
%33 = OpLabel
|
|
%178 = OpVariable %39 Function
|
|
OpStore %178 %62
|
|
OpBranch %179
|
|
%179 = OpLabel
|
|
%237 = OpPhi %38 %62 %33 %187 %182
|
|
OpLoopMerge %181 %182 None
|
|
OpBranch %183
|
|
%183 = OpLabel
|
|
%185 = OpSGreaterThanEqual %49 %237 %55
|
|
OpBranchConditional %185 %180 %181
|
|
%180 = OpLabel
|
|
OpBranch %182
|
|
%182 = OpLabel
|
|
%187 = OpISub %38 %237 %52
|
|
OpStore %178 %187
|
|
OpBranch %179
|
|
%181 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%34 = OpFunction %2 None %3
|
|
%35 = OpLabel
|
|
%188 = OpVariable %39 Function
|
|
OpStore %188 %73
|
|
OpBranch %189
|
|
%189 = OpLabel
|
|
%238 = OpPhi %38 %73 %35 %197 %192
|
|
OpLoopMerge %191 %192 None
|
|
OpBranch %193
|
|
%193 = OpLabel
|
|
%195 = OpSGreaterThanEqual %49 %238 %48
|
|
OpBranchConditional %195 %190 %191
|
|
%190 = OpLabel
|
|
OpBranch %192
|
|
%192 = OpLabel
|
|
%197 = OpISub %38 %238 %52
|
|
OpStore %188 %197
|
|
OpBranch %189
|
|
%191 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%36 = OpFunction %2 None %3
|
|
%37 = OpLabel
|
|
%198 = OpVariable %39 Function
|
|
OpStore %198 %84
|
|
OpBranch %199
|
|
%199 = OpLabel
|
|
%239 = OpPhi %38 %84 %37 %207 %202
|
|
OpLoopMerge %201 %202 None
|
|
OpBranch %203
|
|
%203 = OpLabel
|
|
%205 = OpSGreaterThanEqual %49 %239 %62
|
|
OpBranchConditional %205 %200 %201
|
|
%200 = OpLabel
|
|
OpBranch %202
|
|
%202 = OpLabel
|
|
%207 = OpISub %38 %239 %52
|
|
OpStore %198 %207
|
|
OpBranch %199
|
|
%201 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
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;
|
|
{
|
|
// Function a
|
|
const ir::Function* f = spvtest::GetFunction(module, 6);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetLowerBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
-10);
|
|
EXPECT_EQ(
|
|
analysis.GetUpperBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
-1);
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetTripCount(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
10);
|
|
|
|
EXPECT_EQ(analysis.GetFirstTripInductionNode(loop),
|
|
analysis.GetScalarEvolution()->CreateConstant(-10));
|
|
|
|
EXPECT_EQ(analysis.GetFinalTripInductionNode(
|
|
loop, analysis.GetScalarEvolution()->CreateConstant(1)),
|
|
analysis.GetScalarEvolution()->CreateConstant(-1));
|
|
}
|
|
{
|
|
// Function b
|
|
const ir::Function* f = spvtest::GetFunction(module, 8);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetLowerBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
-5);
|
|
EXPECT_EQ(
|
|
analysis.GetUpperBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
4);
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetTripCount(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
10);
|
|
|
|
EXPECT_EQ(analysis.GetFirstTripInductionNode(loop),
|
|
analysis.GetScalarEvolution()->CreateConstant(-5));
|
|
|
|
EXPECT_EQ(analysis.GetFinalTripInductionNode(
|
|
loop, analysis.GetScalarEvolution()->CreateConstant(1)),
|
|
analysis.GetScalarEvolution()->CreateConstant(4));
|
|
}
|
|
{
|
|
// Function c
|
|
const ir::Function* f = spvtest::GetFunction(module, 10);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetLowerBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
0);
|
|
EXPECT_EQ(
|
|
analysis.GetUpperBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
9);
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetTripCount(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
10);
|
|
|
|
EXPECT_EQ(analysis.GetFirstTripInductionNode(loop),
|
|
analysis.GetScalarEvolution()->CreateConstant(0));
|
|
|
|
EXPECT_EQ(analysis.GetFinalTripInductionNode(
|
|
loop, analysis.GetScalarEvolution()->CreateConstant(1)),
|
|
analysis.GetScalarEvolution()->CreateConstant(9));
|
|
}
|
|
{
|
|
// Function d
|
|
const ir::Function* f = spvtest::GetFunction(module, 12);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetLowerBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
5);
|
|
EXPECT_EQ(
|
|
analysis.GetUpperBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
14);
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetTripCount(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
10);
|
|
|
|
EXPECT_EQ(analysis.GetFirstTripInductionNode(loop),
|
|
analysis.GetScalarEvolution()->CreateConstant(5));
|
|
|
|
EXPECT_EQ(analysis.GetFinalTripInductionNode(
|
|
loop, analysis.GetScalarEvolution()->CreateConstant(1)),
|
|
analysis.GetScalarEvolution()->CreateConstant(14));
|
|
}
|
|
{
|
|
// Function e
|
|
const ir::Function* f = spvtest::GetFunction(module, 14);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetLowerBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
-10);
|
|
EXPECT_EQ(
|
|
analysis.GetUpperBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
0);
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetTripCount(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
11);
|
|
|
|
EXPECT_EQ(analysis.GetFirstTripInductionNode(loop),
|
|
analysis.GetScalarEvolution()->CreateConstant(-10));
|
|
|
|
EXPECT_EQ(analysis.GetFinalTripInductionNode(
|
|
loop, analysis.GetScalarEvolution()->CreateConstant(1)),
|
|
analysis.GetScalarEvolution()->CreateConstant(0));
|
|
}
|
|
{
|
|
// Function f
|
|
const ir::Function* f = spvtest::GetFunction(module, 16);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetLowerBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
-5);
|
|
EXPECT_EQ(
|
|
analysis.GetUpperBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
5);
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetTripCount(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
11);
|
|
|
|
EXPECT_EQ(analysis.GetFirstTripInductionNode(loop),
|
|
analysis.GetScalarEvolution()->CreateConstant(-5));
|
|
|
|
EXPECT_EQ(analysis.GetFinalTripInductionNode(
|
|
loop, analysis.GetScalarEvolution()->CreateConstant(1)),
|
|
analysis.GetScalarEvolution()->CreateConstant(5));
|
|
}
|
|
{
|
|
// Function g
|
|
const ir::Function* f = spvtest::GetFunction(module, 18);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetLowerBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
0);
|
|
EXPECT_EQ(
|
|
analysis.GetUpperBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
10);
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetTripCount(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
11);
|
|
|
|
EXPECT_EQ(analysis.GetFirstTripInductionNode(loop),
|
|
analysis.GetScalarEvolution()->CreateConstant(0));
|
|
|
|
EXPECT_EQ(analysis.GetFinalTripInductionNode(
|
|
loop, analysis.GetScalarEvolution()->CreateConstant(1)),
|
|
analysis.GetScalarEvolution()->CreateConstant(10));
|
|
}
|
|
{
|
|
// Function h
|
|
const ir::Function* f = spvtest::GetFunction(module, 20);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetLowerBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
5);
|
|
EXPECT_EQ(
|
|
analysis.GetUpperBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
15);
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetTripCount(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
11);
|
|
|
|
EXPECT_EQ(analysis.GetFirstTripInductionNode(loop),
|
|
analysis.GetScalarEvolution()->CreateConstant(5));
|
|
|
|
EXPECT_EQ(analysis.GetFinalTripInductionNode(
|
|
loop, analysis.GetScalarEvolution()->CreateConstant(1)),
|
|
analysis.GetScalarEvolution()->CreateConstant(15));
|
|
}
|
|
{
|
|
// Function i
|
|
const ir::Function* f = spvtest::GetFunction(module, 22);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetLowerBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
0);
|
|
EXPECT_EQ(
|
|
analysis.GetUpperBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
-9);
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetTripCount(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
10);
|
|
|
|
EXPECT_EQ(analysis.GetFirstTripInductionNode(loop),
|
|
analysis.GetScalarEvolution()->CreateConstant(0));
|
|
|
|
EXPECT_EQ(analysis.GetFinalTripInductionNode(
|
|
loop, analysis.GetScalarEvolution()->CreateConstant(-1)),
|
|
analysis.GetScalarEvolution()->CreateConstant(-9));
|
|
}
|
|
{
|
|
// Function j
|
|
const ir::Function* f = spvtest::GetFunction(module, 24);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetLowerBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
5);
|
|
EXPECT_EQ(
|
|
analysis.GetUpperBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
-4);
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetTripCount(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
10);
|
|
|
|
EXPECT_EQ(analysis.GetFirstTripInductionNode(loop),
|
|
analysis.GetScalarEvolution()->CreateConstant(5));
|
|
|
|
EXPECT_EQ(analysis.GetFinalTripInductionNode(
|
|
loop, analysis.GetScalarEvolution()->CreateConstant(-1)),
|
|
analysis.GetScalarEvolution()->CreateConstant(-4));
|
|
}
|
|
{
|
|
// Function k
|
|
const ir::Function* f = spvtest::GetFunction(module, 26);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetLowerBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
10);
|
|
EXPECT_EQ(
|
|
analysis.GetUpperBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
1);
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetTripCount(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
10);
|
|
|
|
EXPECT_EQ(analysis.GetFirstTripInductionNode(loop),
|
|
analysis.GetScalarEvolution()->CreateConstant(10));
|
|
|
|
EXPECT_EQ(analysis.GetFinalTripInductionNode(
|
|
loop, analysis.GetScalarEvolution()->CreateConstant(-1)),
|
|
analysis.GetScalarEvolution()->CreateConstant(1));
|
|
}
|
|
{
|
|
// Function l
|
|
const ir::Function* f = spvtest::GetFunction(module, 28);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetLowerBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
15);
|
|
EXPECT_EQ(
|
|
analysis.GetUpperBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
6);
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetTripCount(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
10);
|
|
|
|
EXPECT_EQ(analysis.GetFirstTripInductionNode(loop),
|
|
analysis.GetScalarEvolution()->CreateConstant(15));
|
|
|
|
EXPECT_EQ(analysis.GetFinalTripInductionNode(
|
|
loop, analysis.GetScalarEvolution()->CreateConstant(-1)),
|
|
analysis.GetScalarEvolution()->CreateConstant(6));
|
|
}
|
|
{
|
|
// Function m
|
|
const ir::Function* f = spvtest::GetFunction(module, 30);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetLowerBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
0);
|
|
EXPECT_EQ(
|
|
analysis.GetUpperBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
-10);
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetTripCount(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
11);
|
|
|
|
EXPECT_EQ(analysis.GetFirstTripInductionNode(loop),
|
|
analysis.GetScalarEvolution()->CreateConstant(0));
|
|
|
|
EXPECT_EQ(analysis.GetFinalTripInductionNode(
|
|
loop, analysis.GetScalarEvolution()->CreateConstant(-1)),
|
|
analysis.GetScalarEvolution()->CreateConstant(-10));
|
|
}
|
|
{
|
|
// Function n
|
|
const ir::Function* f = spvtest::GetFunction(module, 32);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetLowerBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
5);
|
|
EXPECT_EQ(
|
|
analysis.GetUpperBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
-5);
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetTripCount(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
11);
|
|
|
|
EXPECT_EQ(analysis.GetFirstTripInductionNode(loop),
|
|
analysis.GetScalarEvolution()->CreateConstant(5));
|
|
|
|
EXPECT_EQ(analysis.GetFinalTripInductionNode(
|
|
loop, analysis.GetScalarEvolution()->CreateConstant(-1)),
|
|
analysis.GetScalarEvolution()->CreateConstant(-5));
|
|
}
|
|
{
|
|
// Function o
|
|
const ir::Function* f = spvtest::GetFunction(module, 34);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetLowerBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
10);
|
|
EXPECT_EQ(
|
|
analysis.GetUpperBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
0);
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetTripCount(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
11);
|
|
|
|
EXPECT_EQ(analysis.GetFirstTripInductionNode(loop),
|
|
analysis.GetScalarEvolution()->CreateConstant(10));
|
|
|
|
EXPECT_EQ(analysis.GetFinalTripInductionNode(
|
|
loop, analysis.GetScalarEvolution()->CreateConstant(-1)),
|
|
analysis.GetScalarEvolution()->CreateConstant(0));
|
|
}
|
|
{
|
|
// Function p
|
|
const ir::Function* f = spvtest::GetFunction(module, 36);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetLowerBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
15);
|
|
EXPECT_EQ(
|
|
analysis.GetUpperBound(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
5);
|
|
|
|
EXPECT_EQ(
|
|
analysis.GetTripCount(loop)->AsSEConstantNode()->FoldToSingleValue(),
|
|
11);
|
|
|
|
EXPECT_EQ(analysis.GetFirstTripInductionNode(loop),
|
|
analysis.GetScalarEvolution()->CreateConstant(15));
|
|
|
|
EXPECT_EQ(analysis.GetFinalTripInductionNode(
|
|
loop, analysis.GetScalarEvolution()->CreateConstant(-1)),
|
|
analysis.GetScalarEvolution()->CreateConstant(5));
|
|
}
|
|
}
|
|
|
|
/*
|
|
Generated from the following GLSL fragment shader
|
|
with --eliminate-local-multi-store
|
|
#version 440 core
|
|
void main(){
|
|
for (int i = 0; i < 10; i++) {
|
|
|
|
}
|
|
}
|
|
*/
|
|
TEST(DependencyAnalysisHelpers, bounds_checks) {
|
|
const std::string text = R"( OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource GLSL 440
|
|
OpName %4 "main"
|
|
OpName %8 "i"
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeInt 32 1
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstant %6 0
|
|
%16 = OpConstant %6 10
|
|
%17 = OpTypeBool
|
|
%20 = OpConstant %6 1
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%8 = OpVariable %7 Function
|
|
OpStore %8 %9
|
|
OpBranch %10
|
|
%10 = OpLabel
|
|
%22 = OpPhi %6 %9 %5 %21 %13
|
|
OpLoopMerge %12 %13 None
|
|
OpBranch %14
|
|
%14 = OpLabel
|
|
%18 = OpSLessThan %17 %22 %16
|
|
OpBranchConditional %18 %11 %12
|
|
%11 = OpLabel
|
|
OpBranch %13
|
|
%13 = OpLabel
|
|
%21 = OpIAdd %6 %22 %20
|
|
OpStore %8 %21
|
|
OpBranch %10
|
|
%12 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
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;
|
|
// We need a shader that includes a loop for this test so we can build a
|
|
// LoopDependenceAnalaysis
|
|
const ir::Function* f = spvtest::GetFunction(module, 4);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
EXPECT_TRUE(analysis.IsWithinBounds(0, 0, 0));
|
|
EXPECT_TRUE(analysis.IsWithinBounds(0, -1, 0));
|
|
EXPECT_TRUE(analysis.IsWithinBounds(0, 0, 1));
|
|
EXPECT_TRUE(analysis.IsWithinBounds(0, -1, 1));
|
|
EXPECT_TRUE(analysis.IsWithinBounds(-2, -2, -2));
|
|
EXPECT_TRUE(analysis.IsWithinBounds(-2, -3, 0));
|
|
EXPECT_TRUE(analysis.IsWithinBounds(-2, 0, -3));
|
|
EXPECT_TRUE(analysis.IsWithinBounds(2, 2, 2));
|
|
EXPECT_TRUE(analysis.IsWithinBounds(2, 3, 0));
|
|
|
|
EXPECT_FALSE(analysis.IsWithinBounds(2, 3, 3));
|
|
EXPECT_FALSE(analysis.IsWithinBounds(0, 1, 5));
|
|
EXPECT_FALSE(analysis.IsWithinBounds(0, -1, -4));
|
|
EXPECT_FALSE(analysis.IsWithinBounds(-2, -4, -3));
|
|
}
|
|
|
|
/*
|
|
Generated from the following GLSL fragment shader
|
|
with --eliminate-local-multi-store
|
|
#version 440 core
|
|
layout(location = 0) in vec4 in_vec;
|
|
// Loop iterates from constant to symbolic
|
|
void a() {
|
|
int N = int(in_vec.x);
|
|
int arr[10];
|
|
for (int i = 0; i < N; i++) { // Bounds are N - 0 - 1
|
|
arr[i] = arr[i+N]; // |distance| = N
|
|
arr[i+N] = arr[i]; // |distance| = N
|
|
}
|
|
}
|
|
void b() {
|
|
int N = int(in_vec.x);
|
|
int arr[10];
|
|
for (int i = 0; i <= N; i++) { // Bounds are N - 0
|
|
arr[i] = arr[i+N]; // |distance| = N
|
|
arr[i+N] = arr[i]; // |distance| = N
|
|
}
|
|
}
|
|
void c() {
|
|
int N = int(in_vec.x);
|
|
int arr[10];
|
|
for (int i = 9; i > N; i--) { // Bounds are 9 - N - 1
|
|
arr[i] = arr[i+N]; // |distance| = N
|
|
arr[i+N] = arr[i]; // |distance| = N
|
|
}
|
|
}
|
|
void d() {
|
|
int N = int(in_vec.x);
|
|
int arr[10];
|
|
for (int i = 9; i >= N; i--) { // Bounds are 9 - N
|
|
arr[i] = arr[i+N]; // |distance| = N
|
|
arr[i+N] = arr[i]; // |distance| = N
|
|
}
|
|
}
|
|
void main(){
|
|
a();
|
|
b();
|
|
c();
|
|
d();
|
|
}
|
|
*/
|
|
TEST(DependencyAnalysisHelpers, const_to_symbolic) {
|
|
const std::string text = R"( OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %20
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource GLSL 440
|
|
OpName %4 "main"
|
|
OpName %6 "a("
|
|
OpName %8 "b("
|
|
OpName %10 "c("
|
|
OpName %12 "d("
|
|
OpName %16 "N"
|
|
OpName %20 "in_vec"
|
|
OpName %27 "i"
|
|
OpName %41 "arr"
|
|
OpName %59 "N"
|
|
OpName %63 "i"
|
|
OpName %72 "arr"
|
|
OpName %89 "N"
|
|
OpName %93 "i"
|
|
OpName %103 "arr"
|
|
OpName %120 "N"
|
|
OpName %124 "i"
|
|
OpName %133 "arr"
|
|
OpDecorate %20 Location 0
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%14 = OpTypeInt 32 1
|
|
%15 = OpTypePointer Function %14
|
|
%17 = OpTypeFloat 32
|
|
%18 = OpTypeVector %17 4
|
|
%19 = OpTypePointer Input %18
|
|
%20 = OpVariable %19 Input
|
|
%21 = OpTypeInt 32 0
|
|
%22 = OpConstant %21 0
|
|
%23 = OpTypePointer Input %17
|
|
%28 = OpConstant %14 0
|
|
%36 = OpTypeBool
|
|
%38 = OpConstant %21 10
|
|
%39 = OpTypeArray %14 %38
|
|
%40 = OpTypePointer Function %39
|
|
%57 = OpConstant %14 1
|
|
%94 = OpConstant %14 9
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%150 = OpFunctionCall %2 %6
|
|
%151 = OpFunctionCall %2 %8
|
|
%152 = OpFunctionCall %2 %10
|
|
%153 = OpFunctionCall %2 %12
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%6 = OpFunction %2 None %3
|
|
%7 = OpLabel
|
|
%16 = OpVariable %15 Function
|
|
%27 = OpVariable %15 Function
|
|
%41 = OpVariable %40 Function
|
|
%24 = OpAccessChain %23 %20 %22
|
|
%25 = OpLoad %17 %24
|
|
%26 = OpConvertFToS %14 %25
|
|
OpStore %16 %26
|
|
OpStore %27 %28
|
|
OpBranch %29
|
|
%29 = OpLabel
|
|
%154 = OpPhi %14 %28 %7 %58 %32
|
|
OpLoopMerge %31 %32 None
|
|
OpBranch %33
|
|
%33 = OpLabel
|
|
%37 = OpSLessThan %36 %154 %26
|
|
OpBranchConditional %37 %30 %31
|
|
%30 = OpLabel
|
|
%45 = OpIAdd %14 %154 %26
|
|
%46 = OpAccessChain %15 %41 %45
|
|
%47 = OpLoad %14 %46
|
|
%48 = OpAccessChain %15 %41 %154
|
|
OpStore %48 %47
|
|
%51 = OpIAdd %14 %154 %26
|
|
%53 = OpAccessChain %15 %41 %154
|
|
%54 = OpLoad %14 %53
|
|
%55 = OpAccessChain %15 %41 %51
|
|
OpStore %55 %54
|
|
OpBranch %32
|
|
%32 = OpLabel
|
|
%58 = OpIAdd %14 %154 %57
|
|
OpStore %27 %58
|
|
OpBranch %29
|
|
%31 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%8 = OpFunction %2 None %3
|
|
%9 = OpLabel
|
|
%59 = OpVariable %15 Function
|
|
%63 = OpVariable %15 Function
|
|
%72 = OpVariable %40 Function
|
|
%60 = OpAccessChain %23 %20 %22
|
|
%61 = OpLoad %17 %60
|
|
%62 = OpConvertFToS %14 %61
|
|
OpStore %59 %62
|
|
OpStore %63 %28
|
|
OpBranch %64
|
|
%64 = OpLabel
|
|
%155 = OpPhi %14 %28 %9 %88 %67
|
|
OpLoopMerge %66 %67 None
|
|
OpBranch %68
|
|
%68 = OpLabel
|
|
%71 = OpSLessThanEqual %36 %155 %62
|
|
OpBranchConditional %71 %65 %66
|
|
%65 = OpLabel
|
|
%76 = OpIAdd %14 %155 %62
|
|
%77 = OpAccessChain %15 %72 %76
|
|
%78 = OpLoad %14 %77
|
|
%79 = OpAccessChain %15 %72 %155
|
|
OpStore %79 %78
|
|
%82 = OpIAdd %14 %155 %62
|
|
%84 = OpAccessChain %15 %72 %155
|
|
%85 = OpLoad %14 %84
|
|
%86 = OpAccessChain %15 %72 %82
|
|
OpStore %86 %85
|
|
OpBranch %67
|
|
%67 = OpLabel
|
|
%88 = OpIAdd %14 %155 %57
|
|
OpStore %63 %88
|
|
OpBranch %64
|
|
%66 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%10 = OpFunction %2 None %3
|
|
%11 = OpLabel
|
|
%89 = OpVariable %15 Function
|
|
%93 = OpVariable %15 Function
|
|
%103 = OpVariable %40 Function
|
|
%90 = OpAccessChain %23 %20 %22
|
|
%91 = OpLoad %17 %90
|
|
%92 = OpConvertFToS %14 %91
|
|
OpStore %89 %92
|
|
OpStore %93 %94
|
|
OpBranch %95
|
|
%95 = OpLabel
|
|
%156 = OpPhi %14 %94 %11 %119 %98
|
|
OpLoopMerge %97 %98 None
|
|
OpBranch %99
|
|
%99 = OpLabel
|
|
%102 = OpSGreaterThan %36 %156 %92
|
|
OpBranchConditional %102 %96 %97
|
|
%96 = OpLabel
|
|
%107 = OpIAdd %14 %156 %92
|
|
%108 = OpAccessChain %15 %103 %107
|
|
%109 = OpLoad %14 %108
|
|
%110 = OpAccessChain %15 %103 %156
|
|
OpStore %110 %109
|
|
%113 = OpIAdd %14 %156 %92
|
|
%115 = OpAccessChain %15 %103 %156
|
|
%116 = OpLoad %14 %115
|
|
%117 = OpAccessChain %15 %103 %113
|
|
OpStore %117 %116
|
|
OpBranch %98
|
|
%98 = OpLabel
|
|
%119 = OpISub %14 %156 %57
|
|
OpStore %93 %119
|
|
OpBranch %95
|
|
%97 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%12 = OpFunction %2 None %3
|
|
%13 = OpLabel
|
|
%120 = OpVariable %15 Function
|
|
%124 = OpVariable %15 Function
|
|
%133 = OpVariable %40 Function
|
|
%121 = OpAccessChain %23 %20 %22
|
|
%122 = OpLoad %17 %121
|
|
%123 = OpConvertFToS %14 %122
|
|
OpStore %120 %123
|
|
OpStore %124 %94
|
|
OpBranch %125
|
|
%125 = OpLabel
|
|
%157 = OpPhi %14 %94 %13 %149 %128
|
|
OpLoopMerge %127 %128 None
|
|
OpBranch %129
|
|
%129 = OpLabel
|
|
%132 = OpSGreaterThanEqual %36 %157 %123
|
|
OpBranchConditional %132 %126 %127
|
|
%126 = OpLabel
|
|
%137 = OpIAdd %14 %157 %123
|
|
%138 = OpAccessChain %15 %133 %137
|
|
%139 = OpLoad %14 %138
|
|
%140 = OpAccessChain %15 %133 %157
|
|
OpStore %140 %139
|
|
%143 = OpIAdd %14 %157 %123
|
|
%145 = OpAccessChain %15 %133 %157
|
|
%146 = OpLoad %14 %145
|
|
%147 = OpAccessChain %15 %133 %143
|
|
OpStore %147 %146
|
|
OpBranch %128
|
|
%128 = OpLabel
|
|
%149 = OpISub %14 %157 %57
|
|
OpStore %124 %149
|
|
OpBranch %125
|
|
%127 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
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;
|
|
|
|
{
|
|
// Function a
|
|
const ir::Function* f = spvtest::GetFunction(module, 6);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
const ir::Instruction* stores[2];
|
|
int stores_found = 0;
|
|
for (const ir::Instruction& inst : *spvtest::GetBasicBlock(f, 30)) {
|
|
if (inst.opcode() == SpvOp::SpvOpStore) {
|
|
stores[stores_found] = &inst;
|
|
++stores_found;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
EXPECT_TRUE(stores[i]);
|
|
}
|
|
|
|
// 47 -> 48
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(47)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[0]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
// Independent and supported.
|
|
EXPECT_TRUE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
|
|
// 54 -> 55
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(54)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[1]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
// Independent but not supported.
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
}
|
|
{
|
|
// Function b
|
|
const ir::Function* f = spvtest::GetFunction(module, 8);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
const ir::Instruction* stores[2];
|
|
int stores_found = 0;
|
|
for (const ir::Instruction& inst : *spvtest::GetBasicBlock(f, 65)) {
|
|
if (inst.opcode() == SpvOp::SpvOpStore) {
|
|
stores[stores_found] = &inst;
|
|
++stores_found;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
EXPECT_TRUE(stores[i]);
|
|
}
|
|
|
|
// 78 -> 79
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(78)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[0]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
// Dependent.
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
|
|
// 85 -> 86
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(85)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[1]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
// Dependent.
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
}
|
|
{
|
|
// Function c
|
|
const ir::Function* f = spvtest::GetFunction(module, 10);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
const ir::Instruction* stores[2];
|
|
int stores_found = 0;
|
|
for (const ir::Instruction& inst : *spvtest::GetBasicBlock(f, 96)) {
|
|
if (inst.opcode() == SpvOp::SpvOpStore) {
|
|
stores[stores_found] = &inst;
|
|
++stores_found;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
EXPECT_TRUE(stores[i]);
|
|
}
|
|
|
|
// 109 -> 110
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(109)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[0]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
// Independent but not supported.
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
|
|
// 116 -> 117
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(116)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[1]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
// Independent but not supported.
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
}
|
|
{
|
|
// Function d
|
|
const ir::Function* f = spvtest::GetFunction(module, 12);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
const ir::Instruction* stores[2];
|
|
int stores_found = 0;
|
|
for (const ir::Instruction& inst : *spvtest::GetBasicBlock(f, 126)) {
|
|
if (inst.opcode() == SpvOp::SpvOpStore) {
|
|
stores[stores_found] = &inst;
|
|
++stores_found;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
EXPECT_TRUE(stores[i]);
|
|
}
|
|
|
|
// 139 -> 140
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(139)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[0]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
// Dependent.
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
|
|
// 146 -> 147
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(146)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[1]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
// Dependent.
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
Generated from the following GLSL fragment shader
|
|
with --eliminate-local-multi-store
|
|
#version 440 core
|
|
layout(location = 0) in vec4 in_vec;
|
|
// Loop iterates from symbolic to constant
|
|
void a() {
|
|
int N = int(in_vec.x);
|
|
int arr[10];
|
|
for (int i = N; i < 9; i++) { // Bounds are 9 - N - 1
|
|
arr[i] = arr[i+N]; // |distance| = N
|
|
arr[i+N] = arr[i]; // |distance| = N
|
|
}
|
|
}
|
|
void b() {
|
|
int N = int(in_vec.x);
|
|
int arr[10];
|
|
for (int i = N; i <= 9; i++) { // Bounds are 9 - N
|
|
arr[i] = arr[i+N]; // |distance| = N
|
|
arr[i+N] = arr[i]; // |distance| = N
|
|
}
|
|
}
|
|
void c() {
|
|
int N = int(in_vec.x);
|
|
int arr[10];
|
|
for (int i = N; i > 0; i--) { // Bounds are N - 0 - 1
|
|
arr[i] = arr[i+N]; // |distance| = N
|
|
arr[i+N] = arr[i]; // |distance| = N
|
|
}
|
|
}
|
|
void d() {
|
|
int N = int(in_vec.x);
|
|
int arr[10];
|
|
for (int i = N; i >= 0; i--) { // Bounds are N - 0
|
|
arr[i] = arr[i+N]; // |distance| = N
|
|
arr[i+N] = arr[i]; // |distance| = N
|
|
}
|
|
}
|
|
void main(){
|
|
a();
|
|
b();
|
|
c();
|
|
d();
|
|
}
|
|
*/
|
|
TEST(DependencyAnalysisHelpers, symbolic_to_const) {
|
|
const std::string text = R"( OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %20
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource GLSL 440
|
|
OpName %4 "main"
|
|
OpName %6 "a("
|
|
OpName %8 "b("
|
|
OpName %10 "c("
|
|
OpName %12 "d("
|
|
OpName %16 "N"
|
|
OpName %20 "in_vec"
|
|
OpName %27 "i"
|
|
OpName %41 "arr"
|
|
OpName %59 "N"
|
|
OpName %63 "i"
|
|
OpName %72 "arr"
|
|
OpName %89 "N"
|
|
OpName %93 "i"
|
|
OpName %103 "arr"
|
|
OpName %120 "N"
|
|
OpName %124 "i"
|
|
OpName %133 "arr"
|
|
OpDecorate %20 Location 0
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%14 = OpTypeInt 32 1
|
|
%15 = OpTypePointer Function %14
|
|
%17 = OpTypeFloat 32
|
|
%18 = OpTypeVector %17 4
|
|
%19 = OpTypePointer Input %18
|
|
%20 = OpVariable %19 Input
|
|
%21 = OpTypeInt 32 0
|
|
%22 = OpConstant %21 0
|
|
%23 = OpTypePointer Input %17
|
|
%35 = OpConstant %14 9
|
|
%36 = OpTypeBool
|
|
%38 = OpConstant %21 10
|
|
%39 = OpTypeArray %14 %38
|
|
%40 = OpTypePointer Function %39
|
|
%57 = OpConstant %14 1
|
|
%101 = OpConstant %14 0
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%150 = OpFunctionCall %2 %6
|
|
%151 = OpFunctionCall %2 %8
|
|
%152 = OpFunctionCall %2 %10
|
|
%153 = OpFunctionCall %2 %12
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%6 = OpFunction %2 None %3
|
|
%7 = OpLabel
|
|
%16 = OpVariable %15 Function
|
|
%27 = OpVariable %15 Function
|
|
%41 = OpVariable %40 Function
|
|
%24 = OpAccessChain %23 %20 %22
|
|
%25 = OpLoad %17 %24
|
|
%26 = OpConvertFToS %14 %25
|
|
OpStore %16 %26
|
|
OpStore %27 %26
|
|
OpBranch %29
|
|
%29 = OpLabel
|
|
%154 = OpPhi %14 %26 %7 %58 %32
|
|
OpLoopMerge %31 %32 None
|
|
OpBranch %33
|
|
%33 = OpLabel
|
|
%37 = OpSLessThan %36 %154 %35
|
|
OpBranchConditional %37 %30 %31
|
|
%30 = OpLabel
|
|
%45 = OpIAdd %14 %154 %26
|
|
%46 = OpAccessChain %15 %41 %45
|
|
%47 = OpLoad %14 %46
|
|
%48 = OpAccessChain %15 %41 %154
|
|
OpStore %48 %47
|
|
%51 = OpIAdd %14 %154 %26
|
|
%53 = OpAccessChain %15 %41 %154
|
|
%54 = OpLoad %14 %53
|
|
%55 = OpAccessChain %15 %41 %51
|
|
OpStore %55 %54
|
|
OpBranch %32
|
|
%32 = OpLabel
|
|
%58 = OpIAdd %14 %154 %57
|
|
OpStore %27 %58
|
|
OpBranch %29
|
|
%31 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%8 = OpFunction %2 None %3
|
|
%9 = OpLabel
|
|
%59 = OpVariable %15 Function
|
|
%63 = OpVariable %15 Function
|
|
%72 = OpVariable %40 Function
|
|
%60 = OpAccessChain %23 %20 %22
|
|
%61 = OpLoad %17 %60
|
|
%62 = OpConvertFToS %14 %61
|
|
OpStore %59 %62
|
|
OpStore %63 %62
|
|
OpBranch %65
|
|
%65 = OpLabel
|
|
%155 = OpPhi %14 %62 %9 %88 %68
|
|
OpLoopMerge %67 %68 None
|
|
OpBranch %69
|
|
%69 = OpLabel
|
|
%71 = OpSLessThanEqual %36 %155 %35
|
|
OpBranchConditional %71 %66 %67
|
|
%66 = OpLabel
|
|
%76 = OpIAdd %14 %155 %62
|
|
%77 = OpAccessChain %15 %72 %76
|
|
%78 = OpLoad %14 %77
|
|
%79 = OpAccessChain %15 %72 %155
|
|
OpStore %79 %78
|
|
%82 = OpIAdd %14 %155 %62
|
|
%84 = OpAccessChain %15 %72 %155
|
|
%85 = OpLoad %14 %84
|
|
%86 = OpAccessChain %15 %72 %82
|
|
OpStore %86 %85
|
|
OpBranch %68
|
|
%68 = OpLabel
|
|
%88 = OpIAdd %14 %155 %57
|
|
OpStore %63 %88
|
|
OpBranch %65
|
|
%67 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%10 = OpFunction %2 None %3
|
|
%11 = OpLabel
|
|
%89 = OpVariable %15 Function
|
|
%93 = OpVariable %15 Function
|
|
%103 = OpVariable %40 Function
|
|
%90 = OpAccessChain %23 %20 %22
|
|
%91 = OpLoad %17 %90
|
|
%92 = OpConvertFToS %14 %91
|
|
OpStore %89 %92
|
|
OpStore %93 %92
|
|
OpBranch %95
|
|
%95 = OpLabel
|
|
%156 = OpPhi %14 %92 %11 %119 %98
|
|
OpLoopMerge %97 %98 None
|
|
OpBranch %99
|
|
%99 = OpLabel
|
|
%102 = OpSGreaterThan %36 %156 %101
|
|
OpBranchConditional %102 %96 %97
|
|
%96 = OpLabel
|
|
%107 = OpIAdd %14 %156 %92
|
|
%108 = OpAccessChain %15 %103 %107
|
|
%109 = OpLoad %14 %108
|
|
%110 = OpAccessChain %15 %103 %156
|
|
OpStore %110 %109
|
|
%113 = OpIAdd %14 %156 %92
|
|
%115 = OpAccessChain %15 %103 %156
|
|
%116 = OpLoad %14 %115
|
|
%117 = OpAccessChain %15 %103 %113
|
|
OpStore %117 %116
|
|
OpBranch %98
|
|
%98 = OpLabel
|
|
%119 = OpISub %14 %156 %57
|
|
OpStore %93 %119
|
|
OpBranch %95
|
|
%97 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%12 = OpFunction %2 None %3
|
|
%13 = OpLabel
|
|
%120 = OpVariable %15 Function
|
|
%124 = OpVariable %15 Function
|
|
%133 = OpVariable %40 Function
|
|
%121 = OpAccessChain %23 %20 %22
|
|
%122 = OpLoad %17 %121
|
|
%123 = OpConvertFToS %14 %122
|
|
OpStore %120 %123
|
|
OpStore %124 %123
|
|
OpBranch %126
|
|
%126 = OpLabel
|
|
%157 = OpPhi %14 %123 %13 %149 %129
|
|
OpLoopMerge %128 %129 None
|
|
OpBranch %130
|
|
%130 = OpLabel
|
|
%132 = OpSGreaterThanEqual %36 %157 %101
|
|
OpBranchConditional %132 %127 %128
|
|
%127 = OpLabel
|
|
%137 = OpIAdd %14 %157 %123
|
|
%138 = OpAccessChain %15 %133 %137
|
|
%139 = OpLoad %14 %138
|
|
%140 = OpAccessChain %15 %133 %157
|
|
OpStore %140 %139
|
|
%143 = OpIAdd %14 %157 %123
|
|
%145 = OpAccessChain %15 %133 %157
|
|
%146 = OpLoad %14 %145
|
|
%147 = OpAccessChain %15 %133 %143
|
|
OpStore %147 %146
|
|
OpBranch %129
|
|
%129 = OpLabel
|
|
%149 = OpISub %14 %157 %57
|
|
OpStore %124 %149
|
|
OpBranch %126
|
|
%128 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
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;
|
|
{
|
|
// Function a
|
|
const ir::Function* f = spvtest::GetFunction(module, 6);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
const ir::Instruction* stores[2];
|
|
int stores_found = 0;
|
|
for (const ir::Instruction& inst : *spvtest::GetBasicBlock(f, 30)) {
|
|
if (inst.opcode() == SpvOp::SpvOpStore) {
|
|
stores[stores_found] = &inst;
|
|
++stores_found;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
EXPECT_TRUE(stores[i]);
|
|
}
|
|
|
|
// 47 -> 48
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(47)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[0]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
// Independent but not supported.
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
|
|
// 54 -> 55
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(54)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[1]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
// Independent but not supported.
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
}
|
|
{
|
|
// Function b
|
|
const ir::Function* f = spvtest::GetFunction(module, 8);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
const ir::Instruction* stores[2];
|
|
int stores_found = 0;
|
|
for (const ir::Instruction& inst : *spvtest::GetBasicBlock(f, 66)) {
|
|
if (inst.opcode() == SpvOp::SpvOpStore) {
|
|
stores[stores_found] = &inst;
|
|
++stores_found;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
EXPECT_TRUE(stores[i]);
|
|
}
|
|
|
|
// 78 -> 79
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(78)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[0]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
// Dependent.
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
|
|
// 85 -> 86
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(85)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[1]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
// Dependent.
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
}
|
|
{
|
|
// Function c
|
|
const ir::Function* f = spvtest::GetFunction(module, 10);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
const ir::Instruction* stores[2];
|
|
int stores_found = 0;
|
|
for (const ir::Instruction& inst : *spvtest::GetBasicBlock(f, 96)) {
|
|
if (inst.opcode() == SpvOp::SpvOpStore) {
|
|
stores[stores_found] = &inst;
|
|
++stores_found;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
EXPECT_TRUE(stores[i]);
|
|
}
|
|
|
|
// 109 -> 110
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(109)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[0]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
// Independent and supported.
|
|
EXPECT_TRUE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
|
|
// 116 -> 117
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(116)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[1]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
// Independent but not supported.
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
}
|
|
{
|
|
// Function d
|
|
const ir::Function* f = spvtest::GetFunction(module, 12);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
const ir::Instruction* stores[2];
|
|
int stores_found = 0;
|
|
for (const ir::Instruction& inst : *spvtest::GetBasicBlock(f, 127)) {
|
|
if (inst.opcode() == SpvOp::SpvOpStore) {
|
|
stores[stores_found] = &inst;
|
|
++stores_found;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
EXPECT_TRUE(stores[i]);
|
|
}
|
|
|
|
// 139 -> 140
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(139)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[0]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
// Dependent
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
|
|
// 146 -> 147
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(146)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[1]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
// Dependent
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
Generated from the following GLSL fragment shader
|
|
with --eliminate-local-multi-store
|
|
#version 440 core
|
|
layout(location = 0) in vec4 in_vec;
|
|
// Loop iterates from symbolic to symbolic
|
|
void a() {
|
|
int M = int(in_vec.x);
|
|
int N = int(in_vec.y);
|
|
int arr[10];
|
|
for (int i = M; i < N; i++) { // Bounds are N - M - 1
|
|
arr[i+M+N] = arr[i+M+2*N]; // |distance| = N
|
|
arr[i+M+2*N] = arr[i+M+N]; // |distance| = N
|
|
}
|
|
}
|
|
void b() {
|
|
int M = int(in_vec.x);
|
|
int N = int(in_vec.y);
|
|
int arr[10];
|
|
for (int i = M; i <= N; i++) { // Bounds are N - M
|
|
arr[i+M+N] = arr[i+M+2*N]; // |distance| = N
|
|
arr[i+M+2*N] = arr[i+M+N]; // |distance| = N
|
|
}
|
|
}
|
|
void c() {
|
|
int M = int(in_vec.x);
|
|
int N = int(in_vec.y);
|
|
int arr[10];
|
|
for (int i = M; i > N; i--) { // Bounds are M - N - 1
|
|
arr[i+M+N] = arr[i+M+2*N]; // |distance| = N
|
|
arr[i+M+2*N] = arr[i+M+N]; // |distance| = N
|
|
}
|
|
}
|
|
void d() {
|
|
int M = int(in_vec.x);
|
|
int N = int(in_vec.y);
|
|
int arr[10];
|
|
for (int i = M; i >= N; i--) { // Bounds are M - N
|
|
arr[i+M+N] = arr[i+M+2*N]; // |distance| = N
|
|
arr[i+M+2*N] = arr[i+M+N]; // |distance| = N
|
|
}
|
|
}
|
|
void main(){
|
|
a();
|
|
b();
|
|
c();
|
|
d();
|
|
}
|
|
*/
|
|
TEST(DependencyAnalysisHelpers, symbolic_to_symbolic) {
|
|
const std::string text = R"( OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %20
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource GLSL 440
|
|
OpName %4 "main"
|
|
OpName %6 "a("
|
|
OpName %8 "b("
|
|
OpName %10 "c("
|
|
OpName %12 "d("
|
|
OpName %16 "M"
|
|
OpName %20 "in_vec"
|
|
OpName %27 "N"
|
|
OpName %32 "i"
|
|
OpName %46 "arr"
|
|
OpName %79 "M"
|
|
OpName %83 "N"
|
|
OpName %87 "i"
|
|
OpName %97 "arr"
|
|
OpName %128 "M"
|
|
OpName %132 "N"
|
|
OpName %136 "i"
|
|
OpName %146 "arr"
|
|
OpName %177 "M"
|
|
OpName %181 "N"
|
|
OpName %185 "i"
|
|
OpName %195 "arr"
|
|
OpDecorate %20 Location 0
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%14 = OpTypeInt 32 1
|
|
%15 = OpTypePointer Function %14
|
|
%17 = OpTypeFloat 32
|
|
%18 = OpTypeVector %17 4
|
|
%19 = OpTypePointer Input %18
|
|
%20 = OpVariable %19 Input
|
|
%21 = OpTypeInt 32 0
|
|
%22 = OpConstant %21 0
|
|
%23 = OpTypePointer Input %17
|
|
%28 = OpConstant %21 1
|
|
%41 = OpTypeBool
|
|
%43 = OpConstant %21 10
|
|
%44 = OpTypeArray %14 %43
|
|
%45 = OpTypePointer Function %44
|
|
%55 = OpConstant %14 2
|
|
%77 = OpConstant %14 1
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%226 = OpFunctionCall %2 %6
|
|
%227 = OpFunctionCall %2 %8
|
|
%228 = OpFunctionCall %2 %10
|
|
%229 = OpFunctionCall %2 %12
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%6 = OpFunction %2 None %3
|
|
%7 = OpLabel
|
|
%16 = OpVariable %15 Function
|
|
%27 = OpVariable %15 Function
|
|
%32 = OpVariable %15 Function
|
|
%46 = OpVariable %45 Function
|
|
%24 = OpAccessChain %23 %20 %22
|
|
%25 = OpLoad %17 %24
|
|
%26 = OpConvertFToS %14 %25
|
|
OpStore %16 %26
|
|
%29 = OpAccessChain %23 %20 %28
|
|
%30 = OpLoad %17 %29
|
|
%31 = OpConvertFToS %14 %30
|
|
OpStore %27 %31
|
|
OpStore %32 %26
|
|
OpBranch %34
|
|
%34 = OpLabel
|
|
%230 = OpPhi %14 %26 %7 %78 %37
|
|
OpLoopMerge %36 %37 None
|
|
OpBranch %38
|
|
%38 = OpLabel
|
|
%42 = OpSLessThan %41 %230 %31
|
|
OpBranchConditional %42 %35 %36
|
|
%35 = OpLabel
|
|
%49 = OpIAdd %14 %230 %26
|
|
%51 = OpIAdd %14 %49 %31
|
|
%54 = OpIAdd %14 %230 %26
|
|
%57 = OpIMul %14 %55 %31
|
|
%58 = OpIAdd %14 %54 %57
|
|
%59 = OpAccessChain %15 %46 %58
|
|
%60 = OpLoad %14 %59
|
|
%61 = OpAccessChain %15 %46 %51
|
|
OpStore %61 %60
|
|
%64 = OpIAdd %14 %230 %26
|
|
%66 = OpIMul %14 %55 %31
|
|
%67 = OpIAdd %14 %64 %66
|
|
%70 = OpIAdd %14 %230 %26
|
|
%72 = OpIAdd %14 %70 %31
|
|
%73 = OpAccessChain %15 %46 %72
|
|
%74 = OpLoad %14 %73
|
|
%75 = OpAccessChain %15 %46 %67
|
|
OpStore %75 %74
|
|
OpBranch %37
|
|
%37 = OpLabel
|
|
%78 = OpIAdd %14 %230 %77
|
|
OpStore %32 %78
|
|
OpBranch %34
|
|
%36 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%8 = OpFunction %2 None %3
|
|
%9 = OpLabel
|
|
%79 = OpVariable %15 Function
|
|
%83 = OpVariable %15 Function
|
|
%87 = OpVariable %15 Function
|
|
%97 = OpVariable %45 Function
|
|
%80 = OpAccessChain %23 %20 %22
|
|
%81 = OpLoad %17 %80
|
|
%82 = OpConvertFToS %14 %81
|
|
OpStore %79 %82
|
|
%84 = OpAccessChain %23 %20 %28
|
|
%85 = OpLoad %17 %84
|
|
%86 = OpConvertFToS %14 %85
|
|
OpStore %83 %86
|
|
OpStore %87 %82
|
|
OpBranch %89
|
|
%89 = OpLabel
|
|
%231 = OpPhi %14 %82 %9 %127 %92
|
|
OpLoopMerge %91 %92 None
|
|
OpBranch %93
|
|
%93 = OpLabel
|
|
%96 = OpSLessThanEqual %41 %231 %86
|
|
OpBranchConditional %96 %90 %91
|
|
%90 = OpLabel
|
|
%100 = OpIAdd %14 %231 %82
|
|
%102 = OpIAdd %14 %100 %86
|
|
%105 = OpIAdd %14 %231 %82
|
|
%107 = OpIMul %14 %55 %86
|
|
%108 = OpIAdd %14 %105 %107
|
|
%109 = OpAccessChain %15 %97 %108
|
|
%110 = OpLoad %14 %109
|
|
%111 = OpAccessChain %15 %97 %102
|
|
OpStore %111 %110
|
|
%114 = OpIAdd %14 %231 %82
|
|
%116 = OpIMul %14 %55 %86
|
|
%117 = OpIAdd %14 %114 %116
|
|
%120 = OpIAdd %14 %231 %82
|
|
%122 = OpIAdd %14 %120 %86
|
|
%123 = OpAccessChain %15 %97 %122
|
|
%124 = OpLoad %14 %123
|
|
%125 = OpAccessChain %15 %97 %117
|
|
OpStore %125 %124
|
|
OpBranch %92
|
|
%92 = OpLabel
|
|
%127 = OpIAdd %14 %231 %77
|
|
OpStore %87 %127
|
|
OpBranch %89
|
|
%91 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%10 = OpFunction %2 None %3
|
|
%11 = OpLabel
|
|
%128 = OpVariable %15 Function
|
|
%132 = OpVariable %15 Function
|
|
%136 = OpVariable %15 Function
|
|
%146 = OpVariable %45 Function
|
|
%129 = OpAccessChain %23 %20 %22
|
|
%130 = OpLoad %17 %129
|
|
%131 = OpConvertFToS %14 %130
|
|
OpStore %128 %131
|
|
%133 = OpAccessChain %23 %20 %28
|
|
%134 = OpLoad %17 %133
|
|
%135 = OpConvertFToS %14 %134
|
|
OpStore %132 %135
|
|
OpStore %136 %131
|
|
OpBranch %138
|
|
%138 = OpLabel
|
|
%232 = OpPhi %14 %131 %11 %176 %141
|
|
OpLoopMerge %140 %141 None
|
|
OpBranch %142
|
|
%142 = OpLabel
|
|
%145 = OpSGreaterThan %41 %232 %135
|
|
OpBranchConditional %145 %139 %140
|
|
%139 = OpLabel
|
|
%149 = OpIAdd %14 %232 %131
|
|
%151 = OpIAdd %14 %149 %135
|
|
%154 = OpIAdd %14 %232 %131
|
|
%156 = OpIMul %14 %55 %135
|
|
%157 = OpIAdd %14 %154 %156
|
|
%158 = OpAccessChain %15 %146 %157
|
|
%159 = OpLoad %14 %158
|
|
%160 = OpAccessChain %15 %146 %151
|
|
OpStore %160 %159
|
|
%163 = OpIAdd %14 %232 %131
|
|
%165 = OpIMul %14 %55 %135
|
|
%166 = OpIAdd %14 %163 %165
|
|
%169 = OpIAdd %14 %232 %131
|
|
%171 = OpIAdd %14 %169 %135
|
|
%172 = OpAccessChain %15 %146 %171
|
|
%173 = OpLoad %14 %172
|
|
%174 = OpAccessChain %15 %146 %166
|
|
OpStore %174 %173
|
|
OpBranch %141
|
|
%141 = OpLabel
|
|
%176 = OpISub %14 %232 %77
|
|
OpStore %136 %176
|
|
OpBranch %138
|
|
%140 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%12 = OpFunction %2 None %3
|
|
%13 = OpLabel
|
|
%177 = OpVariable %15 Function
|
|
%181 = OpVariable %15 Function
|
|
%185 = OpVariable %15 Function
|
|
%195 = OpVariable %45 Function
|
|
%178 = OpAccessChain %23 %20 %22
|
|
%179 = OpLoad %17 %178
|
|
%180 = OpConvertFToS %14 %179
|
|
OpStore %177 %180
|
|
%182 = OpAccessChain %23 %20 %28
|
|
%183 = OpLoad %17 %182
|
|
%184 = OpConvertFToS %14 %183
|
|
OpStore %181 %184
|
|
OpStore %185 %180
|
|
OpBranch %187
|
|
%187 = OpLabel
|
|
%233 = OpPhi %14 %180 %13 %225 %190
|
|
OpLoopMerge %189 %190 None
|
|
OpBranch %191
|
|
%191 = OpLabel
|
|
%194 = OpSGreaterThanEqual %41 %233 %184
|
|
OpBranchConditional %194 %188 %189
|
|
%188 = OpLabel
|
|
%198 = OpIAdd %14 %233 %180
|
|
%200 = OpIAdd %14 %198 %184
|
|
%203 = OpIAdd %14 %233 %180
|
|
%205 = OpIMul %14 %55 %184
|
|
%206 = OpIAdd %14 %203 %205
|
|
%207 = OpAccessChain %15 %195 %206
|
|
%208 = OpLoad %14 %207
|
|
%209 = OpAccessChain %15 %195 %200
|
|
OpStore %209 %208
|
|
%212 = OpIAdd %14 %233 %180
|
|
%214 = OpIMul %14 %55 %184
|
|
%215 = OpIAdd %14 %212 %214
|
|
%218 = OpIAdd %14 %233 %180
|
|
%220 = OpIAdd %14 %218 %184
|
|
%221 = OpAccessChain %15 %195 %220
|
|
%222 = OpLoad %14 %221
|
|
%223 = OpAccessChain %15 %195 %215
|
|
OpStore %223 %222
|
|
OpBranch %190
|
|
%190 = OpLabel
|
|
%225 = OpISub %14 %233 %77
|
|
OpStore %185 %225
|
|
OpBranch %187
|
|
%189 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
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;
|
|
{
|
|
// Function a
|
|
const ir::Function* f = spvtest::GetFunction(module, 6);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
const ir::Instruction* stores[2];
|
|
int stores_found = 0;
|
|
for (const ir::Instruction& inst : *spvtest::GetBasicBlock(f, 35)) {
|
|
if (inst.opcode() == SpvOp::SpvOpStore) {
|
|
stores[stores_found] = &inst;
|
|
++stores_found;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
EXPECT_TRUE(stores[i]);
|
|
}
|
|
|
|
// 60 -> 61
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(60)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[0]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
|
|
// 74 -> 75
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(74)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[1]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
}
|
|
{
|
|
// Function b
|
|
const ir::Function* f = spvtest::GetFunction(module, 8);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
const ir::Instruction* stores[2];
|
|
int stores_found = 0;
|
|
for (const ir::Instruction& inst : *spvtest::GetBasicBlock(f, 90)) {
|
|
if (inst.opcode() == SpvOp::SpvOpStore) {
|
|
stores[stores_found] = &inst;
|
|
++stores_found;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
EXPECT_TRUE(stores[i]);
|
|
}
|
|
|
|
// 110 -> 111
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(110)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[0]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
|
|
// 124 -> 125
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(124)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[1]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
}
|
|
{
|
|
// Function c
|
|
const ir::Function* f = spvtest::GetFunction(module, 10);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
const ir::Instruction* stores[2];
|
|
int stores_found = 0;
|
|
for (const ir::Instruction& inst : *spvtest::GetBasicBlock(f, 139)) {
|
|
if (inst.opcode() == SpvOp::SpvOpStore) {
|
|
stores[stores_found] = &inst;
|
|
++stores_found;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
EXPECT_TRUE(stores[i]);
|
|
}
|
|
|
|
// 159 -> 160
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(159)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[0]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
|
|
// 173 -> 174
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(173)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[1]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
}
|
|
{
|
|
// Function d
|
|
const ir::Function* f = spvtest::GetFunction(module, 12);
|
|
ir::LoopDescriptor& ld = *context->GetLoopDescriptor(f);
|
|
ir::Loop* loop = &ld.GetLoopByIndex(0);
|
|
std::vector<const ir::Loop*> loops{loop};
|
|
opt::LoopDependenceAnalysis analysis{context.get(), loops};
|
|
|
|
const ir::Instruction* stores[2];
|
|
int stores_found = 0;
|
|
for (const ir::Instruction& inst : *spvtest::GetBasicBlock(f, 188)) {
|
|
if (inst.opcode() == SpvOp::SpvOpStore) {
|
|
stores[stores_found] = &inst;
|
|
++stores_found;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
EXPECT_TRUE(stores[i]);
|
|
}
|
|
|
|
// 208 -> 209
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(208)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[0]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
|
|
// 222 -> 223
|
|
{
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// load.
|
|
ir::Instruction* load_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(context->get_def_use_mgr()
|
|
->GetDef(222)
|
|
->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* load = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(load_var));
|
|
|
|
// Analyse and simplify the instruction behind the access chain of this
|
|
// store.
|
|
ir::Instruction* store_var = context->get_def_use_mgr()->GetDef(
|
|
context->get_def_use_mgr()
|
|
->GetDef(stores[1]->GetSingleWordInOperand(0))
|
|
->GetSingleWordInOperand(1));
|
|
opt::SENode* store = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->AnalyzeInstruction(store_var));
|
|
|
|
opt::SENode* delta = analysis.GetScalarEvolution()->SimplifyExpression(
|
|
analysis.GetScalarEvolution()->CreateSubtraction(load, store));
|
|
|
|
EXPECT_FALSE(analysis.IsProvablyOutsideOfLoopBounds(
|
|
loop, delta, store->AsSERecurrentNode()->GetCoefficient()));
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace
|