// 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 #include #include "gmock/gmock.h" #include "test/opt/assembly_builder.h" #include "test/opt/pass_fixture.h" #include "test/opt/pass_utils.h" namespace spvtools { namespace opt { namespace { using ::testing::HasSubstr; using EliminateDeadFunctionsBasicTest = PassTest<::testing::Test>; TEST_F(EliminateDeadFunctionsBasicTest, BasicDeleteDeadFunction) { // The function Dead should be removed because it is never called. const std::vector common_code = { // clang-format off "OpCapability Shader", "OpMemoryModel Logical GLSL450", "OpEntryPoint Fragment %main \"main\"", "OpName %main \"main\"", "OpName %Live \"Live\"", "%void = OpTypeVoid", "%7 = OpTypeFunction %void", "%main = OpFunction %void None %7", "%15 = OpLabel", "%16 = OpFunctionCall %void %Live", "%17 = OpFunctionCall %void %Live", "OpReturn", "OpFunctionEnd", "%Live = OpFunction %void None %7", "%20 = OpLabel", "OpReturn", "OpFunctionEnd" // clang-format on }; const std::vector dead_function = { // clang-format off "%Dead = OpFunction %void None %7", "%19 = OpLabel", "OpReturn", "OpFunctionEnd", // clang-format on }; SetAssembleOptions(SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS); SinglePassRunAndCheck( JoinAllInsts(Concat(common_code, dead_function)), JoinAllInsts(common_code), /* skip_nop = */ true); } TEST_F(EliminateDeadFunctionsBasicTest, BasicKeepLiveFunction) { // Everything is reachable from an entry point, so no functions should be // deleted. const std::vector text = { // clang-format off "OpCapability Shader", "OpMemoryModel Logical GLSL450", "OpEntryPoint Fragment %main \"main\"", "OpName %main \"main\"", "OpName %Live1 \"Live1\"", "OpName %Live2 \"Live2\"", "%void = OpTypeVoid", "%7 = OpTypeFunction %void", "%main = OpFunction %void None %7", "%15 = OpLabel", "%16 = OpFunctionCall %void %Live2", "%17 = OpFunctionCall %void %Live1", "OpReturn", "OpFunctionEnd", "%Live1 = OpFunction %void None %7", "%19 = OpLabel", "OpReturn", "OpFunctionEnd", "%Live2 = OpFunction %void None %7", "%20 = OpLabel", "OpReturn", "OpFunctionEnd" // clang-format on }; SetAssembleOptions(SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS); std::string assembly = JoinAllInsts(text); auto result = SinglePassRunAndDisassemble( assembly, /* skip_nop = */ true, /* do_validation = */ false); EXPECT_EQ(Pass::Status::SuccessWithoutChange, std::get<1>(result)); EXPECT_EQ(assembly, std::get<0>(result)); } TEST_F(EliminateDeadFunctionsBasicTest, BasicKeepExportFunctions) { // All functions are reachable. In particular, ExportedFunc and Constant are // reachable because ExportedFunc is exported. Nothing should be removed. const std::vector text = { // clang-format off "OpCapability Shader", "OpCapability Linkage", "OpMemoryModel Logical GLSL450", "OpEntryPoint Fragment %main \"main\"", "OpName %main \"main\"", "OpName %ExportedFunc \"ExportedFunc\"", "OpName %Live \"Live\"", "OpDecorate %ExportedFunc LinkageAttributes \"ExportedFunc\" Export", "%void = OpTypeVoid", "%7 = OpTypeFunction %void", "%main = OpFunction %void None %7", "%15 = OpLabel", "OpReturn", "OpFunctionEnd", "%ExportedFunc = OpFunction %void None %7", "%19 = OpLabel", "%16 = OpFunctionCall %void %Live", "OpReturn", "OpFunctionEnd", "%Live = OpFunction %void None %7", "%20 = OpLabel", "OpReturn", "OpFunctionEnd" // clang-format on }; SetAssembleOptions(SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS); std::string assembly = JoinAllInsts(text); auto result = SinglePassRunAndDisassemble( assembly, /* skip_nop = */ true, /* do_validation = */ false); EXPECT_EQ(Pass::Status::SuccessWithoutChange, std::get<1>(result)); EXPECT_EQ(assembly, std::get<0>(result)); } TEST_F(EliminateDeadFunctionsBasicTest, BasicRemoveDecorationsAndNames) { // We want to remove the names and decorations associated with results that // are removed. This test will check for that. const std::string text = R"( OpCapability Shader OpMemoryModel Logical GLSL450 OpEntryPoint Vertex %main "main" OpName %main "main" OpName %Dead "Dead" OpName %x "x" OpName %y "y" OpName %z "z" OpDecorate %x RelaxedPrecision OpDecorate %y RelaxedPrecision OpDecorate %z RelaxedPrecision OpDecorate %6 RelaxedPrecision OpDecorate %7 RelaxedPrecision OpDecorate %8 RelaxedPrecision %void = OpTypeVoid %10 = OpTypeFunction %void %float = OpTypeFloat 32 %_ptr_Function_float = OpTypePointer Function %float %float_1 = OpConstant %float 1 %main = OpFunction %void None %10 %14 = OpLabel OpReturn OpFunctionEnd %Dead = OpFunction %void None %10 %15 = OpLabel %x = OpVariable %_ptr_Function_float Function %y = OpVariable %_ptr_Function_float Function %z = OpVariable %_ptr_Function_float Function OpStore %x %float_1 OpStore %y %float_1 %6 = OpLoad %float %x %7 = OpLoad %float %y %8 = OpFAdd %float %6 %7 OpStore %z %8 OpReturn OpFunctionEnd)"; const std::string expected_output = R"(OpCapability Shader OpMemoryModel Logical GLSL450 OpEntryPoint Vertex %main "main" OpName %main "main" %void = OpTypeVoid %10 = OpTypeFunction %void %float = OpTypeFloat 32 %_ptr_Function_float = OpTypePointer Function %float %float_1 = OpConstant %float 1 %main = OpFunction %void None %10 %14 = OpLabel OpReturn OpFunctionEnd )"; SetAssembleOptions(SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS); SinglePassRunAndCheck(text, expected_output, /* skip_nop = */ true); } TEST_F(EliminateDeadFunctionsBasicTest, DebugRemoveFunctionFromDebugFunction) { // We want to remove id of OpFunction from DebugFunction. const std::string text = R"(OpCapability Shader %1 = OpExtInstImport "OpenCL.DebugInfo.100" OpMemoryModel Logical GLSL450 OpEntryPoint Fragment %2 "main" %3 %4 OpExecutionMode %2 OriginUpperLeft %5 = OpString "ps.hlsl" OpSource HLSL 600 %5 "float4 foo() { return 1; } float4 main(float4 color : COLOR) : SV_TARGET { return foo() + color; } " %6 = OpString "float" %7 = OpString "main" %8 = OpString "foo" ; CHECK: [[foo:%\d+]] = OpString "foo" OpDecorate %3 Location 0 OpDecorate %4 Location 0 %uint = OpTypeInt 32 0 %uint_32 = OpConstant %uint 32 %float = OpTypeFloat 32 %float_1 = OpConstant %float 1 %v4float = OpTypeVector %float 4 %14 = OpConstantComposite %v4float %float_1 %float_1 %float_1 %float_1 %_ptr_Input_v4float = OpTypePointer Input %v4float %_ptr_Output_v4float = OpTypePointer Output %v4float %void = OpTypeVoid %18 = OpTypeFunction %void %19 = OpTypeFunction %v4float %3 = OpVariable %_ptr_Input_v4float Input %4 = OpVariable %_ptr_Output_v4float Output %_ptr_Function_v4float = OpTypePointer Function %v4float ; CHECK: [[info_none:%\d+]] = OpExtInst %void %1 DebugInfoNone %20 = OpExtInst %void %1 DebugSource %5 %21 = OpExtInst %void %1 DebugCompilationUnit 1 4 %20 HLSL %22 = OpExtInst %void %1 DebugTypeBasic %6 %uint_32 Float %23 = OpExtInst %void %1 DebugTypeVector %22 4 %24 = OpExtInst %void %1 DebugTypeFunction FlagIsProtected|FlagIsPrivate %23 %23 %25 = OpExtInst %void %1 DebugTypeFunction FlagIsProtected|FlagIsPrivate %23 %26 = OpExtInst %void %1 DebugFunction %7 %24 %20 4 1 %21 %7 FlagIsProtected|FlagIsPrivate 4 %2 %27 = OpExtInst %void %1 DebugFunction %8 %25 %20 1 1 %21 %8 FlagIsProtected|FlagIsPrivate 1 %28 ; CHECK: {{%\d+}} = OpExtInst %void %1 DebugFunction [[foo]] {{%\d+}} {{%\d+}} 1 1 {{%\d+}} {{%\d+}} FlagIsProtected|FlagIsPrivate 1 [[info_none]] %29 = OpExtInst %void %1 DebugLexicalBlock %20 1 14 %27 %40 = OpExtInst %void %1 DebugInlinedAt 4 %26 %2 = OpFunction %void None %18 %30 = OpLabel %39 = OpVariable %_ptr_Function_v4float Function %41 = OpExtInst %void %1 DebugScope %27 %40 OpStore %39 %14 %32 = OpLoad %v4float %39 %42 = OpExtInst %void %1 DebugScope %26 %33 = OpLoad %v4float %3 %34 = OpFAdd %v4float %32 %33 OpStore %4 %34 %43 = OpExtInst %void %1 DebugNoScope OpReturn OpFunctionEnd %28 = OpFunction %v4float None %19 %36 = OpLabel OpReturnValue %14 OpFunctionEnd )"; SinglePassRunAndMatch(text, false); } TEST_F(EliminateDeadFunctionsBasicTest, DebugRemoveFunctionUsingExistingDebugInfoNone) { // We want to remove id of OpFunction from DebugFunction. const std::string text = R"(OpCapability Shader %1 = OpExtInstImport "OpenCL.DebugInfo.100" OpMemoryModel Logical GLSL450 OpEntryPoint Fragment %2 "main" %3 %4 OpExecutionMode %2 OriginUpperLeft %5 = OpString "ps.hlsl" OpSource HLSL 600 %5 "float4 foo() { return 1; } float4 main(float4 color : COLOR) : SV_TARGET { return foo() + color; } " %6 = OpString "float" %7 = OpString "main" %8 = OpString "foo" ; CHECK: [[foo:%\d+]] = OpString "foo" OpDecorate %3 Location 0 OpDecorate %4 Location 0 %uint = OpTypeInt 32 0 %uint_32 = OpConstant %uint 32 %float = OpTypeFloat 32 %float_1 = OpConstant %float 1 %v4float = OpTypeVector %float 4 %14 = OpConstantComposite %v4float %float_1 %float_1 %float_1 %float_1 %_ptr_Input_v4float = OpTypePointer Input %v4float %_ptr_Output_v4float = OpTypePointer Output %v4float %void = OpTypeVoid %18 = OpTypeFunction %void %19 = OpTypeFunction %v4float %3 = OpVariable %_ptr_Input_v4float Input %4 = OpVariable %_ptr_Output_v4float Output %_ptr_Function_v4float = OpTypePointer Function %v4float ; CHECK: [[info_none:%\d+]] = OpExtInst %void %1 DebugInfoNone %20 = OpExtInst %void %1 DebugSource %5 %21 = OpExtInst %void %1 DebugCompilationUnit 1 4 %20 HLSL %22 = OpExtInst %void %1 DebugTypeBasic %6 %uint_32 Float %23 = OpExtInst %void %1 DebugTypeVector %22 4 %24 = OpExtInst %void %1 DebugTypeFunction FlagIsProtected|FlagIsPrivate %23 %23 %25 = OpExtInst %void %1 DebugTypeFunction FlagIsProtected|FlagIsPrivate %23 %26 = OpExtInst %void %1 DebugFunction %7 %24 %20 4 1 %21 %7 FlagIsProtected|FlagIsPrivate 4 %2 %27 = OpExtInst %void %1 DebugFunction %8 %25 %20 1 1 %21 %8 FlagIsProtected|FlagIsPrivate 1 %28 ; CHECK: {{%\d+}} = OpExtInst %void %1 DebugFunction [[foo]] {{%\d+}} {{%\d+}} 1 1 {{%\d+}} {{%\d+}} FlagIsProtected|FlagIsPrivate 1 [[info_none]] %29 = OpExtInst %void %1 DebugLexicalBlock %20 1 14 %27 %35 = OpExtInst %void %1 DebugInfoNone %40 = OpExtInst %void %1 DebugInlinedAt 4 %26 %2 = OpFunction %void None %18 %30 = OpLabel %39 = OpVariable %_ptr_Function_v4float Function %41 = OpExtInst %void %1 DebugScope %27 %40 OpStore %39 %14 %32 = OpLoad %v4float %39 %42 = OpExtInst %void %1 DebugScope %26 %33 = OpLoad %v4float %3 %34 = OpFAdd %v4float %32 %33 OpStore %4 %34 %43 = OpExtInst %void %1 DebugNoScope OpReturn OpFunctionEnd %28 = OpFunction %v4float None %19 %36 = OpLabel OpReturnValue %14 OpFunctionEnd )"; SinglePassRunAndMatch(text, false); } } // namespace } // namespace opt } // namespace spvtools