| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511 |
- // Copyright (c) 2017 Google Inc.
- //
- // 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 <string>
- #include <vector>
- #include "gmock/gmock.h"
- #include "spirv-tools/libspirv.hpp"
- #include "spirv-tools/optimizer.hpp"
- #include "test/opt/pass_fixture.h"
- namespace spvtools {
- namespace opt {
- namespace {
- using ::testing::Eq;
- // Return a string that contains the minimum instructions needed to form
- // a valid module. Other instructions can be appended to this string.
- std::string Header() {
- return R"(OpCapability Shader
- OpCapability Linkage
- OpMemoryModel Logical GLSL450
- )";
- }
- TEST(Optimizer, CanRunNullPassWithDistinctInputOutputVectors) {
- SpirvTools tools(SPV_ENV_UNIVERSAL_1_0);
- std::vector<uint32_t> binary_in;
- tools.Assemble(Header() + "OpName %foo \"foo\"\n%foo = OpTypeVoid",
- &binary_in);
- Optimizer opt(SPV_ENV_UNIVERSAL_1_0);
- opt.RegisterPass(CreateNullPass());
- std::vector<uint32_t> binary_out;
- opt.Run(binary_in.data(), binary_in.size(), &binary_out);
- std::string disassembly;
- tools.Disassemble(binary_out.data(), binary_out.size(), &disassembly);
- EXPECT_THAT(disassembly,
- Eq(Header() + "OpName %foo \"foo\"\n%foo = OpTypeVoid\n"));
- }
- TEST(Optimizer, CanRunTransformingPassWithDistinctInputOutputVectors) {
- SpirvTools tools(SPV_ENV_UNIVERSAL_1_0);
- std::vector<uint32_t> binary_in;
- tools.Assemble(Header() + "OpName %foo \"foo\"\n%foo = OpTypeVoid",
- &binary_in);
- Optimizer opt(SPV_ENV_UNIVERSAL_1_0);
- opt.RegisterPass(CreateStripDebugInfoPass());
- std::vector<uint32_t> binary_out;
- opt.Run(binary_in.data(), binary_in.size(), &binary_out);
- std::string disassembly;
- tools.Disassemble(binary_out.data(), binary_out.size(), &disassembly);
- EXPECT_THAT(disassembly, Eq(Header() + "%void = OpTypeVoid\n"));
- }
- TEST(Optimizer, CanRunNullPassWithAliasedVectors) {
- SpirvTools tools(SPV_ENV_UNIVERSAL_1_0);
- std::vector<uint32_t> binary;
- tools.Assemble("OpName %foo \"foo\"\n%foo = OpTypeVoid", &binary);
- Optimizer opt(SPV_ENV_UNIVERSAL_1_0);
- opt.RegisterPass(CreateNullPass());
- opt.Run(binary.data(), binary.size(), &binary); // This is the key.
- std::string disassembly;
- tools.Disassemble(binary.data(), binary.size(), &disassembly);
- EXPECT_THAT(disassembly, Eq("OpName %foo \"foo\"\n%foo = OpTypeVoid\n"));
- }
- TEST(Optimizer, CanRunNullPassWithAliasedVectorDataButDifferentSize) {
- SpirvTools tools(SPV_ENV_UNIVERSAL_1_0);
- std::vector<uint32_t> binary;
- tools.Assemble(Header() + "OpName %foo \"foo\"\n%foo = OpTypeVoid", &binary);
- Optimizer opt(SPV_ENV_UNIVERSAL_1_0);
- opt.RegisterPass(CreateNullPass());
- auto orig_size = binary.size();
- // Now change the size. Add a word that will be ignored
- // by the optimizer.
- binary.push_back(42);
- EXPECT_THAT(orig_size + 1, Eq(binary.size()));
- opt.Run(binary.data(), orig_size, &binary); // This is the key.
- // The binary vector should have been rewritten.
- EXPECT_THAT(binary.size(), Eq(orig_size));
- std::string disassembly;
- tools.Disassemble(binary.data(), binary.size(), &disassembly);
- EXPECT_THAT(disassembly,
- Eq(Header() + "OpName %foo \"foo\"\n%foo = OpTypeVoid\n"));
- }
- TEST(Optimizer, CanRunTransformingPassWithAliasedVectors) {
- SpirvTools tools(SPV_ENV_UNIVERSAL_1_0);
- std::vector<uint32_t> binary;
- tools.Assemble(Header() + "OpName %foo \"foo\"\n%foo = OpTypeVoid", &binary);
- Optimizer opt(SPV_ENV_UNIVERSAL_1_0);
- opt.RegisterPass(CreateStripDebugInfoPass());
- opt.Run(binary.data(), binary.size(), &binary); // This is the key
- std::string disassembly;
- tools.Disassemble(binary.data(), binary.size(), &disassembly);
- EXPECT_THAT(disassembly, Eq(Header() + "%void = OpTypeVoid\n"));
- }
- TEST(Optimizer, CanValidateFlags) {
- Optimizer opt(SPV_ENV_UNIVERSAL_1_0);
- EXPECT_FALSE(opt.FlagHasValidForm("bad-flag"));
- EXPECT_TRUE(opt.FlagHasValidForm("-O"));
- EXPECT_TRUE(opt.FlagHasValidForm("-Os"));
- EXPECT_FALSE(opt.FlagHasValidForm("-O2"));
- EXPECT_TRUE(opt.FlagHasValidForm("--this_flag"));
- }
- TEST(Optimizer, CanRegisterPassesFromFlags) {
- SpirvTools tools(SPV_ENV_UNIVERSAL_1_0);
- Optimizer opt(SPV_ENV_UNIVERSAL_1_0);
- spv_message_level_t msg_level;
- const char* msg_fname;
- spv_position_t msg_position;
- const char* msg;
- auto examine_message = [&msg_level, &msg_fname, &msg_position, &msg](
- spv_message_level_t ml, const char* f,
- const spv_position_t& p, const char* m) {
- msg_level = ml;
- msg_fname = f;
- msg_position = p;
- msg = m;
- };
- opt.SetMessageConsumer(examine_message);
- std::vector<std::string> pass_flags = {
- "--strip-debug",
- "--strip-nonsemantic",
- "--set-spec-const-default-value=23:42 21:12",
- "--if-conversion",
- "--freeze-spec-const",
- "--inline-entry-points-exhaustive",
- "--inline-entry-points-opaque",
- "--convert-local-access-chains",
- "--eliminate-dead-code-aggressive",
- "--eliminate-insert-extract",
- "--eliminate-local-single-block",
- "--eliminate-local-single-store",
- "--merge-blocks",
- "--merge-return",
- "--eliminate-dead-branches",
- "--eliminate-dead-functions",
- "--eliminate-local-multi-store",
- "--eliminate-dead-const",
- "--eliminate-dead-inserts",
- "--eliminate-dead-variables",
- "--fold-spec-const-op-composite",
- "--loop-unswitch",
- "--scalar-replacement=300",
- "--scalar-replacement",
- "--strength-reduction",
- "--unify-const",
- "--flatten-decorations",
- "--compact-ids",
- "--cfg-cleanup",
- "--local-redundancy-elimination",
- "--loop-invariant-code-motion",
- "--reduce-load-size",
- "--redundancy-elimination",
- "--private-to-local",
- "--remove-duplicates",
- "--workaround-1209",
- "--replace-invalid-opcode",
- "--simplify-instructions",
- "--ssa-rewrite",
- "--copy-propagate-arrays",
- "--loop-fission=20",
- "--loop-fusion=2",
- "--loop-unroll",
- "--vector-dce",
- "--loop-unroll-partial=3",
- "--loop-peeling",
- "--ccp",
- "-O",
- "-Os",
- "--legalize-hlsl"};
- EXPECT_TRUE(opt.RegisterPassesFromFlags(pass_flags));
- // Test some invalid flags.
- EXPECT_FALSE(opt.RegisterPassFromFlag("-O2"));
- EXPECT_EQ(msg_level, SPV_MSG_ERROR);
- EXPECT_FALSE(opt.RegisterPassFromFlag("-loop-unroll"));
- EXPECT_EQ(msg_level, SPV_MSG_ERROR);
- EXPECT_FALSE(opt.RegisterPassFromFlag("--set-spec-const-default-value"));
- EXPECT_EQ(msg_level, SPV_MSG_ERROR);
- EXPECT_FALSE(opt.RegisterPassFromFlag("--scalar-replacement=s"));
- EXPECT_EQ(msg_level, SPV_MSG_ERROR);
- EXPECT_FALSE(opt.RegisterPassFromFlag("--loop-fission=-4"));
- EXPECT_EQ(msg_level, SPV_MSG_ERROR);
- EXPECT_FALSE(opt.RegisterPassFromFlag("--loop-fusion=xx"));
- EXPECT_EQ(msg_level, SPV_MSG_ERROR);
- EXPECT_FALSE(opt.RegisterPassFromFlag("--loop-unroll-partial"));
- EXPECT_EQ(msg_level, SPV_MSG_ERROR);
- }
- TEST(Optimizer, RemoveNop) {
- // Test that OpNops are removed even if no optimizations are run.
- const std::string before = R"(OpCapability Shader
- OpCapability Linkage
- OpMemoryModel Logical GLSL450
- %void = OpTypeVoid
- %2 = OpTypeFunction %void
- %3 = OpFunction %void None %2
- %4 = OpLabel
- OpNop
- OpReturn
- OpFunctionEnd
- )";
- const std::string after = R"(OpCapability Shader
- OpCapability Linkage
- OpMemoryModel Logical GLSL450
- %void = OpTypeVoid
- %2 = OpTypeFunction %void
- %3 = OpFunction %void None %2
- %4 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::vector<uint32_t> binary;
- {
- SpirvTools tools(SPV_ENV_VULKAN_1_1);
- tools.Assemble(before, &binary);
- }
- Optimizer opt(SPV_ENV_VULKAN_1_1);
- std::vector<uint32_t> optimized;
- class ValidatorOptions validator_options;
- ASSERT_TRUE(opt.Run(binary.data(), binary.size(), &optimized,
- validator_options, true))
- << before << "\n";
- std::string disassembly;
- {
- SpirvTools tools(SPV_ENV_VULKAN_1_1);
- tools.Disassemble(optimized.data(), optimized.size(), &disassembly);
- }
- EXPECT_EQ(after, disassembly)
- << "Was expecting the OpNop to have been removed.";
- }
- TEST(Optimizer, AvoidIntegrityCheckForExtraLineInfo) {
- // Test that it avoids the integrity check when no optimizations are run and
- // OpLines are propagated.
- const std::string before = R"(OpCapability Shader
- OpCapability Linkage
- OpMemoryModel Logical GLSL450
- %1 = OpString "Test"
- %void = OpTypeVoid
- %3 = OpTypeFunction %void
- %uint = OpTypeInt 32 0
- %_ptr_Function_uint = OpTypePointer Function %uint
- %6 = OpFunction %void None %3
- %7 = OpLabel
- OpLine %1 10 0
- %8 = OpVariable %_ptr_Function_uint Function
- OpLine %1 10 0
- %9 = OpVariable %_ptr_Function_uint Function
- OpLine %1 20 0
- OpReturn
- OpFunctionEnd
- )";
- const std::string after = R"(OpCapability Shader
- OpCapability Linkage
- OpMemoryModel Logical GLSL450
- %1 = OpString "Test"
- %void = OpTypeVoid
- %3 = OpTypeFunction %void
- %uint = OpTypeInt 32 0
- %_ptr_Function_uint = OpTypePointer Function %uint
- %6 = OpFunction %void None %3
- %7 = OpLabel
- OpLine %1 10 0
- %8 = OpVariable %_ptr_Function_uint Function
- %9 = OpVariable %_ptr_Function_uint Function
- OpLine %1 20 0
- OpReturn
- OpFunctionEnd
- )";
- std::vector<uint32_t> binary;
- SpirvTools tools(SPV_ENV_VULKAN_1_1);
- tools.Assemble(before, &binary);
- Optimizer opt(SPV_ENV_VULKAN_1_1);
- std::vector<uint32_t> optimized;
- class ValidatorOptions validator_options;
- ASSERT_TRUE(opt.Run(binary.data(), binary.size(), &optimized,
- validator_options, true))
- << before << "\n";
- std::string disassembly;
- tools.Disassemble(optimized.data(), optimized.size(), &disassembly);
- EXPECT_EQ(after, disassembly)
- << "Was expecting the OpLine to have been propagated.";
- }
- TEST(Optimizer, AvoidIntegrityCheckForDebugScope) {
- // Test that it avoids the integrity check when the code contains DebugScope.
- const std::string before = R"(OpCapability Shader
- %1 = OpExtInstImport "OpenCL.DebugInfo.100"
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- OpExecutionMode %main OriginUpperLeft
- %3 = OpString "simple_vs.hlsl"
- OpSource HLSL 600 %3
- OpName %main "main"
- %void = OpTypeVoid
- %5 = OpTypeFunction %void
- %6 = OpExtInst %void %1 DebugSource %3
- %7 = OpExtInst %void %1 DebugCompilationUnit 2 4 %6 HLSL
- %main = OpFunction %void None %5
- %14 = OpLabel
- %26 = OpExtInst %void %1 DebugScope %7
- OpReturn
- %27 = OpExtInst %void %1 DebugNoScope
- OpFunctionEnd
- )";
- const std::string after = R"(OpCapability Shader
- %1 = OpExtInstImport "OpenCL.DebugInfo.100"
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- OpExecutionMode %main OriginUpperLeft
- %3 = OpString "simple_vs.hlsl"
- OpSource HLSL 600 %3
- OpName %main "main"
- %void = OpTypeVoid
- %5 = OpTypeFunction %void
- %6 = OpExtInst %void %1 DebugSource %3
- %7 = OpExtInst %void %1 DebugCompilationUnit 2 4 %6 HLSL
- %main = OpFunction %void None %5
- %8 = OpLabel
- %11 = OpExtInst %void %1 DebugScope %7
- OpReturn
- %12 = OpExtInst %void %1 DebugNoScope
- OpFunctionEnd
- )";
- std::vector<uint32_t> binary;
- SpirvTools tools(SPV_ENV_VULKAN_1_1);
- tools.Assemble(before, &binary);
- Optimizer opt(SPV_ENV_VULKAN_1_1);
- std::vector<uint32_t> optimized;
- ASSERT_TRUE(opt.Run(binary.data(), binary.size(), &optimized))
- << before << "\n";
- std::string disassembly;
- tools.Disassemble(optimized.data(), optimized.size(), &disassembly);
- EXPECT_EQ(after, disassembly)
- << "Was expecting the result id of DebugScope to have been changed.";
- }
- TEST(Optimizer, CheckDefaultPerformancePassesLargeStructScalarization) {
- std::string start = R"(OpCapability Shader
- %1 = OpExtInstImport "GLSL.std.450"
- OpMemoryModel Logical GLSL450
- OpEntryPoint Vertex %4 "main" %46 %48
- OpSource GLSL 430
- OpName %4 "main"
- OpDecorate %44 Block
- OpMemberDecorate %44 0 BuiltIn Position
- OpMemberDecorate %44 1 BuiltIn PointSize
- OpMemberDecorate %44 2 BuiltIn ClipDistance
- OpDecorate %48 Location 0
- %2 = OpTypeVoid
- %3 = OpTypeFunction %2
- %6 = OpTypeFloat 32
- %7 = OpTypeVector %6 4
- %8 = OpTypePointer Function %7
- %9 = OpTypeStruct %7)";
- // add 200 float members to the struct
- for (int i = 0; i < 200; i++) {
- start += " %6";
- }
- start += R"(
- %10 = OpTypeFunction %9 %8
- %14 = OpTypeFunction %6 %9
- %18 = OpTypePointer Function %9
- %20 = OpTypeInt 32 1
- %21 = OpConstant %20 0
- %24 = OpConstant %20 1
- %25 = OpTypeInt 32 0
- %26 = OpConstant %25 1
- %27 = OpTypePointer Function %6
- %43 = OpTypeArray %6 %26
- %44 = OpTypeStruct %7 %6 %43
- %45 = OpTypePointer Output %44
- %46 = OpVariable %45 Output
- %47 = OpTypePointer Input %7
- %48 = OpVariable %47 Input
- %54 = OpTypePointer Output %7
- %4 = OpFunction %2 None %3
- %5 = OpLabel
- %49 = OpVariable %8 Function
- %50 = OpLoad %7 %48
- OpStore %49 %50
- %51 = OpFunctionCall %9 %12 %49
- %52 = OpFunctionCall %6 %16 %51
- %53 = OpCompositeConstruct %7 %52 %52 %52 %52
- %55 = OpAccessChain %54 %46 %21
- OpStore %55 %53
- OpReturn
- OpFunctionEnd
- %12 = OpFunction %9 None %10
- %11 = OpFunctionParameter %8
- %13 = OpLabel
- %19 = OpVariable %18 Function
- %22 = OpLoad %7 %11
- %23 = OpAccessChain %8 %19 %21
- OpStore %23 %22
- %28 = OpAccessChain %27 %11 %26
- %29 = OpLoad %6 %28
- %30 = OpConvertFToS %20 %29
- %31 = OpAccessChain %27 %19 %21 %30
- %32 = OpLoad %6 %31
- %33 = OpAccessChain %27 %19 %24
- OpStore %33 %32
- %34 = OpLoad %9 %19
- OpReturnValue %34
- OpFunctionEnd
- %16 = OpFunction %6 None %14
- %15 = OpFunctionParameter %9
- %17 = OpLabel
- %37 = OpCompositeExtract %6 %15 1
- %38 = OpConvertFToS %20 %37
- %39 = OpCompositeExtract %7 %15 0
- %40 = OpVectorExtractDynamic %6 %39 %38
- OpReturnValue %40
- OpFunctionEnd)";
- std::vector<uint32_t> binary;
- SpirvTools tools(SPV_ENV_VULKAN_1_3);
- tools.Assemble(start, &binary,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- std::string test_disassembly;
- std::string default_disassembly;
- {
- Optimizer opt(SPV_ENV_VULKAN_1_3);
- opt.RegisterPerformancePasses();
- std::vector<uint32_t> optimized;
- ASSERT_TRUE(opt.Run(binary.data(), binary.size(), &optimized))
- << start << "\n";
- tools.Disassemble(optimized.data(), optimized.size(), &default_disassembly,
- SPV_BINARY_TO_TEXT_OPTION_NO_HEADER);
- }
- {
- // default passes should not benefit from additional scalar replacement
- Optimizer opt(SPV_ENV_VULKAN_1_3);
- opt.RegisterPerformancePasses()
- .RegisterPass(CreateScalarReplacementPass(201))
- .RegisterPass(CreateAggressiveDCEPass());
- std::vector<uint32_t> optimized;
- ASSERT_TRUE(opt.Run(binary.data(), binary.size(), &optimized))
- << start << "\n";
- tools.Disassemble(optimized.data(), optimized.size(), &test_disassembly,
- SPV_BINARY_TO_TEXT_OPTION_NO_HEADER);
- }
- EXPECT_EQ(test_disassembly, default_disassembly);
- }
- } // namespace
- } // namespace opt
- } // namespace spvtools
|