mirror of
https://github.com/KhronosGroup/SPIRV-Tools
synced 2025-01-09 16:20:06 +00:00
65ecfd1093
Found via `codespell -q 3 -L fo,lod,parm
1808 lines
70 KiB
C++
1808 lines
70 KiB
C++
// Copyright (c) 2021 Shiyu Liu
|
|
//
|
|
// 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 "gtest/gtest.h"
|
|
#include "source/fuzz/fuzzer_util.h"
|
|
#include "test/fuzz/fuzz_test_util.h"
|
|
|
|
namespace spvtools {
|
|
namespace fuzz {
|
|
namespace {
|
|
|
|
TEST(FuzzerUtilMaybeFindBlockTest, 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
|
|
OpDecorate %8 RelaxedPrecision
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeInt 32 1
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstant %6 1
|
|
%10 = OpConstant %6 2
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%8 = OpVariable %7 Function
|
|
OpBranch %11
|
|
%11 = OpLabel
|
|
OpStore %8 %9
|
|
OpBranch %12
|
|
%12 = OpLabel
|
|
OpStore %8 %10
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_4;
|
|
const auto consumer = nullptr;
|
|
const std::unique_ptr<opt::IRContext> context =
|
|
BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
spvtools::ValidatorOptions validator_options;
|
|
ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
|
|
kConsoleMessageConsumer));
|
|
|
|
// Only blocks with id 11 and 12 can be found.
|
|
// Should return nullptr when id is not a label or id was not found.
|
|
uint32_t block_id1 = 11;
|
|
uint32_t block_id2 = 12;
|
|
uint32_t block_id3 = 13;
|
|
uint32_t block_id4 = 8;
|
|
|
|
opt::IRContext* ir_context = context.get();
|
|
// Block with id 11 should be found.
|
|
ASSERT_TRUE(fuzzerutil::MaybeFindBlock(ir_context, block_id1) != nullptr);
|
|
// Block with id 12 should be found.
|
|
ASSERT_TRUE(fuzzerutil::MaybeFindBlock(ir_context, block_id2) != nullptr);
|
|
// Block with id 13 cannot be found.
|
|
ASSERT_FALSE(fuzzerutil::MaybeFindBlock(ir_context, block_id3) != nullptr);
|
|
// Block with id 8 exists but don't not of type OpLabel.
|
|
ASSERT_FALSE(fuzzerutil::MaybeFindBlock(ir_context, block_id4) != nullptr);
|
|
}
|
|
|
|
TEST(FuzzerutilTest, FuzzerUtilMaybeGetBoolConstantTest) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %36
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
OpName %4 "main"
|
|
OpName %8 "b1"
|
|
OpName %10 "b2"
|
|
OpName %12 "b3"
|
|
OpName %13 "b4"
|
|
OpName %16 "f1"
|
|
OpName %18 "f2"
|
|
OpName %20 "cf1"
|
|
OpName %22 "cf2"
|
|
OpName %26 "i1"
|
|
OpName %28 "i2"
|
|
OpName %30 "ci1"
|
|
OpName %32 "ci2"
|
|
OpName %36 "value"
|
|
OpDecorate %26 RelaxedPrecision
|
|
OpDecorate %28 RelaxedPrecision
|
|
OpDecorate %30 RelaxedPrecision
|
|
OpDecorate %32 RelaxedPrecision
|
|
OpDecorate %36 Location 0
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeBool
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstantTrue %6
|
|
%11 = OpConstantFalse %6
|
|
%14 = OpTypeFloat 32
|
|
%15 = OpTypePointer Function %14
|
|
%17 = OpConstant %14 1.23000002
|
|
%19 = OpConstant %14 1.11000001
|
|
%21 = OpConstant %14 2
|
|
%23 = OpConstant %14 3.29999995
|
|
%24 = OpTypeInt 32 1
|
|
%25 = OpTypePointer Function %24
|
|
%27 = OpConstant %24 1
|
|
%29 = OpConstant %24 100
|
|
%31 = OpConstant %24 123
|
|
%33 = OpConstant %24 1111
|
|
%35 = OpTypePointer Input %14
|
|
%36 = OpVariable %35 Input
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%8 = OpVariable %7 Function
|
|
%10 = OpVariable %7 Function
|
|
%12 = OpVariable %7 Function
|
|
%13 = OpVariable %7 Function
|
|
%16 = OpVariable %15 Function
|
|
%18 = OpVariable %15 Function
|
|
%20 = OpVariable %15 Function
|
|
%22 = OpVariable %15 Function
|
|
%26 = OpVariable %25 Function
|
|
%28 = OpVariable %25 Function
|
|
%30 = OpVariable %25 Function
|
|
%32 = OpVariable %25 Function
|
|
OpStore %8 %9
|
|
OpStore %10 %11
|
|
OpStore %12 %9
|
|
OpStore %13 %11
|
|
OpStore %16 %17
|
|
OpStore %18 %19
|
|
OpStore %20 %21
|
|
OpStore %22 %23
|
|
OpStore %26 %27
|
|
OpStore %28 %29
|
|
OpStore %30 %31
|
|
OpStore %32 %33
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_4;
|
|
const auto consumer = nullptr;
|
|
const std::unique_ptr<opt::IRContext> context =
|
|
BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
spvtools::ValidatorOptions validator_options;
|
|
ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
|
|
kConsoleMessageConsumer));
|
|
TransformationContext transformation_context(
|
|
MakeUnique<FactManager>(context.get()), validator_options);
|
|
|
|
opt::IRContext* ir_context = context.get();
|
|
// A bool constant with value false exists and the id is 11.
|
|
ASSERT_EQ(11, fuzzerutil::MaybeGetBoolConstant(
|
|
ir_context, transformation_context, false, false));
|
|
// A bool constant with value true exists and the id is 9.
|
|
ASSERT_EQ(9, fuzzerutil::MaybeGetBoolConstant(
|
|
ir_context, transformation_context, true, false));
|
|
}
|
|
|
|
TEST(FuzzerutilTest, FuzzerUtilMaybeGetBoolTypeTest) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %92 %52 %53
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
OpDecorate %92 BuiltIn FragCoord
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeInt 32 1
|
|
%7 = OpTypeFloat 32
|
|
%8 = OpTypeStruct %6 %7
|
|
%9 = OpTypePointer Function %8
|
|
%10 = OpTypeFunction %6 %9
|
|
%14 = OpConstant %6 0
|
|
%15 = OpTypePointer Function %6
|
|
%51 = OpTypePointer Private %6
|
|
%21 = OpConstant %6 2
|
|
%23 = OpConstant %6 1
|
|
%24 = OpConstant %7 1
|
|
%25 = OpTypePointer Function %7
|
|
%50 = OpTypePointer Private %7
|
|
%34 = OpTypeBool
|
|
%35 = OpConstantFalse %34
|
|
%52 = OpVariable %50 Private
|
|
%53 = OpVariable %51 Private
|
|
%80 = OpConstantComposite %8 %21 %24
|
|
%90 = OpTypeVector %7 4
|
|
%91 = OpTypePointer Input %90
|
|
%92 = OpVariable %91 Input
|
|
%93 = OpConstantComposite %90 %24 %24 %24 %24
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%20 = OpVariable %9 Function
|
|
%27 = OpVariable %9 Function
|
|
%22 = OpAccessChain %15 %20 %14
|
|
%44 = OpCopyObject %9 %20
|
|
%26 = OpAccessChain %25 %20 %23
|
|
%29 = OpFunctionCall %6 %12 %27
|
|
%30 = OpAccessChain %15 %20 %14
|
|
%45 = OpCopyObject %15 %30
|
|
%81 = OpCopyObject %9 %27
|
|
%33 = OpAccessChain %15 %20 %14
|
|
OpSelectionMerge %37 None
|
|
OpBranchConditional %35 %36 %37
|
|
%36 = OpLabel
|
|
%38 = OpAccessChain %15 %20 %14
|
|
%40 = OpAccessChain %15 %20 %14
|
|
%43 = OpAccessChain %15 %20 %14
|
|
%82 = OpCopyObject %9 %27
|
|
OpBranch %37
|
|
%37 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%12 = OpFunction %6 None %10
|
|
%11 = OpFunctionParameter %9
|
|
%13 = OpLabel
|
|
%46 = OpCopyObject %9 %11
|
|
%16 = OpAccessChain %15 %11 %14
|
|
%95 = OpCopyObject %8 %80
|
|
OpReturnValue %21
|
|
%100 = OpLabel
|
|
OpUnreachable
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_4;
|
|
const auto consumer = nullptr;
|
|
const std::unique_ptr<opt::IRContext> context =
|
|
BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
spvtools::ValidatorOptions validator_options;
|
|
ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
|
|
kConsoleMessageConsumer));
|
|
|
|
opt::IRContext* ir_context = context.get();
|
|
// A bool type with result id of 34 exists.
|
|
ASSERT_TRUE(fuzzerutil::MaybeGetBoolType(ir_context));
|
|
}
|
|
|
|
TEST(FuzzerutilTest, FuzzerUtilMaybeGetCompositeConstantTest) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %54
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
OpName %4 "main"
|
|
OpName %8 "b1"
|
|
OpName %10 "b2"
|
|
OpName %12 "b3"
|
|
OpName %13 "b4"
|
|
OpName %16 "f1"
|
|
OpName %18 "f2"
|
|
OpName %22 "zc"
|
|
OpName %24 "i1"
|
|
OpName %28 "i2"
|
|
OpName %30 "i3"
|
|
OpName %32 "i4"
|
|
OpName %37 "f_arr"
|
|
OpName %47 "i_arr"
|
|
OpName %54 "value"
|
|
OpDecorate %22 RelaxedPrecision
|
|
OpDecorate %24 RelaxedPrecision
|
|
OpDecorate %28 RelaxedPrecision
|
|
OpDecorate %30 RelaxedPrecision
|
|
OpDecorate %32 RelaxedPrecision
|
|
OpDecorate %47 RelaxedPrecision
|
|
OpDecorate %54 Location 0
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeBool
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstantTrue %6
|
|
%11 = OpConstantFalse %6
|
|
%14 = OpTypeFloat 32
|
|
%15 = OpTypePointer Function %14
|
|
%17 = OpConstant %14 1.23000002
|
|
%19 = OpConstant %14 1.11000001
|
|
%20 = OpTypeInt 32 1
|
|
%21 = OpTypePointer Function %20
|
|
%23 = OpConstant %20 0
|
|
%25 = OpConstant %20 1
|
|
%26 = OpTypeInt 32 0
|
|
%27 = OpTypePointer Function %26
|
|
%29 = OpConstant %26 100
|
|
%31 = OpConstant %20 -1
|
|
%33 = OpConstant %20 -99
|
|
%34 = OpConstant %26 5
|
|
%35 = OpTypeArray %14 %34
|
|
%36 = OpTypePointer Function %35
|
|
%38 = OpConstant %14 5.5
|
|
%39 = OpConstant %14 4.4000001
|
|
%40 = OpConstant %14 3.29999995
|
|
%41 = OpConstant %14 2.20000005
|
|
%42 = OpConstant %14 1.10000002
|
|
%43 = OpConstantComposite %35 %38 %39 %40 %41 %42
|
|
%44 = OpConstant %26 3
|
|
%45 = OpTypeArray %20 %44
|
|
%46 = OpTypePointer Function %45
|
|
%48 = OpConstant %20 3
|
|
%49 = OpConstant %20 7
|
|
%50 = OpConstant %20 9
|
|
%51 = OpConstantComposite %45 %48 %49 %50
|
|
%53 = OpTypePointer Input %14
|
|
%54 = OpVariable %53 Input
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%8 = OpVariable %7 Function
|
|
%10 = OpVariable %7 Function
|
|
%12 = OpVariable %7 Function
|
|
%13 = OpVariable %7 Function
|
|
%16 = OpVariable %15 Function
|
|
%18 = OpVariable %15 Function
|
|
%22 = OpVariable %21 Function
|
|
%24 = OpVariable %21 Function
|
|
%28 = OpVariable %27 Function
|
|
%30 = OpVariable %21 Function
|
|
%32 = OpVariable %21 Function
|
|
%37 = OpVariable %36 Function
|
|
%47 = OpVariable %46 Function
|
|
OpStore %8 %9
|
|
OpStore %10 %11
|
|
OpStore %12 %9
|
|
OpStore %13 %11
|
|
OpStore %16 %17
|
|
OpStore %18 %19
|
|
OpStore %22 %23
|
|
OpStore %24 %25
|
|
OpStore %28 %29
|
|
OpStore %30 %31
|
|
OpStore %32 %33
|
|
OpStore %37 %43
|
|
OpStore %47 %51
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_4;
|
|
const auto consumer = nullptr;
|
|
const std::unique_ptr<opt::IRContext> context =
|
|
BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
spvtools::ValidatorOptions validator_options;
|
|
ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
|
|
kConsoleMessageConsumer));
|
|
TransformationContext transformation_context(
|
|
MakeUnique<FactManager>(context.get()), validator_options);
|
|
|
|
opt::IRContext* ir_context = context.get();
|
|
|
|
// %43 = OpConstantComposite %35 %38 %39 %40 %41 %42
|
|
// %51 = OpConstantComposite %45 %48 %49 %50
|
|
// This should pass as a float array with 5 elements exist and its id is 43.
|
|
ASSERT_EQ(43, fuzzerutil::MaybeGetCompositeConstant(
|
|
ir_context, transformation_context, {38, 39, 40, 41, 42},
|
|
35, false));
|
|
// This should pass as an int array with 3 elements exist and its id is 51.
|
|
ASSERT_EQ(51,
|
|
fuzzerutil::MaybeGetCompositeConstant(
|
|
ir_context, transformation_context, {48, 49, 50}, 45, false));
|
|
// An int array with 2 elements does not exist.
|
|
ASSERT_EQ(0, fuzzerutil::MaybeGetCompositeConstant(
|
|
ir_context, transformation_context, {48, 49}, 45, false));
|
|
}
|
|
|
|
TEST(FuzzerutilTest, FuzzerUtilMaybeGetFloatConstantTest) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %36
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
OpName %4 "main"
|
|
OpName %8 "b1"
|
|
OpName %10 "b2"
|
|
OpName %12 "b3"
|
|
OpName %13 "b4"
|
|
OpName %16 "f1"
|
|
OpName %18 "f2"
|
|
OpName %20 "cf1"
|
|
OpName %22 "cf2"
|
|
OpName %26 "i1"
|
|
OpName %28 "i2"
|
|
OpName %30 "ci1"
|
|
OpName %32 "ci2"
|
|
OpName %36 "value"
|
|
OpDecorate %26 RelaxedPrecision
|
|
OpDecorate %28 RelaxedPrecision
|
|
OpDecorate %30 RelaxedPrecision
|
|
OpDecorate %32 RelaxedPrecision
|
|
OpDecorate %36 Location 0
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeBool
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstantTrue %6
|
|
%11 = OpConstantFalse %6
|
|
%14 = OpTypeFloat 32
|
|
%15 = OpTypePointer Function %14
|
|
%17 = OpConstant %14 1.23000002
|
|
%19 = OpConstant %14 1.11000001
|
|
%21 = OpConstant %14 2
|
|
%23 = OpConstant %14 3.29999995
|
|
%24 = OpTypeInt 32 1
|
|
%25 = OpTypePointer Function %24
|
|
%27 = OpConstant %24 1
|
|
%29 = OpConstant %24 100
|
|
%31 = OpConstant %24 123
|
|
%33 = OpConstant %24 1111
|
|
%35 = OpTypePointer Input %14
|
|
%36 = OpVariable %35 Input
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%8 = OpVariable %7 Function
|
|
%10 = OpVariable %7 Function
|
|
%12 = OpVariable %7 Function
|
|
%13 = OpVariable %7 Function
|
|
%16 = OpVariable %15 Function
|
|
%18 = OpVariable %15 Function
|
|
%20 = OpVariable %15 Function
|
|
%22 = OpVariable %15 Function
|
|
%26 = OpVariable %25 Function
|
|
%28 = OpVariable %25 Function
|
|
%30 = OpVariable %25 Function
|
|
%32 = OpVariable %25 Function
|
|
OpStore %8 %9
|
|
OpStore %10 %11
|
|
OpStore %12 %9
|
|
OpStore %13 %11
|
|
OpStore %16 %17
|
|
OpStore %18 %19
|
|
OpStore %20 %21
|
|
OpStore %22 %23
|
|
OpStore %26 %27
|
|
OpStore %28 %29
|
|
OpStore %30 %31
|
|
OpStore %32 %33
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_4;
|
|
const auto consumer = nullptr;
|
|
const std::unique_ptr<opt::IRContext> context =
|
|
BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
spvtools::ValidatorOptions validator_options;
|
|
ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
|
|
kConsoleMessageConsumer));
|
|
TransformationContext transformation_context(
|
|
MakeUnique<FactManager>(context.get()), validator_options);
|
|
|
|
opt::IRContext* ir_context = context.get();
|
|
|
|
uint32_t word1 = fuzzerutil::FloatToWord(2);
|
|
uint32_t word2 = fuzzerutil::FloatToWord(1.23f);
|
|
|
|
// A 32 bit float constant of value 2 exists and its id is 21.
|
|
ASSERT_EQ(21, fuzzerutil::MaybeGetFloatConstant(
|
|
ir_context, transformation_context,
|
|
std::vector<uint32_t>{word1}, 32, false));
|
|
// A 32 bit float constant of value 1.23 exists and its id is 17.
|
|
ASSERT_EQ(17, fuzzerutil::MaybeGetFloatConstant(
|
|
ir_context, transformation_context,
|
|
std::vector<uint32_t>{word2}, 32, false));
|
|
}
|
|
|
|
TEST(FuzzerutilTest, FuzzerUtilMaybeGetFloatTypeTest) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %92 %52 %53
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
OpDecorate %92 BuiltIn FragCoord
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeInt 32 1
|
|
%7 = OpTypeFloat 32
|
|
%8 = OpTypeStruct %6 %7
|
|
%9 = OpTypePointer Function %8
|
|
%10 = OpTypeFunction %6 %9
|
|
%14 = OpConstant %6 0
|
|
%15 = OpTypePointer Function %6
|
|
%51 = OpTypePointer Private %6
|
|
%21 = OpConstant %6 2
|
|
%23 = OpConstant %6 1
|
|
%24 = OpConstant %7 1
|
|
%25 = OpTypePointer Function %7
|
|
%50 = OpTypePointer Private %7
|
|
%34 = OpTypeBool
|
|
%35 = OpConstantFalse %34
|
|
%52 = OpVariable %50 Private
|
|
%53 = OpVariable %51 Private
|
|
%80 = OpConstantComposite %8 %21 %24
|
|
%90 = OpTypeVector %7 4
|
|
%91 = OpTypePointer Input %90
|
|
%92 = OpVariable %91 Input
|
|
%93 = OpConstantComposite %90 %24 %24 %24 %24
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%20 = OpVariable %9 Function
|
|
%27 = OpVariable %9 Function
|
|
%22 = OpAccessChain %15 %20 %14
|
|
%44 = OpCopyObject %9 %20
|
|
%26 = OpAccessChain %25 %20 %23
|
|
%29 = OpFunctionCall %6 %12 %27
|
|
%30 = OpAccessChain %15 %20 %14
|
|
%45 = OpCopyObject %15 %30
|
|
%81 = OpCopyObject %9 %27
|
|
%33 = OpAccessChain %15 %20 %14
|
|
OpSelectionMerge %37 None
|
|
OpBranchConditional %35 %36 %37
|
|
%36 = OpLabel
|
|
%38 = OpAccessChain %15 %20 %14
|
|
%40 = OpAccessChain %15 %20 %14
|
|
%43 = OpAccessChain %15 %20 %14
|
|
%82 = OpCopyObject %9 %27
|
|
OpBranch %37
|
|
%37 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%12 = OpFunction %6 None %10
|
|
%11 = OpFunctionParameter %9
|
|
%13 = OpLabel
|
|
%46 = OpCopyObject %9 %11
|
|
%16 = OpAccessChain %15 %11 %14
|
|
%95 = OpCopyObject %8 %80
|
|
OpReturnValue %21
|
|
%100 = OpLabel
|
|
OpUnreachable
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_4;
|
|
const auto consumer = nullptr;
|
|
const std::unique_ptr<opt::IRContext> context =
|
|
BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
spvtools::ValidatorOptions validator_options;
|
|
ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
|
|
kConsoleMessageConsumer));
|
|
|
|
opt::IRContext* ir_context = context.get();
|
|
// A float type with width = 32 and result id of 7 exists.
|
|
ASSERT_EQ(7, fuzzerutil::MaybeGetFloatType(ir_context, 32));
|
|
|
|
// A float int type with width = 32 exists, but the id should be 7.
|
|
ASSERT_NE(5, fuzzerutil::MaybeGetFloatType(ir_context, 32));
|
|
|
|
// A float type with width 30 does not exist.
|
|
ASSERT_EQ(0, fuzzerutil::MaybeGetFloatType(ir_context, 30));
|
|
}
|
|
|
|
TEST(FuzzerutilTest, FuzzerUtilMaybeGetIntegerConstantFromValueAndTypeTest) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %36
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
OpName %4 "main"
|
|
OpName %8 "b1"
|
|
OpName %10 "b2"
|
|
OpName %12 "b3"
|
|
OpName %13 "b4"
|
|
OpName %16 "f1"
|
|
OpName %18 "f2"
|
|
OpName %22 "zc"
|
|
OpName %24 "i1"
|
|
OpName %28 "i2"
|
|
OpName %30 "i3"
|
|
OpName %32 "i4"
|
|
OpName %36 "value"
|
|
OpDecorate %22 RelaxedPrecision
|
|
OpDecorate %24 RelaxedPrecision
|
|
OpDecorate %28 RelaxedPrecision
|
|
OpDecorate %30 RelaxedPrecision
|
|
OpDecorate %32 RelaxedPrecision
|
|
OpDecorate %36 Location 0
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeBool
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstantTrue %6
|
|
%11 = OpConstantFalse %6
|
|
%14 = OpTypeFloat 32
|
|
%15 = OpTypePointer Function %14
|
|
%17 = OpConstant %14 1.23000002
|
|
%19 = OpConstant %14 1.11000001
|
|
%20 = OpTypeInt 32 1
|
|
%21 = OpTypePointer Function %20
|
|
%23 = OpConstant %20 0
|
|
%25 = OpConstant %20 1
|
|
%26 = OpTypeInt 32 0
|
|
%27 = OpTypePointer Function %26
|
|
%29 = OpConstant %26 100
|
|
%31 = OpConstant %20 -1
|
|
%33 = OpConstant %20 -99
|
|
%35 = OpTypePointer Input %14
|
|
%36 = OpVariable %35 Input
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%8 = OpVariable %7 Function
|
|
%10 = OpVariable %7 Function
|
|
%12 = OpVariable %7 Function
|
|
%13 = OpVariable %7 Function
|
|
%16 = OpVariable %15 Function
|
|
%18 = OpVariable %15 Function
|
|
%22 = OpVariable %21 Function
|
|
%24 = OpVariable %21 Function
|
|
%28 = OpVariable %27 Function
|
|
%30 = OpVariable %21 Function
|
|
%32 = OpVariable %21 Function
|
|
OpStore %8 %9
|
|
OpStore %10 %11
|
|
OpStore %12 %9
|
|
OpStore %13 %11
|
|
OpStore %16 %17
|
|
OpStore %18 %19
|
|
OpStore %22 %23
|
|
OpStore %24 %25
|
|
OpStore %28 %29
|
|
OpStore %30 %31
|
|
OpStore %32 %33
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_4;
|
|
const auto consumer = nullptr;
|
|
const std::unique_ptr<opt::IRContext> context =
|
|
BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
spvtools::ValidatorOptions validator_options;
|
|
ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
|
|
kConsoleMessageConsumer));
|
|
|
|
opt::IRContext* ir_context = context.get();
|
|
|
|
// A 32 bit signed int constant (with int type id 20) with value 1 exists and
|
|
// the id is 25.
|
|
ASSERT_EQ(25, fuzzerutil::MaybeGetIntegerConstantFromValueAndType(ir_context,
|
|
1, 20));
|
|
// A 32 bit unsigned int constant (with int type id 0) with value 100 exists
|
|
// and the id is 29.
|
|
ASSERT_EQ(29, fuzzerutil::MaybeGetIntegerConstantFromValueAndType(ir_context,
|
|
100, 26));
|
|
// A 32 bit unsigned int constant with value 50 does not exist.
|
|
ASSERT_EQ(0, fuzzerutil::MaybeGetIntegerConstantFromValueAndType(ir_context,
|
|
50, 26));
|
|
}
|
|
|
|
TEST(FuzzerutilTest, FuzzerUtilMaybeGetIntegerConstantTest) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %36
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
OpName %4 "main"
|
|
OpName %8 "b1"
|
|
OpName %10 "b2"
|
|
OpName %12 "b3"
|
|
OpName %13 "b4"
|
|
OpName %16 "f1"
|
|
OpName %18 "f2"
|
|
OpName %22 "zc"
|
|
OpName %24 "i1"
|
|
OpName %28 "i2"
|
|
OpName %30 "i3"
|
|
OpName %32 "i4"
|
|
OpName %36 "value"
|
|
OpDecorate %22 RelaxedPrecision
|
|
OpDecorate %24 RelaxedPrecision
|
|
OpDecorate %28 RelaxedPrecision
|
|
OpDecorate %30 RelaxedPrecision
|
|
OpDecorate %32 RelaxedPrecision
|
|
OpDecorate %36 Location 0
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeBool
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstantTrue %6
|
|
%11 = OpConstantFalse %6
|
|
%14 = OpTypeFloat 32
|
|
%15 = OpTypePointer Function %14
|
|
%17 = OpConstant %14 1.23000002
|
|
%19 = OpConstant %14 1.11000001
|
|
%20 = OpTypeInt 32 1
|
|
%21 = OpTypePointer Function %20
|
|
%23 = OpConstant %20 0
|
|
%25 = OpConstant %20 1
|
|
%26 = OpTypeInt 32 0
|
|
%27 = OpTypePointer Function %26
|
|
%29 = OpConstant %26 100
|
|
%31 = OpConstant %20 -1
|
|
%33 = OpConstant %20 -99
|
|
%35 = OpTypePointer Input %14
|
|
%36 = OpVariable %35 Input
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%8 = OpVariable %7 Function
|
|
%10 = OpVariable %7 Function
|
|
%12 = OpVariable %7 Function
|
|
%13 = OpVariable %7 Function
|
|
%16 = OpVariable %15 Function
|
|
%18 = OpVariable %15 Function
|
|
%22 = OpVariable %21 Function
|
|
%24 = OpVariable %21 Function
|
|
%28 = OpVariable %27 Function
|
|
%30 = OpVariable %21 Function
|
|
%32 = OpVariable %21 Function
|
|
OpStore %8 %9
|
|
OpStore %10 %11
|
|
OpStore %12 %9
|
|
OpStore %13 %11
|
|
OpStore %16 %17
|
|
OpStore %18 %19
|
|
OpStore %22 %23
|
|
OpStore %24 %25
|
|
OpStore %28 %29
|
|
OpStore %30 %31
|
|
OpStore %32 %33
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_4;
|
|
const auto consumer = nullptr;
|
|
const std::unique_ptr<opt::IRContext> context =
|
|
BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
spvtools::ValidatorOptions validator_options;
|
|
ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
|
|
kConsoleMessageConsumer));
|
|
TransformationContext transformation_context(
|
|
MakeUnique<FactManager>(context.get()), validator_options);
|
|
|
|
opt::IRContext* ir_context = context.get();
|
|
|
|
// A 32 bit unsigned int constant with value 1 exists and the id is 25.
|
|
ASSERT_EQ(25, fuzzerutil::MaybeGetIntegerConstant(
|
|
ir_context, transformation_context,
|
|
std::vector<uint32_t>{1}, 32, true, false));
|
|
// A 32 bit unsigned int constant with value 100 exists and the id is 29.
|
|
ASSERT_EQ(29, fuzzerutil::MaybeGetIntegerConstant(
|
|
ir_context, transformation_context,
|
|
std::vector<uint32_t>{100}, 32, false, false));
|
|
// A 32 bit signed int constant with value 99 doesn't not exist and should
|
|
// return 0.
|
|
ASSERT_EQ(0, fuzzerutil::MaybeGetIntegerConstant(
|
|
ir_context, transformation_context,
|
|
std::vector<uint32_t>{99}, 32, true, false));
|
|
}
|
|
|
|
TEST(FuzzerutilTest, FuzzerUtilMaybeGetIntegerTypeTest) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %92 %52 %53
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
OpDecorate %92 BuiltIn FragCoord
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeInt 32 1
|
|
%7 = OpTypeFloat 32
|
|
%8 = OpTypeStruct %6 %7
|
|
%9 = OpTypePointer Function %8
|
|
%10 = OpTypeFunction %6 %9
|
|
%14 = OpConstant %6 0
|
|
%15 = OpTypePointer Function %6
|
|
%51 = OpTypePointer Private %6
|
|
%21 = OpConstant %6 2
|
|
%23 = OpConstant %6 1
|
|
%24 = OpConstant %7 1
|
|
%25 = OpTypePointer Function %7
|
|
%50 = OpTypePointer Private %7
|
|
%34 = OpTypeBool
|
|
%35 = OpConstantFalse %34
|
|
%52 = OpVariable %50 Private
|
|
%53 = OpVariable %51 Private
|
|
%80 = OpConstantComposite %8 %21 %24
|
|
%90 = OpTypeVector %7 4
|
|
%91 = OpTypePointer Input %90
|
|
%92 = OpVariable %91 Input
|
|
%93 = OpConstantComposite %90 %24 %24 %24 %24
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%20 = OpVariable %9 Function
|
|
%27 = OpVariable %9 Function
|
|
%22 = OpAccessChain %15 %20 %14
|
|
%44 = OpCopyObject %9 %20
|
|
%26 = OpAccessChain %25 %20 %23
|
|
%29 = OpFunctionCall %6 %12 %27
|
|
%30 = OpAccessChain %15 %20 %14
|
|
%45 = OpCopyObject %15 %30
|
|
%81 = OpCopyObject %9 %27
|
|
%33 = OpAccessChain %15 %20 %14
|
|
OpSelectionMerge %37 None
|
|
OpBranchConditional %35 %36 %37
|
|
%36 = OpLabel
|
|
%38 = OpAccessChain %15 %20 %14
|
|
%40 = OpAccessChain %15 %20 %14
|
|
%43 = OpAccessChain %15 %20 %14
|
|
%82 = OpCopyObject %9 %27
|
|
OpBranch %37
|
|
%37 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%12 = OpFunction %6 None %10
|
|
%11 = OpFunctionParameter %9
|
|
%13 = OpLabel
|
|
%46 = OpCopyObject %9 %11
|
|
%16 = OpAccessChain %15 %11 %14
|
|
%95 = OpCopyObject %8 %80
|
|
OpReturnValue %21
|
|
%100 = OpLabel
|
|
OpUnreachable
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_4;
|
|
const auto consumer = nullptr;
|
|
const std::unique_ptr<opt::IRContext> context =
|
|
BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
spvtools::ValidatorOptions validator_options;
|
|
ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
|
|
kConsoleMessageConsumer));
|
|
|
|
opt::IRContext* ir_context = context.get();
|
|
|
|
// A signed int type with width = 32 and result id of 6 exists.
|
|
ASSERT_EQ(6, fuzzerutil::MaybeGetIntegerType(ir_context, 32, true));
|
|
|
|
// A signed int type with width = 32 exists, but the id should be 6.
|
|
ASSERT_FALSE(fuzzerutil::MaybeGetIntegerType(ir_context, 32, true) == 5);
|
|
|
|
// A int type with width = 32 and result id of 6 exists, but it should be a
|
|
// signed int.
|
|
ASSERT_EQ(0, fuzzerutil::MaybeGetIntegerType(ir_context, 32, false));
|
|
// A signed int type with width 30 does not exist.
|
|
ASSERT_EQ(0, fuzzerutil::MaybeGetIntegerType(ir_context, 30, true));
|
|
// An unsigned int type with width 22 does not exist.
|
|
ASSERT_EQ(0, fuzzerutil::MaybeGetIntegerType(ir_context, 22, false));
|
|
}
|
|
|
|
TEST(FuzzerutilTest, FuzzerUtilMaybeGetPointerTypeTest) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %92 %52 %53
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
OpDecorate %92 BuiltIn FragCoord
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeInt 32 1
|
|
%7 = OpTypeFloat 32
|
|
%8 = OpTypeStruct %6 %7
|
|
%9 = OpTypePointer Function %8
|
|
%10 = OpTypeFunction %6 %9
|
|
%14 = OpConstant %6 0
|
|
%15 = OpTypePointer Function %6
|
|
%51 = OpTypePointer Private %6
|
|
%21 = OpConstant %6 2
|
|
%23 = OpConstant %6 1
|
|
%24 = OpConstant %7 1
|
|
%25 = OpTypePointer Function %7
|
|
%50 = OpTypePointer Private %7
|
|
%34 = OpTypeBool
|
|
%35 = OpConstantFalse %34
|
|
%52 = OpVariable %50 Private
|
|
%53 = OpVariable %51 Private
|
|
%80 = OpConstantComposite %8 %21 %24
|
|
%90 = OpTypeVector %7 4
|
|
%91 = OpTypePointer Input %90
|
|
%92 = OpVariable %91 Input
|
|
%93 = OpConstantComposite %90 %24 %24 %24 %24
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%20 = OpVariable %9 Function
|
|
%27 = OpVariable %9 Function
|
|
%22 = OpAccessChain %15 %20 %14
|
|
%44 = OpCopyObject %9 %20
|
|
%26 = OpAccessChain %25 %20 %23
|
|
%29 = OpFunctionCall %6 %12 %27
|
|
%30 = OpAccessChain %15 %20 %14
|
|
%45 = OpCopyObject %15 %30
|
|
%81 = OpCopyObject %9 %27
|
|
%33 = OpAccessChain %15 %20 %14
|
|
OpSelectionMerge %37 None
|
|
OpBranchConditional %35 %36 %37
|
|
%36 = OpLabel
|
|
%38 = OpAccessChain %15 %20 %14
|
|
%40 = OpAccessChain %15 %20 %14
|
|
%43 = OpAccessChain %15 %20 %14
|
|
%82 = OpCopyObject %9 %27
|
|
OpBranch %37
|
|
%37 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%12 = OpFunction %6 None %10
|
|
%11 = OpFunctionParameter %9
|
|
%13 = OpLabel
|
|
%46 = OpCopyObject %9 %11
|
|
%16 = OpAccessChain %15 %11 %14
|
|
%95 = OpCopyObject %8 %80
|
|
OpReturnValue %21
|
|
%100 = OpLabel
|
|
OpUnreachable
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_4;
|
|
const auto consumer = nullptr;
|
|
const std::unique_ptr<opt::IRContext> context =
|
|
BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
spvtools::ValidatorOptions validator_options;
|
|
ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
|
|
kConsoleMessageConsumer));
|
|
|
|
opt::IRContext* ir_context = context.get();
|
|
auto private_storage_class = SpvStorageClassPrivate;
|
|
auto function_storage_class = SpvStorageClassFunction;
|
|
auto input_storage_class = SpvStorageClassInput;
|
|
|
|
// A valid pointer must have the correct |pointee_type_id| and |storageClass|.
|
|
// A function type pointer with id = 9 and pointee type id 8 should be found.
|
|
ASSERT_EQ(9, fuzzerutil::MaybeGetPointerType(ir_context, 8,
|
|
function_storage_class));
|
|
// A function type pointer with id = 15 and pointee type id 6 should be found.
|
|
ASSERT_EQ(15, fuzzerutil::MaybeGetPointerType(ir_context, 6,
|
|
function_storage_class));
|
|
// A function type pointer with id = 25 and pointee type id 7 should be found.
|
|
ASSERT_EQ(25, fuzzerutil::MaybeGetPointerType(ir_context, 7,
|
|
function_storage_class));
|
|
|
|
// A private type pointer with id=51 and pointee type id 6 should be found.
|
|
ASSERT_EQ(51, fuzzerutil::MaybeGetPointerType(ir_context, 6,
|
|
private_storage_class));
|
|
// A function pointer with id=50 and pointee type id 7 should be found.
|
|
ASSERT_EQ(50, fuzzerutil::MaybeGetPointerType(ir_context, 7,
|
|
private_storage_class));
|
|
|
|
// A input type pointer with id=91 and pointee type id 90 should be found.
|
|
ASSERT_EQ(
|
|
91, fuzzerutil::MaybeGetPointerType(ir_context, 90, input_storage_class));
|
|
|
|
// A pointer with id=91 and pointee type 90 exists, but the type should be
|
|
// input.
|
|
ASSERT_EQ(0, fuzzerutil::MaybeGetPointerType(ir_context, 90,
|
|
function_storage_class));
|
|
// A input type pointer with id=91 exists but the pointee id should be 90.
|
|
ASSERT_EQ(
|
|
0, fuzzerutil::MaybeGetPointerType(ir_context, 89, input_storage_class));
|
|
// A input type pointer with pointee id 90 exists but result id of the pointer
|
|
// should be 91.
|
|
ASSERT_NE(
|
|
58, fuzzerutil::MaybeGetPointerType(ir_context, 90, input_storage_class));
|
|
}
|
|
|
|
TEST(FuzzerutilTest, FuzzerUtilMaybeGetScalarConstantTest) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %56
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
OpName %4 "main"
|
|
OpName %8 "b1"
|
|
OpName %10 "b2"
|
|
OpName %12 "b3"
|
|
OpName %13 "b4"
|
|
OpName %16 "f1"
|
|
OpName %18 "f2"
|
|
OpName %22 "zc"
|
|
OpName %24 "i1"
|
|
OpName %28 "i2"
|
|
OpName %30 "i"
|
|
OpName %32 "i3"
|
|
OpName %34 "i4"
|
|
OpName %39 "f_arr"
|
|
OpName %49 "i_arr"
|
|
OpName %56 "value"
|
|
OpDecorate %22 RelaxedPrecision
|
|
OpDecorate %24 RelaxedPrecision
|
|
OpDecorate %28 RelaxedPrecision
|
|
OpDecorate %30 RelaxedPrecision
|
|
OpDecorate %32 RelaxedPrecision
|
|
OpDecorate %34 RelaxedPrecision
|
|
OpDecorate %49 RelaxedPrecision
|
|
OpDecorate %56 Location 0
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeBool
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstantTrue %6
|
|
%11 = OpConstantFalse %6
|
|
%14 = OpTypeFloat 32
|
|
%15 = OpTypePointer Function %14
|
|
%17 = OpConstant %14 1.23000002
|
|
%19 = OpConstant %14 1.11000001
|
|
%20 = OpTypeInt 32 1
|
|
%21 = OpTypePointer Function %20
|
|
%23 = OpConstant %20 0
|
|
%25 = OpConstant %20 1
|
|
%26 = OpTypeInt 32 0
|
|
%27 = OpTypePointer Function %26
|
|
%29 = OpConstant %26 100
|
|
%31 = OpConstant %26 0
|
|
%33 = OpConstant %20 -1
|
|
%35 = OpConstant %20 -99
|
|
%36 = OpConstant %26 5
|
|
%37 = OpTypeArray %14 %36
|
|
%38 = OpTypePointer Function %37
|
|
%40 = OpConstant %14 5.5
|
|
%41 = OpConstant %14 4.4000001
|
|
%42 = OpConstant %14 3.29999995
|
|
%43 = OpConstant %14 2.20000005
|
|
%44 = OpConstant %14 1.10000002
|
|
%45 = OpConstantComposite %37 %40 %41 %42 %43 %44
|
|
%46 = OpConstant %26 3
|
|
%47 = OpTypeArray %20 %46
|
|
%48 = OpTypePointer Function %47
|
|
%50 = OpConstant %20 3
|
|
%51 = OpConstant %20 7
|
|
%52 = OpConstant %20 9
|
|
%53 = OpConstantComposite %47 %50 %51 %52
|
|
%55 = OpTypePointer Input %14
|
|
%56 = OpVariable %55 Input
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%8 = OpVariable %7 Function
|
|
%10 = OpVariable %7 Function
|
|
%12 = OpVariable %7 Function
|
|
%13 = OpVariable %7 Function
|
|
%16 = OpVariable %15 Function
|
|
%18 = OpVariable %15 Function
|
|
%22 = OpVariable %21 Function
|
|
%24 = OpVariable %21 Function
|
|
%28 = OpVariable %27 Function
|
|
%30 = OpVariable %27 Function
|
|
%32 = OpVariable %21 Function
|
|
%34 = OpVariable %21 Function
|
|
%39 = OpVariable %38 Function
|
|
%49 = OpVariable %48 Function
|
|
OpStore %8 %9
|
|
OpStore %10 %11
|
|
OpStore %12 %9
|
|
OpStore %13 %11
|
|
OpStore %16 %17
|
|
OpStore %18 %19
|
|
OpStore %22 %23
|
|
OpStore %24 %25
|
|
OpStore %28 %29
|
|
OpStore %30 %31
|
|
OpStore %32 %33
|
|
OpStore %34 %35
|
|
OpStore %39 %45
|
|
OpStore %49 %53
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_4;
|
|
const auto consumer = nullptr;
|
|
const std::unique_ptr<opt::IRContext> context =
|
|
BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
spvtools::ValidatorOptions validator_options;
|
|
ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
|
|
kConsoleMessageConsumer));
|
|
TransformationContext transformation_context(
|
|
MakeUnique<FactManager>(context.get()), validator_options);
|
|
|
|
opt::IRContext* ir_context = context.get();
|
|
|
|
std::vector<uint32_t> uint_words1 = fuzzerutil::IntToWords(100, 32, false);
|
|
std::vector<uint32_t> uint_words2 = fuzzerutil::IntToWords(0, 32, false);
|
|
std::vector<uint32_t> int_words1 = fuzzerutil::IntToWords(-99, 32, true);
|
|
std::vector<uint32_t> int_words2 = fuzzerutil::IntToWords(1, 32, true);
|
|
uint32_t float_word1 = fuzzerutil::FloatToWord(1.11f);
|
|
uint32_t float_word2 = fuzzerutil::FloatToWord(4.4f);
|
|
|
|
// A unsigned int of value 100 that has a scalar type id of 26 exists and its
|
|
// id is 29.
|
|
ASSERT_EQ(
|
|
29, fuzzerutil::MaybeGetScalarConstant(ir_context, transformation_context,
|
|
uint_words1, 26, false));
|
|
// A unsigned int of value 0 that has a scalar type id of 26 exists and its id
|
|
// is 29.
|
|
ASSERT_EQ(
|
|
31, fuzzerutil::MaybeGetScalarConstant(ir_context, transformation_context,
|
|
uint_words2, 26, false));
|
|
// A signed int of value -99 that has a scalar type id of 20 exists and its id
|
|
// is 35.
|
|
ASSERT_EQ(35, fuzzerutil::MaybeGetScalarConstant(
|
|
ir_context, transformation_context, int_words1, 20, false));
|
|
// A signed int of value 1 that has a scalar type id of 20 exists and its id
|
|
// is 25.
|
|
ASSERT_EQ(25, fuzzerutil::MaybeGetScalarConstant(
|
|
ir_context, transformation_context, int_words2, 20, false));
|
|
// A float of value 1.11 that has a scalar type id of 14 exists and its id
|
|
// is 19.
|
|
ASSERT_EQ(19, fuzzerutil::MaybeGetScalarConstant(
|
|
ir_context, transformation_context,
|
|
std::vector<uint32_t>{float_word1}, 14, false));
|
|
// A signed int of value 1 that has a scalar type id of 20 exists and its id
|
|
// is 25.
|
|
ASSERT_EQ(41, fuzzerutil::MaybeGetScalarConstant(
|
|
ir_context, transformation_context,
|
|
std::vector<uint32_t>{float_word2}, 14, false));
|
|
}
|
|
|
|
TEST(FuzzerutilTest, FuzzerUtilMaybeGetStructTypeTest) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %92 %52 %53
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
OpDecorate %92 BuiltIn FragCoord
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeInt 32 1
|
|
%7 = OpTypeFloat 32
|
|
%8 = OpTypeStruct %6 %7
|
|
%9 = OpTypePointer Function %8
|
|
%10 = OpTypeFunction %6 %9
|
|
%14 = OpConstant %6 0
|
|
%15 = OpTypePointer Function %6
|
|
%51 = OpTypePointer Private %6
|
|
%21 = OpConstant %6 2
|
|
%23 = OpConstant %6 1
|
|
%24 = OpConstant %7 1
|
|
%25 = OpTypePointer Function %7
|
|
%50 = OpTypePointer Private %7
|
|
%34 = OpTypeBool
|
|
%35 = OpConstantFalse %34
|
|
%52 = OpVariable %50 Private
|
|
%53 = OpVariable %51 Private
|
|
%80 = OpConstantComposite %8 %21 %24
|
|
%90 = OpTypeVector %7 4
|
|
%91 = OpTypePointer Input %90
|
|
%92 = OpVariable %91 Input
|
|
%93 = OpConstantComposite %90 %24 %24 %24 %24
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%20 = OpVariable %9 Function
|
|
%27 = OpVariable %9 Function
|
|
%22 = OpAccessChain %15 %20 %14
|
|
%44 = OpCopyObject %9 %20
|
|
%26 = OpAccessChain %25 %20 %23
|
|
%29 = OpFunctionCall %6 %12 %27
|
|
%30 = OpAccessChain %15 %20 %14
|
|
%45 = OpCopyObject %15 %30
|
|
%81 = OpCopyObject %9 %27
|
|
%33 = OpAccessChain %15 %20 %14
|
|
OpSelectionMerge %37 None
|
|
OpBranchConditional %35 %36 %37
|
|
%36 = OpLabel
|
|
%38 = OpAccessChain %15 %20 %14
|
|
%40 = OpAccessChain %15 %20 %14
|
|
%43 = OpAccessChain %15 %20 %14
|
|
%82 = OpCopyObject %9 %27
|
|
OpBranch %37
|
|
%37 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%12 = OpFunction %6 None %10
|
|
%11 = OpFunctionParameter %9
|
|
%13 = OpLabel
|
|
%46 = OpCopyObject %9 %11
|
|
%16 = OpAccessChain %15 %11 %14
|
|
%95 = OpCopyObject %8 %80
|
|
OpReturnValue %21
|
|
%100 = OpLabel
|
|
OpUnreachable
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_4;
|
|
const auto consumer = nullptr;
|
|
const std::unique_ptr<opt::IRContext> context =
|
|
BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
spvtools::ValidatorOptions validator_options;
|
|
ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
|
|
kConsoleMessageConsumer));
|
|
|
|
opt::IRContext* ir_context = context.get();
|
|
|
|
// 6 and 7 are all valid ids from OpTypeInt and OpTypeFloat
|
|
// so the result id of 8 should be found.
|
|
ASSERT_EQ(8, fuzzerutil::MaybeGetStructType(ir_context,
|
|
std::vector<uint32_t>{6, 7}));
|
|
|
|
// |component_type_id| of 16 does not exist in the module, so such a struct
|
|
// type cannot be found.
|
|
ASSERT_EQ(0, fuzzerutil::MaybeGetStructType(ir_context,
|
|
std::vector<uint32_t>(6, 16)));
|
|
|
|
// |component_type_id| of 10 is of OpTypeFunction type and thus the struct
|
|
// cannot be found.
|
|
ASSERT_EQ(0, fuzzerutil::MaybeGetStructType(ir_context,
|
|
std::vector<uint32_t>(6, 10)));
|
|
}
|
|
|
|
TEST(FuzzerutilTest, FuzzerUtilMaybeGetVectorTypeTest) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %92 %52 %53
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
OpDecorate %92 BuiltIn FragCoord
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeInt 32 1
|
|
%7 = OpTypeFloat 32
|
|
%8 = OpTypeStruct %6 %7
|
|
%9 = OpTypePointer Function %8
|
|
%10 = OpTypeFunction %6 %9
|
|
%14 = OpConstant %6 0
|
|
%15 = OpTypePointer Function %6
|
|
%51 = OpTypePointer Private %6
|
|
%21 = OpConstant %6 2
|
|
%23 = OpConstant %6 1
|
|
%24 = OpConstant %7 1
|
|
%25 = OpTypePointer Function %7
|
|
%50 = OpTypePointer Private %7
|
|
%34 = OpTypeBool
|
|
%35 = OpConstantFalse %34
|
|
%52 = OpVariable %50 Private
|
|
%53 = OpVariable %51 Private
|
|
%80 = OpConstantComposite %8 %21 %24
|
|
%90 = OpTypeVector %7 4
|
|
%91 = OpTypePointer Input %90
|
|
%92 = OpVariable %91 Input
|
|
%93 = OpConstantComposite %90 %24 %24 %24 %24
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%20 = OpVariable %9 Function
|
|
%27 = OpVariable %9 Function
|
|
%22 = OpAccessChain %15 %20 %14
|
|
%44 = OpCopyObject %9 %20
|
|
%26 = OpAccessChain %25 %20 %23
|
|
%29 = OpFunctionCall %6 %12 %27
|
|
%30 = OpAccessChain %15 %20 %14
|
|
%45 = OpCopyObject %15 %30
|
|
%81 = OpCopyObject %9 %27
|
|
%33 = OpAccessChain %15 %20 %14
|
|
OpSelectionMerge %37 None
|
|
OpBranchConditional %35 %36 %37
|
|
%36 = OpLabel
|
|
%38 = OpAccessChain %15 %20 %14
|
|
%40 = OpAccessChain %15 %20 %14
|
|
%43 = OpAccessChain %15 %20 %14
|
|
%82 = OpCopyObject %9 %27
|
|
OpBranch %37
|
|
%37 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%12 = OpFunction %6 None %10
|
|
%11 = OpFunctionParameter %9
|
|
%13 = OpLabel
|
|
%46 = OpCopyObject %9 %11
|
|
%16 = OpAccessChain %15 %11 %14
|
|
%95 = OpCopyObject %8 %80
|
|
OpReturnValue %21
|
|
%100 = OpLabel
|
|
OpUnreachable
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_4;
|
|
const auto consumer = nullptr;
|
|
const std::unique_ptr<opt::IRContext> context =
|
|
BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
spvtools::ValidatorOptions validator_options;
|
|
ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
|
|
kConsoleMessageConsumer));
|
|
|
|
opt::IRContext* ir_context = context.get();
|
|
// The vector type with |element_count| 4 and |component_type_id| 7
|
|
// is present and has a result id of 90.
|
|
ASSERT_EQ(90, fuzzerutil::MaybeGetVectorType(ir_context, 7, 4));
|
|
|
|
// The vector type with |element_count| 3 and |component_type_id| 7
|
|
// is not present in the module.
|
|
ASSERT_EQ(0, fuzzerutil::MaybeGetVectorType(ir_context, 7, 3));
|
|
|
|
#ifndef NDEBUG
|
|
// It should abort with |component_type_id| of 100
|
|
// |component_type_id| must be a valid result id of an OpTypeInt,
|
|
// OpTypeFloat or OpTypeBool instruction in the module.
|
|
ASSERT_DEATH(fuzzerutil::MaybeGetVectorType(ir_context, 100, 4),
|
|
"\\|component_type_id\\| is invalid");
|
|
|
|
// It should abort with |element_count| of 5.
|
|
// |element_count| must be in the range [2,4].
|
|
ASSERT_DEATH(fuzzerutil::MaybeGetVectorType(ir_context, 7, 5),
|
|
"Precondition: component count must be in range \\[2, 4\\].");
|
|
#endif
|
|
}
|
|
|
|
TEST(FuzzerutilTest, FuzzerUtilMaybeGetVoidTypeTest) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %92 %52 %53
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
OpDecorate %92 BuiltIn FragCoord
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeInt 32 1
|
|
%7 = OpTypeFloat 32
|
|
%8 = OpTypeStruct %6 %7
|
|
%9 = OpTypePointer Function %8
|
|
%10 = OpTypeFunction %6 %9
|
|
%14 = OpConstant %6 0
|
|
%15 = OpTypePointer Function %6
|
|
%51 = OpTypePointer Private %6
|
|
%21 = OpConstant %6 2
|
|
%23 = OpConstant %6 1
|
|
%24 = OpConstant %7 1
|
|
%25 = OpTypePointer Function %7
|
|
%50 = OpTypePointer Private %7
|
|
%34 = OpTypeBool
|
|
%35 = OpConstantFalse %34
|
|
%52 = OpVariable %50 Private
|
|
%53 = OpVariable %51 Private
|
|
%80 = OpConstantComposite %8 %21 %24
|
|
%90 = OpTypeVector %7 4
|
|
%91 = OpTypePointer Input %90
|
|
%92 = OpVariable %91 Input
|
|
%93 = OpConstantComposite %90 %24 %24 %24 %24
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%20 = OpVariable %9 Function
|
|
%27 = OpVariable %9 Function
|
|
%22 = OpAccessChain %15 %20 %14
|
|
%44 = OpCopyObject %9 %20
|
|
%26 = OpAccessChain %25 %20 %23
|
|
%29 = OpFunctionCall %6 %12 %27
|
|
%30 = OpAccessChain %15 %20 %14
|
|
%45 = OpCopyObject %15 %30
|
|
%81 = OpCopyObject %9 %27
|
|
%33 = OpAccessChain %15 %20 %14
|
|
OpSelectionMerge %37 None
|
|
OpBranchConditional %35 %36 %37
|
|
%36 = OpLabel
|
|
%38 = OpAccessChain %15 %20 %14
|
|
%40 = OpAccessChain %15 %20 %14
|
|
%43 = OpAccessChain %15 %20 %14
|
|
%82 = OpCopyObject %9 %27
|
|
OpBranch %37
|
|
%37 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%12 = OpFunction %6 None %10
|
|
%11 = OpFunctionParameter %9
|
|
%13 = OpLabel
|
|
%46 = OpCopyObject %9 %11
|
|
%16 = OpAccessChain %15 %11 %14
|
|
%95 = OpCopyObject %8 %80
|
|
OpReturnValue %21
|
|
%100 = OpLabel
|
|
OpUnreachable
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_4;
|
|
const auto consumer = nullptr;
|
|
const std::unique_ptr<opt::IRContext> context =
|
|
BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
spvtools::ValidatorOptions validator_options;
|
|
ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
|
|
kConsoleMessageConsumer));
|
|
|
|
opt::IRContext* ir_context = context.get();
|
|
// A void type with a result id of 2 can be found.
|
|
ASSERT_EQ(2, fuzzerutil::MaybeGetVoidType(ir_context));
|
|
}
|
|
|
|
TEST(FuzzerutilTest, FuzzerUtilMaybeGetZeroConstantTest) {
|
|
std::string shader = R"(
|
|
OpCapability Shader
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %4 "main" %56
|
|
OpExecutionMode %4 OriginUpperLeft
|
|
OpSource ESSL 310
|
|
OpName %4 "main"
|
|
OpName %8 "b1"
|
|
OpName %10 "b2"
|
|
OpName %12 "b3"
|
|
OpName %13 "b4"
|
|
OpName %16 "f1"
|
|
OpName %18 "f2"
|
|
OpName %22 "zc"
|
|
OpName %24 "i1"
|
|
OpName %28 "i2"
|
|
OpName %30 "i"
|
|
OpName %32 "i3"
|
|
OpName %34 "i4"
|
|
OpName %39 "f_arr"
|
|
OpName %49 "i_arr"
|
|
OpName %56 "value"
|
|
OpDecorate %22 RelaxedPrecision
|
|
OpDecorate %24 RelaxedPrecision
|
|
OpDecorate %28 RelaxedPrecision
|
|
OpDecorate %30 RelaxedPrecision
|
|
OpDecorate %32 RelaxedPrecision
|
|
OpDecorate %34 RelaxedPrecision
|
|
OpDecorate %49 RelaxedPrecision
|
|
OpDecorate %56 Location 0
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%6 = OpTypeBool
|
|
%7 = OpTypePointer Function %6
|
|
%9 = OpConstantTrue %6
|
|
%11 = OpConstantFalse %6
|
|
%14 = OpTypeFloat 32
|
|
%15 = OpTypePointer Function %14
|
|
%17 = OpConstant %14 1.23000002
|
|
%19 = OpConstant %14 1.11000001
|
|
%20 = OpTypeInt 32 1
|
|
%21 = OpTypePointer Function %20
|
|
%23 = OpConstant %20 0
|
|
%25 = OpConstant %20 1
|
|
%26 = OpTypeInt 32 0
|
|
%27 = OpTypePointer Function %26
|
|
%29 = OpConstant %26 100
|
|
%31 = OpConstant %26 0
|
|
%33 = OpConstant %20 -1
|
|
%35 = OpConstant %20 -99
|
|
%36 = OpConstant %26 5
|
|
%37 = OpTypeArray %14 %36
|
|
%38 = OpTypePointer Function %37
|
|
%40 = OpConstant %14 5.5
|
|
%41 = OpConstant %14 4.4000001
|
|
%42 = OpConstant %14 3.29999995
|
|
%43 = OpConstant %14 2.20000005
|
|
%44 = OpConstant %14 1.10000002
|
|
%45 = OpConstantComposite %37 %40 %41 %42 %43 %44
|
|
%46 = OpConstant %26 3
|
|
%47 = OpTypeArray %20 %46
|
|
%48 = OpTypePointer Function %47
|
|
%50 = OpConstant %20 3
|
|
%51 = OpConstant %20 7
|
|
%52 = OpConstant %20 9
|
|
%53 = OpConstantComposite %47 %50 %51 %52
|
|
%55 = OpTypePointer Input %14
|
|
%56 = OpVariable %55 Input
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%8 = OpVariable %7 Function
|
|
%10 = OpVariable %7 Function
|
|
%12 = OpVariable %7 Function
|
|
%13 = OpVariable %7 Function
|
|
%16 = OpVariable %15 Function
|
|
%18 = OpVariable %15 Function
|
|
%22 = OpVariable %21 Function
|
|
%24 = OpVariable %21 Function
|
|
%28 = OpVariable %27 Function
|
|
%30 = OpVariable %27 Function
|
|
%32 = OpVariable %21 Function
|
|
%34 = OpVariable %21 Function
|
|
%39 = OpVariable %38 Function
|
|
%49 = OpVariable %48 Function
|
|
OpStore %8 %9
|
|
OpStore %10 %11
|
|
OpStore %12 %9
|
|
OpStore %13 %11
|
|
OpStore %16 %17
|
|
OpStore %18 %19
|
|
OpStore %22 %23
|
|
OpStore %24 %25
|
|
OpStore %28 %29
|
|
OpStore %30 %31
|
|
OpStore %32 %33
|
|
OpStore %34 %35
|
|
OpStore %39 %45
|
|
OpStore %49 %53
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const auto env = SPV_ENV_UNIVERSAL_1_4;
|
|
const auto consumer = nullptr;
|
|
const std::unique_ptr<opt::IRContext> context =
|
|
BuildModule(env, consumer, shader, kFuzzAssembleOption);
|
|
spvtools::ValidatorOptions validator_options;
|
|
ASSERT_TRUE(fuzzerutil::IsValidAndWellFormed(context.get(), validator_options,
|
|
kConsoleMessageConsumer));
|
|
TransformationContext transformation_context(
|
|
MakeUnique<FactManager>(context.get()), validator_options);
|
|
|
|
opt::IRContext* ir_context = context.get();
|
|
|
|
// The id of a boolean constant will be returned give boolean type id 6.
|
|
uint32_t maybe_bool_id = fuzzerutil::MaybeGetZeroConstant(
|
|
ir_context, transformation_context, 6, false);
|
|
// The id of a 32 bit float constant will be returned given the float type
|
|
// id 14.
|
|
uint32_t maybe_float_id = fuzzerutil::MaybeGetZeroConstant(
|
|
ir_context, transformation_context, 14, false);
|
|
uint32_t maybe_signed_int_id = fuzzerutil::MaybeGetZeroConstant(
|
|
ir_context, transformation_context, 20, false);
|
|
uint32_t maybe_unsigned_int_id = fuzzerutil::MaybeGetZeroConstant(
|
|
ir_context, transformation_context, 26, false);
|
|
|
|
// Lists of possible ids for float, signed int, unsigned int and array.
|
|
std::vector<uint32_t> float_ids{17, 19};
|
|
std::vector<uint32_t> signed_int_ids{23, 25, 31, 33};
|
|
|
|
ASSERT_TRUE(maybe_bool_id == 9 || maybe_bool_id == 11);
|
|
ASSERT_TRUE(std::find(signed_int_ids.begin(), signed_int_ids.end(),
|
|
maybe_signed_int_id) != signed_int_ids.end());
|
|
|
|
// There is a unsigned int typed zero constant and its id is 31.
|
|
ASSERT_EQ(31, maybe_unsigned_int_id);
|
|
|
|
// There is no zero float constant.
|
|
ASSERT_TRUE(std::find(float_ids.begin(), float_ids.end(), maybe_float_id) ==
|
|
float_ids.end());
|
|
}
|
|
|
|
TEST(FuzzerutilTest, TypesAreCompatible) {
|
|
const 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
|
|
%9 = OpTypeInt 32 0
|
|
%8 = OpTypeStruct %6
|
|
%10 = OpTypePointer StorageBuffer %8
|
|
%11 = OpVariable %10 StorageBuffer
|
|
%86 = OpTypeStruct %9
|
|
%87 = OpTypePointer Workgroup %86
|
|
%88 = OpVariable %87 Workgroup
|
|
%89 = OpTypePointer Workgroup %9
|
|
%19 = OpConstant %9 0
|
|
%18 = OpConstant %9 1
|
|
%12 = OpConstant %6 0
|
|
%13 = OpTypePointer StorageBuffer %6
|
|
%15 = OpConstant %6 2
|
|
%16 = OpConstant %6 7
|
|
%20 = OpConstant %9 64
|
|
%4 = OpFunction %2 None %3
|
|
%5 = OpLabel
|
|
%14 = OpAccessChain %13 %11 %12
|
|
%90 = OpAccessChain %89 %88 %19
|
|
%21 = OpAtomicLoad %6 %14 %15 %20
|
|
%22 = OpAtomicExchange %6 %14 %15 %20 %16
|
|
%23 = OpAtomicCompareExchange %6 %14 %15 %20 %12 %16 %15
|
|
%24 = OpAtomicIIncrement %6 %14 %15 %20
|
|
%25 = OpAtomicIDecrement %6 %14 %15 %20
|
|
%26 = OpAtomicIAdd %6 %14 %15 %20 %16
|
|
%27 = OpAtomicISub %6 %14 %15 %20 %16
|
|
%28 = OpAtomicSMin %6 %14 %15 %20 %16
|
|
%29 = OpAtomicUMin %9 %90 %15 %20 %18
|
|
%30 = OpAtomicSMax %6 %14 %15 %20 %15
|
|
%31 = OpAtomicUMax %9 %90 %15 %20 %18
|
|
%32 = OpAtomicAnd %6 %14 %15 %20 %16
|
|
%33 = OpAtomicOr %6 %14 %15 %20 %16
|
|
%34 = OpAtomicXor %6 %14 %15 %20 %16
|
|
OpAtomicStore %14 %15 %20 %16
|
|
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));
|
|
|
|
const uint32_t int_type = 6; // The id of OpTypeInt 32 1
|
|
const uint32_t uint_type = 9; // The id of OpTypeInt 32 0
|
|
|
|
// OpAtomicLoad
|
|
#ifndef NDEBUG
|
|
ASSERT_DEATH(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicLoad, 0,
|
|
int_type, uint_type),
|
|
"Signedness check should not occur on a pointer operand.");
|
|
#endif
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicLoad, 1,
|
|
int_type, uint_type));
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicLoad, 2,
|
|
int_type, uint_type));
|
|
|
|
// OpAtomicExchange
|
|
#ifndef NDEBUG
|
|
ASSERT_DEATH(fuzzerutil::TypesAreCompatible(
|
|
context.get(), SpvOpAtomicExchange, 0, int_type, uint_type),
|
|
"Signedness check should not occur on a pointer operand.");
|
|
#endif
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicExchange,
|
|
1, int_type, uint_type));
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicExchange,
|
|
2, int_type, uint_type));
|
|
ASSERT_FALSE(fuzzerutil::TypesAreCompatible(
|
|
context.get(), SpvOpAtomicExchange, 3, int_type, uint_type));
|
|
|
|
// OpAtomicStore
|
|
#ifndef NDEBUG
|
|
ASSERT_DEATH(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicStore,
|
|
0, int_type, uint_type),
|
|
"Signedness check should not occur on a pointer operand.");
|
|
#endif
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicStore, 1,
|
|
int_type, uint_type));
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicStore, 2,
|
|
int_type, uint_type));
|
|
ASSERT_FALSE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicStore,
|
|
3, int_type, uint_type));
|
|
|
|
// OpAtomicCompareExchange
|
|
#ifndef NDEBUG
|
|
ASSERT_DEATH(
|
|
fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicCompareExchange,
|
|
0, int_type, uint_type),
|
|
"Signedness check should not occur on a pointer operand.");
|
|
#endif
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(
|
|
context.get(), SpvOpAtomicCompareExchange, 1, int_type, uint_type));
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(
|
|
context.get(), SpvOpAtomicCompareExchange, 2, int_type, uint_type));
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(
|
|
context.get(), SpvOpAtomicCompareExchange, 3, int_type, uint_type));
|
|
ASSERT_FALSE(fuzzerutil::TypesAreCompatible(
|
|
context.get(), SpvOpAtomicCompareExchange, 4, int_type, uint_type));
|
|
|
|
// OpAtomicIIncrement
|
|
#ifndef NDEBUG
|
|
ASSERT_DEATH(
|
|
fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicIIncrement, 0,
|
|
int_type, uint_type),
|
|
"Signedness check should not occur on a pointer operand.");
|
|
#endif
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(
|
|
context.get(), SpvOpAtomicIIncrement, 1, int_type, uint_type));
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(
|
|
context.get(), SpvOpAtomicIIncrement, 2, int_type, uint_type));
|
|
|
|
// OpAtomicIDecrement
|
|
#ifndef NDEBUG
|
|
ASSERT_DEATH(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicStore,
|
|
0, int_type, uint_type),
|
|
"Signedness check should not occur on a pointer operand.");
|
|
#endif
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicStore, 1,
|
|
int_type, uint_type));
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicStore, 2,
|
|
int_type, uint_type));
|
|
|
|
// OpAtomicIAdd
|
|
#ifndef NDEBUG
|
|
ASSERT_DEATH(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicIAdd, 0,
|
|
int_type, uint_type),
|
|
"Signedness check should not occur on a pointer operand.");
|
|
#endif
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicIAdd, 1,
|
|
int_type, uint_type));
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicIAdd, 2,
|
|
int_type, uint_type));
|
|
ASSERT_FALSE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicIAdd, 3,
|
|
int_type, uint_type));
|
|
|
|
// OpAtomicISub
|
|
#ifndef NDEBUG
|
|
ASSERT_DEATH(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicISub, 0,
|
|
int_type, uint_type),
|
|
"Signedness check should not occur on a pointer operand.");
|
|
#endif
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicISub, 1,
|
|
int_type, uint_type));
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicISub, 2,
|
|
int_type, uint_type));
|
|
ASSERT_FALSE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicISub, 3,
|
|
int_type, uint_type));
|
|
|
|
// OpAtomicSMin
|
|
#ifndef NDEBUG
|
|
ASSERT_DEATH(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicSMin, 0,
|
|
int_type, uint_type),
|
|
"Signedness check should not occur on a pointer operand.");
|
|
#endif
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicSMin, 1,
|
|
int_type, uint_type));
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicSMin, 2,
|
|
int_type, uint_type));
|
|
ASSERT_FALSE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicSMin, 3,
|
|
int_type, uint_type));
|
|
|
|
// OpAtomicUMin
|
|
#ifndef NDEBUG
|
|
ASSERT_DEATH(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicUMin, 0,
|
|
int_type, uint_type),
|
|
"Signedness check should not occur on a pointer operand.");
|
|
#endif
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicUMin, 1,
|
|
int_type, uint_type));
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicUMin, 2,
|
|
int_type, uint_type));
|
|
ASSERT_FALSE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicUMin, 3,
|
|
int_type, uint_type));
|
|
|
|
// OpAtomicSMax
|
|
#ifndef NDEBUG
|
|
ASSERT_DEATH(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicSMax, 0,
|
|
int_type, uint_type),
|
|
"Signedness check should not occur on a pointer operand.");
|
|
#endif
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicSMax, 1,
|
|
int_type, uint_type));
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicSMax, 2,
|
|
int_type, uint_type));
|
|
ASSERT_FALSE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicSMax, 3,
|
|
int_type, uint_type));
|
|
|
|
// OpAtomicUMax
|
|
#ifndef NDEBUG
|
|
ASSERT_DEATH(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicUMax, 0,
|
|
int_type, uint_type),
|
|
"Signedness check should not occur on a pointer operand.");
|
|
#endif
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicUMax, 1,
|
|
int_type, uint_type));
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicUMax, 2,
|
|
int_type, uint_type));
|
|
ASSERT_FALSE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicUMax, 3,
|
|
int_type, uint_type));
|
|
|
|
// OpAtomicAnd
|
|
#ifndef NDEBUG
|
|
ASSERT_DEATH(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicAnd, 0,
|
|
int_type, uint_type),
|
|
"Signedness check should not occur on a pointer operand.");
|
|
#endif
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicAnd, 1,
|
|
int_type, uint_type));
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicAnd, 2,
|
|
int_type, uint_type));
|
|
ASSERT_FALSE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicAnd, 3,
|
|
int_type, uint_type));
|
|
|
|
// OpAtomicOr
|
|
#ifndef NDEBUG
|
|
ASSERT_DEATH(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicOr, 0,
|
|
int_type, uint_type),
|
|
"Signedness check should not occur on a pointer operand.");
|
|
#endif
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicOr, 1,
|
|
int_type, uint_type));
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicOr, 2,
|
|
int_type, uint_type));
|
|
ASSERT_FALSE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicOr, 3,
|
|
int_type, uint_type));
|
|
|
|
// OpAtomicXor
|
|
#ifndef NDEBUG
|
|
ASSERT_DEATH(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicXor, 0,
|
|
int_type, uint_type),
|
|
"Signedness check should not occur on a pointer operand.");
|
|
#endif
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicXor, 1,
|
|
int_type, uint_type));
|
|
ASSERT_TRUE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicXor, 2,
|
|
int_type, uint_type));
|
|
ASSERT_FALSE(fuzzerutil::TypesAreCompatible(context.get(), SpvOpAtomicXor, 3,
|
|
int_type, uint_type));
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace fuzz
|
|
} // namespace spvtools
|