mirror of
https://github.com/KhronosGroup/SPIRV-Tools
synced 2024-10-19 19:40:13 +00:00
b7056e7e03
Given an instruction (that may use an OpPhi result from the same block as an input operand), try to clone the instruction into each predecessor block, replacing the input operand with the corresponding OpPhi input operand in each case, if necessary. Fixes #3458.
821 lines
26 KiB
C++
821 lines
26 KiB
C++
// Copyright (c) 2020 Vasyl Teliman
|
|
//
|
|
// 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/fuzz/transformation_propagate_instruction_up.h"
|
|
|
|
#include "test/fuzz/fuzz_test_util.h"
|
|
|
|
namespace spvtools {
|
|
namespace fuzz {
|
|
namespace {
|
|
|
|
TEST(TransformationPropagateInstructionUpTest, BasicTest) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeFloat 32
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstant %6 3.5
|
|
%11 = OpConstant %6 3.4000001
|
|
%12 = OpTypeBool
|
|
%17 = OpConstant %6 4
|
|
%20 = OpConstant %6 45
|
|
%27 = OpTypePointer Function %6
|
|
%4 = OpFunction %2 None %3
|
|
|
|
%5 = OpLabel
|
|
%26 = OpVariable %27 Function
|
|
%13 = OpFOrdEqual %12 %9 %11
|
|
OpSelectionMerge %15 None
|
|
OpBranchConditional %13 %14 %19
|
|
|
|
%14 = OpLabel
|
|
%18 = OpFMod %6 %9 %17
|
|
OpBranch %15
|
|
|
|
%19 = OpLabel
|
|
%22 = OpFAdd %6 %11 %20
|
|
OpBranch %15
|
|
|
|
%15 = OpLabel
|
|
%21 = OpPhi %6 %18 %14 %22 %19
|
|
%23 = OpFMul %6 %21 %21
|
|
%24 = OpFDiv %6 %21 %23
|
|
OpBranch %25
|
|
|
|
%25 = OpLabel
|
|
%28 = OpPhi %6 %20 %15
|
|
OpStore %26 %28
|
|
OpReturn
|
|
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_3;
|
|
const auto consumer = nullptr;
|
|
const auto context = BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
ASSERT_TRUE(IsValid(env, context.get()));
|
|
|
|
FactManager fact_manager;
|
|
spvtools::ValidatorOptions validator_options;
|
|
TransformationContext transformation_context(&fact_manager,
|
|
validator_options);
|
|
|
|
// |block_id| is invalid.
|
|
ASSERT_FALSE(TransformationPropagateInstructionUp(40, {{}}).IsApplicable(
|
|
context.get(), transformation_context));
|
|
ASSERT_FALSE(TransformationPropagateInstructionUp(26, {{}}).IsApplicable(
|
|
context.get(), transformation_context));
|
|
|
|
// |block_id| has no predecessors.
|
|
ASSERT_FALSE(TransformationPropagateInstructionUp(5, {{}}).IsApplicable(
|
|
context.get(), transformation_context));
|
|
|
|
// |block_id| has no valid instructions to propagate.
|
|
ASSERT_FALSE(TransformationPropagateInstructionUp(25, {{{15, 40}}})
|
|
.IsApplicable(context.get(), transformation_context));
|
|
|
|
// Not all predecessors have fresh ids.
|
|
ASSERT_FALSE(TransformationPropagateInstructionUp(15, {{{19, 40}, {40, 41}}})
|
|
.IsApplicable(context.get(), transformation_context));
|
|
|
|
// Not all ids are fresh.
|
|
ASSERT_FALSE(
|
|
TransformationPropagateInstructionUp(15, {{{19, 40}, {14, 14}, {40, 42}}})
|
|
.IsApplicable(context.get(), transformation_context));
|
|
ASSERT_FALSE(
|
|
TransformationPropagateInstructionUp(15, {{{19, 19}, {14, 40}, {40, 42}}})
|
|
.IsApplicable(context.get(), transformation_context));
|
|
|
|
// Fresh ids have duplicates.
|
|
ASSERT_FALSE(
|
|
TransformationPropagateInstructionUp(15, {{{19, 40}, {14, 40}, {19, 41}}})
|
|
.IsApplicable(context.get(), transformation_context));
|
|
|
|
// Valid transformations.
|
|
{
|
|
TransformationPropagateInstructionUp transformation(14, {{{5, 40}}});
|
|
ASSERT_TRUE(
|
|
transformation.IsApplicable(context.get(), transformation_context));
|
|
transformation.Apply(context.get(), &transformation_context);
|
|
ASSERT_TRUE(IsValid(env, context.get()));
|
|
}
|
|
{
|
|
TransformationPropagateInstructionUp transformation(19, {{{5, 41}}});
|
|
ASSERT_TRUE(
|
|
transformation.IsApplicable(context.get(), transformation_context));
|
|
transformation.Apply(context.get(), &transformation_context);
|
|
ASSERT_TRUE(IsValid(env, context.get()));
|
|
}
|
|
|
|
std::string after_transformation = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeFloat 32
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstant %6 3.5
|
|
%11 = OpConstant %6 3.4000001
|
|
%12 = OpTypeBool
|
|
%17 = OpConstant %6 4
|
|
%20 = OpConstant %6 45
|
|
%27 = OpTypePointer Function %6
|
|
%4 = OpFunction %2 None %3
|
|
|
|
%5 = OpLabel
|
|
%26 = OpVariable %27 Function
|
|
%13 = OpFOrdEqual %12 %9 %11
|
|
%40 = OpFMod %6 %9 %17 ; propagated from %14
|
|
%41 = OpFAdd %6 %11 %20 ; propagated from %19
|
|
OpSelectionMerge %15 None
|
|
OpBranchConditional %13 %14 %19
|
|
|
|
%14 = OpLabel
|
|
%18 = OpPhi %6 %40 %5 ; propagated into %5
|
|
OpBranch %15
|
|
|
|
%19 = OpLabel
|
|
%22 = OpPhi %6 %41 %5 ; propagated into %5
|
|
OpBranch %15
|
|
|
|
%15 = OpLabel
|
|
%21 = OpPhi %6 %18 %14 %22 %19
|
|
%23 = OpFMul %6 %21 %21
|
|
%24 = OpFDiv %6 %21 %23
|
|
OpBranch %25
|
|
|
|
%25 = OpLabel
|
|
%28 = OpPhi %6 %20 %15
|
|
OpStore %26 %28
|
|
OpReturn
|
|
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
ASSERT_TRUE(IsEqual(env, after_transformation, context.get()));
|
|
|
|
{
|
|
TransformationPropagateInstructionUp transformation(15,
|
|
{{{14, 43}, {19, 44}}});
|
|
ASSERT_TRUE(
|
|
transformation.IsApplicable(context.get(), transformation_context));
|
|
transformation.Apply(context.get(), &transformation_context);
|
|
ASSERT_TRUE(IsValid(env, context.get()));
|
|
}
|
|
|
|
after_transformation = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeFloat 32
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstant %6 3.5
|
|
%11 = OpConstant %6 3.4000001
|
|
%12 = OpTypeBool
|
|
%17 = OpConstant %6 4
|
|
%20 = OpConstant %6 45
|
|
%27 = OpTypePointer Function %6
|
|
%4 = OpFunction %2 None %3
|
|
|
|
%5 = OpLabel
|
|
%26 = OpVariable %27 Function
|
|
%13 = OpFOrdEqual %12 %9 %11
|
|
%40 = OpFMod %6 %9 %17
|
|
%41 = OpFAdd %6 %11 %20
|
|
OpSelectionMerge %15 None
|
|
OpBranchConditional %13 %14 %19
|
|
|
|
%14 = OpLabel
|
|
%18 = OpPhi %6 %40 %5
|
|
%43 = OpFMul %6 %18 %18 ; propagated from %15
|
|
OpBranch %15
|
|
|
|
%19 = OpLabel
|
|
%22 = OpPhi %6 %41 %5
|
|
%44 = OpFMul %6 %22 %22 ; propagated from %15
|
|
OpBranch %15
|
|
|
|
%15 = OpLabel
|
|
%23 = OpPhi %6 %43 %14 %44 %19 ; propagated into %14 and %19
|
|
%21 = OpPhi %6 %18 %14 %22 %19
|
|
%24 = OpFDiv %6 %21 %23
|
|
OpBranch %25
|
|
|
|
%25 = OpLabel
|
|
%28 = OpPhi %6 %20 %15
|
|
OpStore %26 %28
|
|
OpReturn
|
|
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
ASSERT_TRUE(IsEqual(env, after_transformation, context.get()));
|
|
|
|
{
|
|
TransformationPropagateInstructionUp transformation(15,
|
|
{{{14, 45}, {19, 46}}});
|
|
ASSERT_TRUE(
|
|
transformation.IsApplicable(context.get(), transformation_context));
|
|
transformation.Apply(context.get(), &transformation_context);
|
|
ASSERT_TRUE(IsValid(env, context.get()));
|
|
}
|
|
|
|
after_transformation = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeFloat 32
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstant %6 3.5
|
|
%11 = OpConstant %6 3.4000001
|
|
%12 = OpTypeBool
|
|
%17 = OpConstant %6 4
|
|
%20 = OpConstant %6 45
|
|
%27 = OpTypePointer Function %6
|
|
%4 = OpFunction %2 None %3
|
|
|
|
%5 = OpLabel
|
|
%26 = OpVariable %27 Function
|
|
%13 = OpFOrdEqual %12 %9 %11
|
|
%40 = OpFMod %6 %9 %17
|
|
%41 = OpFAdd %6 %11 %20
|
|
OpSelectionMerge %15 None
|
|
OpBranchConditional %13 %14 %19
|
|
|
|
%14 = OpLabel
|
|
%18 = OpPhi %6 %40 %5
|
|
%43 = OpFMul %6 %18 %18
|
|
%45 = OpFDiv %6 %18 %43 ; propagated from %15
|
|
OpBranch %15
|
|
|
|
%19 = OpLabel
|
|
%22 = OpPhi %6 %41 %5
|
|
%44 = OpFMul %6 %22 %22
|
|
%46 = OpFDiv %6 %22 %44 ; propagated from %15
|
|
OpBranch %15
|
|
|
|
%15 = OpLabel
|
|
%24 = OpPhi %6 %45 %14 %46 %19 ; propagated into %14 and %19
|
|
%23 = OpPhi %6 %43 %14 %44 %19
|
|
%21 = OpPhi %6 %18 %14 %22 %19
|
|
OpBranch %25
|
|
|
|
%25 = OpLabel
|
|
%28 = OpPhi %6 %20 %15
|
|
OpStore %26 %28
|
|
OpReturn
|
|
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
ASSERT_TRUE(IsEqual(env, after_transformation, context.get()));
|
|
}
|
|
|
|
TEST(TransformationPropagateInstructionUpTest, BlockDominatesPredecessor1) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeFloat 32
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstant %6 3.5
|
|
%11 = OpConstant %6 3.4000001
|
|
%12 = OpTypeBool
|
|
%17 = OpConstant %6 4
|
|
%20 = OpConstant %6 45
|
|
%4 = OpFunction %2 None %3
|
|
|
|
%5 = OpLabel
|
|
%13 = OpFOrdEqual %12 %9 %11
|
|
OpSelectionMerge %15 None
|
|
OpBranchConditional %13 %14 %19
|
|
|
|
%14 = OpLabel
|
|
%18 = OpFMod %6 %9 %17
|
|
OpBranch %15
|
|
|
|
%19 = OpLabel
|
|
%22 = OpFAdd %6 %11 %20
|
|
OpBranch %15
|
|
|
|
%15 = OpLabel ; dominates %26
|
|
%21 = OpPhi %6 %18 %14 %22 %19 %28 %26
|
|
%23 = OpFMul %6 %21 %21
|
|
%24 = OpFDiv %6 %21 %23
|
|
OpLoopMerge %27 %26 None
|
|
OpBranch %26
|
|
|
|
%26 = OpLabel
|
|
%28 = OpFAdd %6 %24 %23
|
|
OpBranch %15
|
|
|
|
%27 = OpLabel
|
|
OpReturn
|
|
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_3;
|
|
const auto consumer = nullptr;
|
|
const auto context = BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
ASSERT_TRUE(IsValid(env, context.get()));
|
|
|
|
FactManager fact_manager;
|
|
spvtools::ValidatorOptions validator_options;
|
|
TransformationContext transformation_context(&fact_manager,
|
|
validator_options);
|
|
|
|
TransformationPropagateInstructionUp transformation(
|
|
15, {{{14, 40}, {19, 41}, {26, 42}}});
|
|
ASSERT_TRUE(
|
|
transformation.IsApplicable(context.get(), transformation_context));
|
|
transformation.Apply(context.get(), &transformation_context);
|
|
ASSERT_TRUE(IsValid(env, context.get()));
|
|
|
|
std::string after_transformation = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeFloat 32
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstant %6 3.5
|
|
%11 = OpConstant %6 3.4000001
|
|
%12 = OpTypeBool
|
|
%17 = OpConstant %6 4
|
|
%20 = OpConstant %6 45
|
|
%4 = OpFunction %2 None %3
|
|
|
|
%5 = OpLabel
|
|
%13 = OpFOrdEqual %12 %9 %11
|
|
OpSelectionMerge %15 None
|
|
OpBranchConditional %13 %14 %19
|
|
|
|
%14 = OpLabel
|
|
%18 = OpFMod %6 %9 %17
|
|
%40 = OpFMul %6 %18 %18 ; propagated from %15
|
|
OpBranch %15
|
|
|
|
%19 = OpLabel
|
|
%22 = OpFAdd %6 %11 %20
|
|
%41 = OpFMul %6 %22 %22 ; propagated from %15
|
|
OpBranch %15
|
|
|
|
%15 = OpLabel
|
|
%23 = OpPhi %6 %40 %14 %41 %19 %42 %26 ; propagated into %14, %19, %26
|
|
%21 = OpPhi %6 %18 %14 %22 %19 %28 %26
|
|
%24 = OpFDiv %6 %21 %23
|
|
OpLoopMerge %27 %26 None
|
|
OpBranch %26
|
|
|
|
%26 = OpLabel
|
|
%28 = OpFAdd %6 %24 %23
|
|
%42 = OpFMul %6 %28 %28 ; propagated from %15
|
|
OpBranch %15
|
|
|
|
%27 = OpLabel
|
|
OpReturn
|
|
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
ASSERT_TRUE(IsEqual(env, after_transformation, context.get()));
|
|
}
|
|
|
|
TEST(TransformationPropagateInstructionUpTest, BlockDominatesPredecessor2) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeFloat 32
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstant %6 3.5
|
|
%11 = OpConstant %6 3.4000001
|
|
%12 = OpTypeBool
|
|
%17 = OpConstant %6 4
|
|
%20 = OpConstant %6 45
|
|
%4 = OpFunction %2 None %3
|
|
|
|
%5 = OpLabel
|
|
%13 = OpFOrdEqual %12 %9 %11
|
|
OpSelectionMerge %15 None
|
|
OpBranchConditional %13 %14 %19
|
|
|
|
%14 = OpLabel
|
|
%18 = OpFMod %6 %9 %17
|
|
OpBranch %15
|
|
|
|
%19 = OpLabel
|
|
%22 = OpFAdd %6 %11 %20
|
|
OpBranch %15
|
|
|
|
%15 = OpLabel ; doesn't dominate %26
|
|
%21 = OpPhi %6 %18 %14 %22 %19 %20 %26
|
|
%23 = OpFMul %6 %21 %21
|
|
%24 = OpFDiv %6 %21 %23
|
|
OpLoopMerge %27 %26 None
|
|
OpBranch %27
|
|
|
|
%26 = OpLabel
|
|
OpBranch %15
|
|
|
|
%27 = OpLabel
|
|
OpReturn
|
|
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_3;
|
|
const auto consumer = nullptr;
|
|
const auto context = BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
ASSERT_TRUE(IsValid(env, context.get()));
|
|
|
|
FactManager fact_manager;
|
|
spvtools::ValidatorOptions validator_options;
|
|
TransformationContext transformation_context(&fact_manager,
|
|
validator_options);
|
|
|
|
TransformationPropagateInstructionUp transformation(
|
|
15, {{{14, 40}, {19, 41}, {26, 42}}});
|
|
ASSERT_TRUE(
|
|
transformation.IsApplicable(context.get(), transformation_context));
|
|
transformation.Apply(context.get(), &transformation_context);
|
|
ASSERT_TRUE(IsValid(env, context.get()));
|
|
|
|
std::string after_transformation = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeFloat 32
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstant %6 3.5
|
|
%11 = OpConstant %6 3.4000001
|
|
%12 = OpTypeBool
|
|
%17 = OpConstant %6 4
|
|
%20 = OpConstant %6 45
|
|
%4 = OpFunction %2 None %3
|
|
|
|
%5 = OpLabel
|
|
%13 = OpFOrdEqual %12 %9 %11
|
|
OpSelectionMerge %15 None
|
|
OpBranchConditional %13 %14 %19
|
|
|
|
%14 = OpLabel
|
|
%18 = OpFMod %6 %9 %17
|
|
%40 = OpFMul %6 %18 %18 ; propagated from %15
|
|
OpBranch %15
|
|
|
|
%19 = OpLabel
|
|
%22 = OpFAdd %6 %11 %20
|
|
%41 = OpFMul %6 %22 %22 ; propagated from %15
|
|
OpBranch %15
|
|
|
|
%15 = OpLabel
|
|
%23 = OpPhi %6 %40 %14 %41 %19 %42 %26 ; propagated into %14, %19, %26
|
|
%21 = OpPhi %6 %18 %14 %22 %19 %20 %26
|
|
%24 = OpFDiv %6 %21 %23
|
|
OpLoopMerge %27 %26 None
|
|
OpBranch %27
|
|
|
|
%26 = OpLabel
|
|
%42 = OpFMul %6 %20 %20 ; propagated from %15
|
|
OpBranch %15
|
|
|
|
%27 = OpLabel
|
|
OpReturn
|
|
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
ASSERT_TRUE(IsEqual(env, after_transformation, context.get()));
|
|
}
|
|
|
|
TEST(TransformationPropagateInstructionUpTest, BlockDominatesPredecessor3) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeFloat 32
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstant %6 3.5
|
|
%11 = OpConstant %6 3.4000001
|
|
%12 = OpTypeBool
|
|
%17 = OpConstant %6 4
|
|
%20 = OpConstant %6 45
|
|
%4 = OpFunction %2 None %3
|
|
|
|
%5 = OpLabel
|
|
%13 = OpFOrdEqual %12 %9 %11
|
|
OpSelectionMerge %15 None
|
|
OpBranchConditional %13 %14 %19
|
|
|
|
%14 = OpLabel
|
|
%18 = OpFMod %6 %9 %17
|
|
OpBranch %15
|
|
|
|
%19 = OpLabel
|
|
%22 = OpFAdd %6 %11 %20
|
|
OpBranch %15
|
|
|
|
%15 = OpLabel ; branches to itself
|
|
%21 = OpPhi %6 %18 %14 %22 %19 %24 %15
|
|
%23 = OpFMul %6 %21 %21
|
|
%24 = OpFDiv %6 %21 %23
|
|
OpLoopMerge %27 %15 None
|
|
OpBranch %15
|
|
|
|
%27 = OpLabel
|
|
OpReturn
|
|
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_3;
|
|
const auto consumer = nullptr;
|
|
const auto context = BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
ASSERT_TRUE(IsValid(env, context.get()));
|
|
|
|
FactManager fact_manager;
|
|
spvtools::ValidatorOptions validator_options;
|
|
TransformationContext transformation_context(&fact_manager,
|
|
validator_options);
|
|
|
|
TransformationPropagateInstructionUp transformation(
|
|
15, {{{14, 40}, {19, 41}, {15, 42}}});
|
|
ASSERT_TRUE(
|
|
transformation.IsApplicable(context.get(), transformation_context));
|
|
transformation.Apply(context.get(), &transformation_context);
|
|
ASSERT_TRUE(IsValid(env, context.get()));
|
|
|
|
std::string after_transformation = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeFloat 32
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstant %6 3.5
|
|
%11 = OpConstant %6 3.4000001
|
|
%12 = OpTypeBool
|
|
%17 = OpConstant %6 4
|
|
%20 = OpConstant %6 45
|
|
%4 = OpFunction %2 None %3
|
|
|
|
%5 = OpLabel
|
|
%13 = OpFOrdEqual %12 %9 %11
|
|
OpSelectionMerge %15 None
|
|
OpBranchConditional %13 %14 %19
|
|
|
|
%14 = OpLabel
|
|
%18 = OpFMod %6 %9 %17
|
|
%40 = OpFMul %6 %18 %18 ; propagated from %15
|
|
OpBranch %15
|
|
|
|
%19 = OpLabel
|
|
%22 = OpFAdd %6 %11 %20
|
|
%41 = OpFMul %6 %22 %22 ; propagated from %15
|
|
OpBranch %15
|
|
|
|
%15 = OpLabel
|
|
%23 = OpPhi %6 %40 %14 %41 %19 %42 %15 ; propagated into %14, %19, %15
|
|
%21 = OpPhi %6 %18 %14 %22 %19 %24 %15
|
|
%24 = OpFDiv %6 %21 %23
|
|
%42 = OpFMul %6 %24 %24 ; propagated from %15
|
|
OpLoopMerge %27 %15 None
|
|
OpBranch %15
|
|
|
|
%27 = OpLabel
|
|
OpReturn
|
|
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
ASSERT_TRUE(IsEqual(env, after_transformation, context.get()));
|
|
}
|
|
|
|
TEST(TransformationPropagateInstructionUpTest,
|
|
HandlesVariablePointersCapability) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeFloat 32
|
|
%11 = OpConstant %6 23
|
|
%7 = OpTypePointer Function %6
|
|
%4 = OpFunction %2 None %3
|
|
|
|
%5 = OpLabel
|
|
%8 = OpVariable %7 Function
|
|
OpBranch %9
|
|
|
|
%9 = OpLabel
|
|
%10 = OpCopyObject %7 %8
|
|
OpStore %10 %11
|
|
OpReturn
|
|
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_3;
|
|
const auto consumer = nullptr;
|
|
const auto context = BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
ASSERT_TRUE(IsValid(env, context.get()));
|
|
|
|
FactManager fact_manager;
|
|
spvtools::ValidatorOptions validator_options;
|
|
TransformationContext transformation_context(&fact_manager,
|
|
validator_options);
|
|
|
|
// Required capabilities haven't yet been specified.
|
|
TransformationPropagateInstructionUp transformation(9, {{{5, 40}}});
|
|
ASSERT_FALSE(
|
|
transformation.IsApplicable(context.get(), transformation_context));
|
|
|
|
context->AddCapability(SpvCapabilityVariablePointers);
|
|
|
|
ASSERT_TRUE(
|
|
transformation.IsApplicable(context.get(), transformation_context));
|
|
transformation.Apply(context.get(), &transformation_context);
|
|
ASSERT_TRUE(IsValid(env, context.get()));
|
|
|
|
std::string after_transformation = R"(
|
|
OpCapability Shader
|
|
OpCapability VariablePointers
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeFloat 32
|
|
%11 = OpConstant %6 23
|
|
%7 = OpTypePointer Function %6
|
|
%4 = OpFunction %2 None %3
|
|
|
|
%5 = OpLabel
|
|
%8 = OpVariable %7 Function
|
|
%40 = OpCopyObject %7 %8 ; propagated from %9
|
|
OpBranch %9
|
|
|
|
%9 = OpLabel
|
|
%10 = OpPhi %7 %40 %5 ; propagated into %5
|
|
OpStore %10 %11
|
|
OpReturn
|
|
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
ASSERT_TRUE(IsEqual(env, after_transformation, context.get()));
|
|
}
|
|
|
|
TEST(TransformationPropagateInstructionUpTest,
|
|
HandlesVariablePointersStorageBufferCapability) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeFloat 32
|
|
%11 = OpConstant %6 23
|
|
%7 = OpTypePointer Function %6
|
|
%4 = OpFunction %2 None %3
|
|
|
|
%5 = OpLabel
|
|
%8 = OpVariable %7 Function
|
|
OpBranch %9
|
|
|
|
%9 = OpLabel
|
|
%10 = OpCopyObject %7 %8
|
|
OpStore %10 %11
|
|
OpReturn
|
|
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_3;
|
|
const auto consumer = nullptr;
|
|
const auto context = BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
ASSERT_TRUE(IsValid(env, context.get()));
|
|
|
|
FactManager fact_manager;
|
|
spvtools::ValidatorOptions validator_options;
|
|
TransformationContext transformation_context(&fact_manager,
|
|
validator_options);
|
|
|
|
// Required capabilities haven't yet been specified
|
|
TransformationPropagateInstructionUp transformation(9, {{{5, 40}}});
|
|
ASSERT_FALSE(
|
|
transformation.IsApplicable(context.get(), transformation_context));
|
|
|
|
context->AddCapability(SpvCapabilityVariablePointersStorageBuffer);
|
|
|
|
ASSERT_TRUE(
|
|
transformation.IsApplicable(context.get(), transformation_context));
|
|
transformation.Apply(context.get(), &transformation_context);
|
|
ASSERT_TRUE(IsValid(env, context.get()));
|
|
|
|
std::string after_transformation = R"(
|
|
OpCapability Shader
|
|
OpCapability VariablePointersStorageBuffer
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main"
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeFloat 32
|
|
%11 = OpConstant %6 23
|
|
%7 = OpTypePointer Function %6
|
|
%4 = OpFunction %2 None %3
|
|
|
|
%5 = OpLabel
|
|
%8 = OpVariable %7 Function
|
|
%40 = OpCopyObject %7 %8 ; propagated from %9
|
|
OpBranch %9
|
|
|
|
%9 = OpLabel
|
|
%10 = OpPhi %7 %40 %5 ; propagated into %5
|
|
OpStore %10 %11
|
|
OpReturn
|
|
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
ASSERT_TRUE(IsEqual(env, after_transformation, context.get()));
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace fuzz
|
|
} // namespace spvtools
|