| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328 |
- // Copyright (c) 2021 Alastair F. Donaldson
- //
- // 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/available_instructions.h"
- #include "gtest/gtest.h"
- #include "source/fuzz/fuzzer_util.h"
- #include "test/fuzz/fuzz_test_util.h"
- namespace spvtools {
- namespace fuzz {
- namespace {
- TEST(AvailableInstructionsTest, BasicTest) {
- std::string shader = R"(
- OpCapability Shader
- %1 = OpExtInstImport "GLSL.std.450"
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %4 "main"
- OpExecutionMode %4 OriginUpperLeft
- OpSource ESSL 320
- %2 = OpTypeVoid
- %3 = OpTypeFunction %2
- %6 = OpTypeInt 32 1
- %7 = OpTypePointer Function %6
- %8 = OpTypeFloat 32
- %9 = OpTypePointer Function %8
- %10 = OpTypeFunction %6 %7 %9
- %15 = OpTypeVector %8 2
- %16 = OpTypePointer Private %15
- %17 = OpVariable %16 Private
- %18 = OpConstant %8 1
- %19 = OpConstant %8 2
- %20 = OpConstantComposite %15 %18 %19
- %21 = OpTypeVector %8 4
- %22 = OpTypePointer Private %21
- %23 = OpVariable %22 Private
- %24 = OpConstant %8 10
- %25 = OpConstant %8 20
- %26 = OpConstant %8 30
- %27 = OpConstant %8 40
- %28 = OpConstantComposite %21 %24 %25 %26 %27
- %31 = OpTypeInt 32 0
- %32 = OpConstant %31 0
- %33 = OpTypePointer Private %8
- %41 = OpTypeBool
- %46 = OpConstant %6 1
- %54 = OpConstant %6 10
- %57 = OpConstant %31 3
- %61 = OpConstant %6 0
- %66 = OpConstant %6 3
- %4 = OpFunction %2 None %3
- %5 = OpLabel
- %55 = OpVariable %7 Function
- %56 = OpVariable %9 Function
- %65 = OpVariable %7 Function
- %68 = OpVariable %7 Function
- OpStore %17 %20
- OpStore %23 %28
- OpStore %55 %54
- %58 = OpAccessChain %33 %23 %57
- %59 = OpLoad %8 %58
- OpStore %56 %59
- %60 = OpFunctionCall %6 %13 %55 %56
- %100 = OpCopyObject %21 %28
- %62 = OpSGreaterThan %41 %60 %61
- OpSelectionMerge %64 None
- OpBranchConditional %62 %63 %67
- %63 = OpLabel
- OpStore %65 %66
- %101 = OpCopyObject %21 %28
- OpBranch %64
- %67 = OpLabel
- OpStore %68 %61
- OpBranch %69
- %69 = OpLabel
- OpLoopMerge %71 %72 None
- OpBranch %73
- %73 = OpLabel
- %74 = OpLoad %6 %68
- %75 = OpSLessThan %41 %74 %54
- OpBranchConditional %75 %70 %71
- %70 = OpLabel
- %76 = OpLoad %6 %65
- %77 = OpIAdd %6 %76 %46
- OpStore %65 %77
- OpBranch %72
- %72 = OpLabel
- %78 = OpLoad %6 %68
- %79 = OpIAdd %6 %78 %46
- OpStore %68 %79
- OpBranch %69
- %71 = OpLabel
- %102 = OpCopyObject %21 %28
- OpBranch %64
- %64 = OpLabel
- OpReturn
- OpFunctionEnd
- %13 = OpFunction %6 None %10
- %11 = OpFunctionParameter %7
- %12 = OpFunctionParameter %9
- %14 = OpLabel
- %29 = OpVariable %7 Function
- %30 = OpLoad %6 %11
- %34 = OpAccessChain %33 %17 %32
- %35 = OpLoad %8 %34
- %36 = OpConvertFToS %6 %35
- %37 = OpIAdd %6 %30 %36
- OpStore %29 %37
- %38 = OpLoad %6 %11
- %39 = OpLoad %8 %12
- %40 = OpConvertFToS %6 %39
- %42 = OpSLessThan %41 %38 %40
- %103 = OpCopyObject %21 %28
- OpSelectionMerge %44 None
- OpBranchConditional %42 %43 %48
- %43 = OpLabel
- %45 = OpLoad %6 %29
- %47 = OpIAdd %6 %45 %46
- OpStore %29 %47
- OpBranch %44
- %48 = OpLabel
- %49 = OpLoad %6 %29
- %50 = OpISub %6 %49 %46
- OpStore %29 %50
- OpBranch %44
- %44 = OpLabel
- %51 = OpLoad %6 %29
- OpReturnValue %51
- OpFunctionEnd
- )";
- const auto env = SPV_ENV_UNIVERSAL_1_3;
- const auto consumer = nullptr;
- const auto context = BuildModule(env, consumer, shader, kFuzzAssembleOption);
- spvtools::ValidatorOptions validator_options;
- ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
- kConsoleMessageConsumer));
- opt::Instruction* i1 = context->get_def_use_mgr()->GetDef(55);
- opt::Instruction* i2 = context->get_def_use_mgr()->GetDef(101);
- opt::Instruction* i3 = &*context->cfg()->block(67)->begin();
- opt::Instruction* i4 = context->get_def_use_mgr()->GetDef(74);
- opt::Instruction* i5 = context->get_def_use_mgr()->GetDef(102);
- opt::Instruction* i6 = context->get_def_use_mgr()->GetDef(30);
- opt::Instruction* i7 = context->get_def_use_mgr()->GetDef(47);
- opt::Instruction* i8 = context->get_def_use_mgr()->GetDef(50);
- opt::Instruction* i9 = context->get_def_use_mgr()->GetDef(51);
- {
- AvailableInstructions no_instructions(
- context.get(),
- [](opt::IRContext*, opt::Instruction*) -> bool { return false; });
- for (auto i : {i1, i2, i3, i4, i5, i6, i7, i8, i9}) {
- auto available = no_instructions.GetAvailableBeforeInstruction(i);
- ASSERT_EQ(0, available.size());
- ASSERT_TRUE(available.empty());
- }
- }
- {
- AvailableInstructions all_instructions(
- context.get(),
- [](opt::IRContext*, opt::Instruction*) -> bool { return true; });
- {
- auto available = all_instructions.GetAvailableBeforeInstruction(i1);
- ASSERT_FALSE(available.empty());
- ASSERT_EQ(30, available.size());
- ASSERT_EQ(spv::Op::OpTypeVoid, available[0]->opcode());
- ASSERT_EQ(spv::Op::OpVariable, available[15]->opcode());
- }
- {
- auto available = all_instructions.GetAvailableBeforeInstruction(i2);
- ASSERT_FALSE(available.empty());
- ASSERT_EQ(46, available.size());
- ASSERT_EQ(spv::Op::OpTypeVoid, available[0]->opcode());
- ASSERT_EQ(spv::Op::OpTypePointer, available[3]->opcode());
- ASSERT_EQ(spv::Op::OpVariable, available[15]->opcode());
- ASSERT_EQ(spv::Op::OpFunctionCall, available[40]->opcode());
- ASSERT_EQ(spv::Op::OpStore, available[45]->opcode());
- }
- {
- auto available = all_instructions.GetAvailableBeforeInstruction(i3);
- ASSERT_FALSE(available.empty());
- ASSERT_EQ(45, available.size());
- ASSERT_EQ(spv::Op::OpTypeVoid, available[0]->opcode());
- ASSERT_EQ(spv::Op::OpTypePointer, available[3]->opcode());
- ASSERT_EQ(spv::Op::OpVariable, available[15]->opcode());
- ASSERT_EQ(spv::Op::OpFunctionCall, available[40]->opcode());
- ASSERT_EQ(spv::Op::OpBranchConditional, available[44]->opcode());
- }
- {
- auto available = all_instructions.GetAvailableBeforeInstruction(i6);
- ASSERT_FALSE(available.empty());
- ASSERT_EQ(33, available.size());
- ASSERT_EQ(spv::Op::OpTypeVoid, available[0]->opcode());
- ASSERT_EQ(spv::Op::OpTypeFloat, available[4]->opcode());
- ASSERT_EQ(spv::Op::OpTypePointer, available[8]->opcode());
- ASSERT_EQ(spv::Op::OpConstantComposite, available[12]->opcode());
- ASSERT_EQ(spv::Op::OpConstant, available[16]->opcode());
- ASSERT_EQ(spv::Op::OpFunctionParameter, available[30]->opcode());
- ASSERT_EQ(spv::Op::OpFunctionParameter, available[31]->opcode());
- ASSERT_EQ(spv::Op::OpVariable, available[32]->opcode());
- }
- }
- {
- AvailableInstructions vector_instructions(
- context.get(),
- [](opt::IRContext* ir_context, opt::Instruction* inst) -> bool {
- return inst->type_id() != 0 && ir_context->get_type_mgr()
- ->GetType(inst->type_id())
- ->AsVector() != nullptr;
- });
- {
- auto available = vector_instructions.GetAvailableBeforeInstruction(i4);
- ASSERT_FALSE(available.empty());
- ASSERT_EQ(3, available.size());
- ASSERT_EQ(spv::Op::OpConstantComposite, available[0]->opcode());
- ASSERT_EQ(spv::Op::OpConstantComposite, available[1]->opcode());
- ASSERT_EQ(spv::Op::OpCopyObject, available[2]->opcode());
- }
- {
- auto available = vector_instructions.GetAvailableBeforeInstruction(i5);
- ASSERT_FALSE(available.empty());
- ASSERT_EQ(3, available.size());
- ASSERT_EQ(spv::Op::OpConstantComposite, available[0]->opcode());
- ASSERT_EQ(spv::Op::OpConstantComposite, available[1]->opcode());
- ASSERT_EQ(spv::Op::OpCopyObject, available[2]->opcode());
- }
- {
- auto available = vector_instructions.GetAvailableBeforeInstruction(i6);
- ASSERT_FALSE(available.empty());
- ASSERT_EQ(2, available.size());
- ASSERT_EQ(spv::Op::OpConstantComposite, available[0]->opcode());
- ASSERT_EQ(spv::Op::OpConstantComposite, available[1]->opcode());
- }
- }
- {
- AvailableInstructions integer_add_instructions(
- context.get(), [](opt::IRContext*, opt::Instruction* inst) -> bool {
- return inst->opcode() == spv::Op::OpIAdd;
- });
- {
- auto available =
- integer_add_instructions.GetAvailableBeforeInstruction(i7);
- ASSERT_FALSE(available.empty());
- ASSERT_EQ(1, available.size());
- ASSERT_EQ(spv::Op::OpIAdd, available[0]->opcode());
- }
- {
- auto available =
- integer_add_instructions.GetAvailableBeforeInstruction(i8);
- ASSERT_FALSE(available.empty());
- ASSERT_EQ(1, available.size());
- ASSERT_EQ(spv::Op::OpIAdd, available[0]->opcode());
- }
- {
- auto available =
- integer_add_instructions.GetAvailableBeforeInstruction(i9);
- ASSERT_FALSE(available.empty());
- ASSERT_EQ(1, available.size());
- ASSERT_EQ(spv::Op::OpIAdd, available[0]->opcode());
- }
- }
- }
- TEST(AvailableInstructionsTest, UnreachableBlock) {
- std::string shader = R"(
- OpCapability Shader
- %1 = OpExtInstImport "GLSL.std.450"
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %4 "main"
- OpExecutionMode %4 OriginUpperLeft
- OpSource ESSL 320
- OpName %4 "main"
- OpName %8 "x"
- %2 = OpTypeVoid
- %3 = OpTypeFunction %2
- %6 = OpTypeInt 32 1
- %7 = OpTypePointer Function %6
- %9 = OpConstant %6 2
- %4 = OpFunction %2 None %3
- %5 = OpLabel
- %8 = OpVariable %7 Function
- OpStore %8 %9
- %12 = OpLoad %6 %8
- OpReturn
- %10 = OpLabel
- %11 = OpLoad %6 %8
- OpReturn
- OpFunctionEnd
- )";
- const auto env = SPV_ENV_UNIVERSAL_1_3;
- const auto consumer = nullptr;
- const auto context = BuildModule(env, consumer, shader, kFuzzAssembleOption);
- spvtools::ValidatorOptions validator_options;
- ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
- kConsoleMessageConsumer));
- AvailableInstructions all_instructions(
- context.get(),
- [](opt::IRContext*, opt::Instruction*) -> bool { return true; });
- ASSERT_EQ(7, all_instructions
- .GetAvailableBeforeInstruction(
- context->get_def_use_mgr()->GetDef(12))
- .size());
- #ifndef NDEBUG
- ASSERT_DEATH(all_instructions.GetAvailableBeforeInstruction(
- context->get_def_use_mgr()->GetDef(11)),
- "Availability can only be queried for reachable instructions.");
- #endif
- }
- } // namespace
- } // namespace fuzz
- } // namespace spvtools
|