| // 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/reduce/operand_to_undef_reduction_opportunity_finder.h" |
| |
| #include "source/opt/build_module.h" |
| #include "source/reduce/reduction_opportunity.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 |