// Copyright 2020 The Tint Authors. // // 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 "gmock/gmock.h" #include "src/tint/reader/spirv/function.h" #include "src/tint/reader/spirv/parser_impl_test_helper.h" #include "src/tint/reader/spirv/spirv_tools_helpers_test.h" #include "src/tint/utils/string_stream.h" namespace tint::reader::spirv { namespace { using ::testing::HasSubstr; std::string Preamble() { return R"( OpCapability Shader OpMemoryModel Logical Simple OpEntryPoint Fragment %100 "x_100" OpExecutionMode %100 OriginUpperLeft )"; } /// @returns a SPIR-V assembly segment which assigns debug names /// to particular IDs. std::string Names(std::vector ids) { utils::StringStream outs; for (auto& id : ids) { outs << " OpName %" << id << " \"" << id << "\"\n"; } return outs.str(); } std::string CommonTypes() { return R"( %void = OpTypeVoid %voidfn = OpTypeFunction %void %float = OpTypeFloat 32 %uint = OpTypeInt 32 0 %int = OpTypeInt 32 1 %float_0 = OpConstant %float 0.0 )"; } std::string CommonHandleTypes() { return CommonTypes() + R"( %v2float = OpTypeVector %float 2 %v4float = OpTypeVector %float 4 %v2_0 = OpConstantNull %v2float %sampler = OpTypeSampler %tex2d_f32 = OpTypeImage %float 2D 0 0 0 1 Unknown %sampled_image_2d_f32 = OpTypeSampledImage %tex2d_f32 %ptr_sampler = OpTypePointer UniformConstant %sampler %ptr_tex2d_f32 = OpTypePointer UniformConstant %tex2d_f32 )"; } std::string MainBody() { return R"( %100 = OpFunction %void None %voidfn %entry_100 = OpLabel OpReturn OpFunctionEnd )"; } TEST_F(SpvParserTest, Emit_VoidFunctionWithoutParams) { auto p = parser(test::Assemble(Preamble() + CommonTypes() + R"( %100 = OpFunction %void None %voidfn %entry = OpLabel OpReturn OpFunctionEnd )")); ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()); auto fe = p->function_emitter(100); EXPECT_TRUE(fe.Emit()); auto got = test::ToString(p->program()); std::string expect = R"(fn x_100() { return; } )"; EXPECT_EQ(got, expect); } TEST_F(SpvParserTest, Emit_NonVoidResultType) { auto p = parser(test::Assemble(Preamble() + CommonTypes() + R"( %fn_ret_float = OpTypeFunction %float %200 = OpFunction %float None %fn_ret_float %entry = OpLabel OpReturnValue %float_0 OpFunctionEnd )" + MainBody())); ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()); auto fe = p->function_emitter(200); EXPECT_TRUE(fe.Emit()); auto got = test::ToString(p->program()); std::string expect = R"(fn x_200() -> f32 { return 0.0f; } )"; EXPECT_THAT(got, HasSubstr(expect)); } TEST_F(SpvParserTest, Emit_MixedParamTypes) { auto p = parser(test::Assemble(Preamble() + Names({"a", "b", "c"}) + CommonTypes() + R"( %fn_mixed_params = OpTypeFunction %void %uint %float %int %200 = OpFunction %void None %fn_mixed_params %a = OpFunctionParameter %uint %b = OpFunctionParameter %float %c = OpFunctionParameter %int %mixed_entry = OpLabel OpReturn OpFunctionEnd )" + MainBody())); ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()); auto fe = p->function_emitter(200); EXPECT_TRUE(fe.Emit()); auto got = test::ToString(p->program()); std::string expect = R"(fn x_200(a : u32, b : f32, c : i32) { return; } )"; EXPECT_THAT(got, HasSubstr(expect)); } TEST_F(SpvParserTest, Emit_GenerateParamNames) { auto p = parser(test::Assemble(Preamble() + CommonTypes() + R"( %fn_mixed_params = OpTypeFunction %void %uint %float %int %200 = OpFunction %void None %fn_mixed_params %14 = OpFunctionParameter %uint %15 = OpFunctionParameter %float %16 = OpFunctionParameter %int %mixed_entry = OpLabel OpReturn OpFunctionEnd )" + MainBody())); ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()); auto fe = p->function_emitter(200); EXPECT_TRUE(fe.Emit()); auto got = test::ToString(p->program()); std::string expect = R"(fn x_200(x_14 : u32, x_15 : f32, x_16 : i32) { return; } )"; EXPECT_THAT(got, HasSubstr(expect)); } TEST_F(SpvParserTest, Emit_FunctionDecl_ParamPtrTexture_ParamPtrSampler) { auto p = parser(test::Assemble(Preamble() + CommonHandleTypes() + R"( ; This is how Glslang generates functions that take texture and sampler arguments. ; It passes them by pointer. %fn_ty = OpTypeFunction %void %ptr_tex2d_f32 %ptr_sampler %200 = OpFunction %void None %fn_ty %14 = OpFunctionParameter %ptr_tex2d_f32 %15 = OpFunctionParameter %ptr_sampler %mixed_entry = OpLabel ; access the texture, to give the handles usages. %im = OpLoad %tex2d_f32 %14 %sam = OpLoad %sampler %15 %imsam = OpSampledImage %sampled_image_2d_f32 %im %sam %20 = OpImageSampleImplicitLod %v4float %imsam %v2_0 OpReturn OpFunctionEnd )" + MainBody())); ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()); auto fe = p->function_emitter(200); EXPECT_TRUE(fe.Emit()); auto got = test::ToString(p->program()); std::string expect = R"(fn x_200(x_14 : texture_2d, x_15 : sampler) { let x_20 : vec4 = textureSample(x_14, x_15, vec2()); return; } )"; EXPECT_EQ(got, expect); } TEST_F(SpvParserTest, Emit_FunctionDecl_ParamTexture_ParamSampler) { auto assembly = Preamble() + CommonHandleTypes() + R"( ; It is valid in SPIR-V to pass textures and samplers by value. %fn_ty = OpTypeFunction %void %tex2d_f32 %sampler %200 = OpFunction %void None %fn_ty %14 = OpFunctionParameter %tex2d_f32 %15 = OpFunctionParameter %sampler %mixed_entry = OpLabel ; access the texture, to give the handles usages. %imsam = OpSampledImage %sampled_image_2d_f32 %14 %15 %20 = OpImageSampleImplicitLod %v4float %imsam %v2_0 OpReturn OpFunctionEnd )" + MainBody(); auto p = parser(test::Assemble(assembly)); ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error() << assembly; auto fe = p->function_emitter(200); EXPECT_TRUE(fe.Emit()); auto got = test::ToString(p->program()); std::string expect = R"(fn x_200(x_14 : texture_2d, x_15 : sampler) { let x_20 : vec4 = textureSample(x_14, x_15, vec2()); return; } )"; EXPECT_EQ(got, expect); } } // namespace } // namespace tint::reader::spirv