SPIRV-Tools/test/reduce/operand_to_undef_test.cpp

229 lines
7.9 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 "source/opt/build_module.h"
#include "source/reduce/operand_to_undef_reduction_opportunity_finder.h"
#include "test/reduce/reduce_test_util.h"
namespace spvtools {
namespace reduce {
namespace {
TEST(OperandToUndefReductionPassTest, BasicCheck) {
// The following shader has 10 opportunities for replacing with undef.
// #version 310 es
//
// precision highp float;
//
// layout(location=0) out vec4 _GLF_color;
//
// layout(set = 0, binding = 0) uniform buf0 {
// vec2 uniform1;
// };
//
// void main()
// {
// _GLF_color =
// vec4( // opportunity
// uniform1.x / 2.0, // opportunity x2 (2.0 is const)
// uniform1.y / uniform1.x, // opportunity x3
// uniform1.x + uniform1.x, // opportunity x3
// uniform1.y); // opportunity
// }
std::string original = R"(
OpCapability Shader
%1 = OpExtInstImport "GLSL.std.450"
OpMemoryModel Logical GLSL450
OpEntryPoint Fragment %4 "main" %9
OpExecutionMode %4 OriginUpperLeft
OpSource ESSL 310
OpName %4 "main"
OpName %9 "_GLF_color"
OpName %11 "buf0"
OpMemberName %11 0 "uniform1"
OpName %13 ""
OpDecorate %9 Location 0
OpMemberDecorate %11 0 Offset 0
OpDecorate %11 Block
OpDecorate %13 DescriptorSet 0
OpDecorate %13 Binding 0
%2 = OpTypeVoid
%3 = OpTypeFunction %2
%6 = OpTypeFloat 32
%7 = OpTypeVector %6 4
%8 = OpTypePointer Output %7
%9 = OpVariable %8 Output
%10 = OpTypeVector %6 2
%11 = OpTypeStruct %10
%12 = OpTypePointer Uniform %11
%13 = OpVariable %12 Uniform
%14 = OpTypeInt 32 1
%15 = OpConstant %14 0
%16 = OpTypeInt 32 0
%17 = OpConstant %16 0
%18 = OpTypePointer Uniform %6
%21 = OpConstant %6 2
%23 = OpConstant %16 1
%4 = OpFunction %2 None %3
%5 = OpLabel
%19 = OpAccessChain %18 %13 %15 %17
%20 = OpLoad %6 %19
%22 = OpFDiv %6 %20 %21 ; opportunity %20 (%21 is const)
%24 = OpAccessChain %18 %13 %15 %23
%25 = OpLoad %6 %24
%26 = OpAccessChain %18 %13 %15 %17
%27 = OpLoad %6 %26
%28 = OpFDiv %6 %25 %27 ; opportunity %25 %27
%29 = OpAccessChain %18 %13 %15 %17
%30 = OpLoad %6 %29
%31 = OpAccessChain %18 %13 %15 %17
%32 = OpLoad %6 %31
%33 = OpFAdd %6 %30 %32 ; opportunity %30 %32
%34 = OpAccessChain %18 %13 %15 %23
%35 = OpLoad %6 %34
%36 = OpCompositeConstruct %7 %22 %28 %33 %35 ; opportunity %22 %28 %33 %35
OpStore %9 %36 ; opportunity %36
OpReturn
OpFunctionEnd
)";
// This is the same as original, except where noted.
std::string expected = R"(
OpCapability Shader
%1 = OpExtInstImport "GLSL.std.450"
OpMemoryModel Logical GLSL450
OpEntryPoint Fragment %4 "main" %9
OpExecutionMode %4 OriginUpperLeft
OpSource ESSL 310
OpName %4 "main"
OpName %9 "_GLF_color"
OpName %11 "buf0"
OpMemberName %11 0 "uniform1"
OpName %13 ""
OpDecorate %9 Location 0
OpMemberDecorate %11 0 Offset 0
OpDecorate %11 Block
OpDecorate %13 DescriptorSet 0
OpDecorate %13 Binding 0
%2 = OpTypeVoid
%3 = OpTypeFunction %2
%6 = OpTypeFloat 32
%7 = OpTypeVector %6 4
%8 = OpTypePointer Output %7
%9 = OpVariable %8 Output
%10 = OpTypeVector %6 2
%11 = OpTypeStruct %10
%12 = OpTypePointer Uniform %11
%13 = OpVariable %12 Uniform
%14 = OpTypeInt 32 1
%15 = OpConstant %14 0
%16 = OpTypeInt 32 0
%17 = OpConstant %16 0
%18 = OpTypePointer Uniform %6
%21 = OpConstant %6 2
%23 = OpConstant %16 1
%37 = OpUndef %6 ; Added undef float as %37
%4 = OpFunction %2 None %3
%5 = OpLabel
%19 = OpAccessChain %18 %13 %15 %17
%20 = OpLoad %6 %19
%22 = OpFDiv %6 %37 %21 ; Replaced with %37
%24 = OpAccessChain %18 %13 %15 %23
%25 = OpLoad %6 %24
%26 = OpAccessChain %18 %13 %15 %17
%27 = OpLoad %6 %26
%28 = OpFDiv %6 %37 %37 ; Replaced with %37 twice
%29 = OpAccessChain %18 %13 %15 %17
%30 = OpLoad %6 %29
%31 = OpAccessChain %18 %13 %15 %17
%32 = OpLoad %6 %31
%33 = OpFAdd %6 %30 %32
%34 = OpAccessChain %18 %13 %15 %23
%35 = OpLoad %6 %34
%36 = OpCompositeConstruct %7 %22 %28 %33 %35
OpStore %9 %36
OpReturn
OpFunctionEnd
)";
const auto env = SPV_ENV_UNIVERSAL_1_3;
const auto consumer = nullptr;
const auto context =
BuildModule(env, consumer, original, kReduceAssembleOption);
const auto ops =
OperandToUndefReductionOpportunityFinder().GetAvailableOpportunities(
context.get());
ASSERT_EQ(10, ops.size());
// Apply first three opportunities.
ASSERT_TRUE(ops[0]->PreconditionHolds());
ops[0]->TryToApply();
ASSERT_TRUE(ops[1]->PreconditionHolds());
ops[1]->TryToApply();
ASSERT_TRUE(ops[2]->PreconditionHolds());
ops[2]->TryToApply();
CheckEqual(env, expected, context.get());
}
TEST(OperandToUndefReductionPassTest, WithCalledFunction) {
// The following shader has no opportunities.
// Most importantly, the noted function operand is not changed.
std::string shader = R"(
OpCapability Shader
%1 = OpExtInstImport "GLSL.std.450"
OpMemoryModel Logical GLSL450
OpEntryPoint Fragment %4 "main" %10 %12
OpExecutionMode %4 OriginUpperLeft
OpSource ESSL 310
%2 = OpTypeVoid
%3 = OpTypeFunction %2
%6 = OpTypeFloat 32
%7 = OpTypeVector %6 4
%8 = OpTypeFunction %7
%9 = OpTypePointer Output %7
%10 = OpVariable %9 Output
%11 = OpTypePointer Input %7
%12 = OpVariable %11 Input
%13 = OpConstant %6 0
%14 = OpConstantComposite %7 %13 %13 %13 %13
%4 = OpFunction %2 None %3
%5 = OpLabel
%15 = OpFunctionCall %7 %16 ; do not replace %16 with undef
OpReturn
OpFunctionEnd
%16 = OpFunction %7 None %8
%17 = OpLabel
OpReturnValue %14
OpFunctionEnd
)";
const auto env = SPV_ENV_UNIVERSAL_1_3;
const auto consumer = nullptr;
const auto context =
BuildModule(env, consumer, shader, kReduceAssembleOption);
const auto ops =
OperandToUndefReductionOpportunityFinder().GetAvailableOpportunities(
context.get());
ASSERT_EQ(0, ops.size());
}
} // namespace
} // namespace reduce
} // namespace spvtools