// Copyright 2021 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 "src/transform/spirv.h" #include "src/transform/test_helper.h" namespace tint { namespace transform { namespace { using SpirvTest = TransformTest; TEST_F(SpirvTest, HandleEntryPointIOTypes_Parameters) { auto* src = R"( [[stage(fragment)]] fn frag_main([[builtin(frag_coord)]] coord : vec4, [[location(1)]] loc1 : f32) -> void { var col : f32 = (coord.x * loc1); } [[stage(compute)]] fn compute_main([[builtin(local_invocation_id)]] local_id : vec3, [[builtin(local_invocation_index)]] local_index : u32) -> void { var id_x : u32 = local_id.x; } )"; auto* expect = R"( [[builtin(frag_coord)]] var tint_symbol_1 : vec4; [[location(1)]] var tint_symbol_2 : f32; [[stage(fragment)]] fn frag_main() -> void { var col : f32 = (tint_symbol_1.x * tint_symbol_2); } [[builtin(local_invocation_id)]] var tint_symbol_6 : vec3; [[builtin(local_invocation_index)]] var tint_symbol_7 : u32; [[stage(compute)]] fn compute_main() -> void { var id_x : u32 = tint_symbol_6.x; } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(SpirvTest, HandleEntryPointIOTypes_Parameter_TypeAlias) { auto* src = R"( type myf32 = f32; [[stage(fragment)]] fn frag_main([[location(1)]] loc1 : myf32) -> void { } )"; auto* expect = R"( type myf32 = f32; [[location(1)]] var tint_symbol_2 : myf32; [[stage(fragment)]] fn frag_main() -> void { } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(SpirvTest, HandleEntryPointIOTypes_ReturnBuiltin) { auto* src = R"( [[stage(vertex)]] fn vert_main() -> [[builtin(position)]] vec4 { return vec4(1.0, 2.0, 3.0, 0.0); } )"; auto* expect = R"( [[builtin(position)]] var tint_symbol_1 : vec4; fn tint_symbol_2(tint_symbol_3 : vec4) -> void { tint_symbol_1 = tint_symbol_3; } [[stage(vertex)]] fn vert_main() -> void { tint_symbol_2(vec4(1.0, 2.0, 3.0, 0.0)); return; } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(SpirvTest, HandleEntryPointIOTypes_ReturnLocation) { auto* src = R"( [[stage(fragment)]] fn frag_main([[location(0)]] loc_in : u32) -> [[location(0)]] f32 { if (loc_in > 10u) { return 0.5; } return 1.0; } )"; auto* expect = R"( [[location(0)]] var tint_symbol_1 : u32; [[location(0)]] var tint_symbol_2 : f32; fn tint_symbol_3(tint_symbol_4 : f32) -> void { tint_symbol_2 = tint_symbol_4; } [[stage(fragment)]] fn frag_main() -> void { if ((tint_symbol_1 > 10u)) { tint_symbol_3(0.5); return; } tint_symbol_3(1.0); return; } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(SpirvTest, HandleEntryPointIOTypes_ReturnLocation_TypeAlias) { auto* src = R"( type myf32 = f32; [[stage(fragment)]] fn frag_main([[location(0)]] loc_in : u32) -> [[location(0)]] myf32 { if (loc_in > 10u) { return 0.5; } return 1.0; } )"; auto* expect = R"( type myf32 = f32; [[location(0)]] var tint_symbol_2 : u32; [[location(0)]] var tint_symbol_3 : myf32; fn tint_symbol_4(tint_symbol_5 : myf32) -> void { tint_symbol_3 = tint_symbol_5; } [[stage(fragment)]] fn frag_main() -> void { if ((tint_symbol_2 > 10u)) { tint_symbol_4(0.5); return; } tint_symbol_4(1.0); return; } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(SpirvTest, HandleEntryPointIOTypes_StructParameters) { auto* src = R"( struct FragmentInput { [[builtin(frag_coord)]] coord : vec4; [[location(1)]] value : f32; }; [[stage(fragment)]] fn frag_main(inputs : FragmentInput) -> void { var col : f32 = inputs.coord.x * inputs.value; } )"; auto* expect = R"( struct FragmentInput { coord : vec4; value : f32; }; [[builtin(frag_coord)]] var tint_symbol_4 : vec4; [[location(1)]] var tint_symbol_5 : f32; [[stage(fragment)]] fn frag_main() -> void { const tint_symbol_7 : FragmentInput = FragmentInput(tint_symbol_4, tint_symbol_5); var col : f32 = (tint_symbol_7.coord.x * tint_symbol_7.value); } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(SpirvTest, HandleEntryPointIOTypes_StructParameters_EmptyBody) { auto* src = R"( struct FragmentInput { [[location(1)]] value : f32; }; [[stage(fragment)]] fn frag_main(inputs : FragmentInput) -> void { } )"; auto* expect = R"( struct FragmentInput { value : f32; }; [[location(1)]] var tint_symbol_3 : f32; [[stage(fragment)]] fn frag_main() -> void { } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(SpirvTest, HandleEntryPointIOTypes_ReturnStruct) { auto* src = R"( struct VertexOutput { [[builtin(position)]] pos : vec4; [[location(1)]] value : f32; }; [[stage(vertex)]] fn vert_main() -> VertexOutput { if (false) { return VertexOutput(); } var pos : vec4 = vec4(1.0, 2.0, 3.0, 0.0); return VertexOutput(pos, 2.0); } )"; auto* expect = R"( struct VertexOutput { pos : vec4; value : f32; }; [[builtin(position)]] var tint_symbol_4 : vec4; [[location(1)]] var tint_symbol_5 : f32; fn tint_symbol_6(tint_symbol_7 : VertexOutput) -> void { tint_symbol_4 = tint_symbol_7.pos; tint_symbol_5 = tint_symbol_7.value; } [[stage(vertex)]] fn vert_main() -> void { if (false) { tint_symbol_6(VertexOutput()); return; } var pos : vec4 = vec4(1.0, 2.0, 3.0, 0.0); tint_symbol_6(VertexOutput(pos, 2.0)); return; } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(SpirvTest, HandleEntryPointIOTypes_SharedStruct_SameShader) { auto* src = R"( struct Interface { [[location(1)]] value : f32; }; [[stage(vertex)]] fn vert_main(inputs : Interface) -> Interface { return inputs; } )"; auto* expect = R"( struct Interface { value : f32; }; [[location(1)]] var tint_symbol_3 : f32; [[location(1)]] var tint_symbol_4 : f32; fn tint_symbol_5(tint_symbol_6 : Interface) -> void { tint_symbol_4 = tint_symbol_6.value; } [[stage(vertex)]] fn vert_main() -> void { const tint_symbol_8 : Interface = Interface(tint_symbol_3); tint_symbol_5(tint_symbol_8); return; } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(SpirvTest, HandleEntryPointIOTypes_SharedStruct_DifferentShaders) { auto* src = R"( struct Interface { [[location(1)]] value : f32; }; [[stage(vertex)]] fn vert_main() -> Interface { return Interface(42.0); } [[stage(fragment)]] fn frag_main(inputs : Interface) -> void { var x : f32 = inputs.value; } )"; auto* expect = R"( struct Interface { value : f32; }; [[location(1)]] var tint_symbol_3 : f32; fn tint_symbol_4(tint_symbol_5 : Interface) -> void { tint_symbol_3 = tint_symbol_5.value; } [[stage(vertex)]] fn vert_main() -> void { tint_symbol_4(Interface(42.0)); return; } [[location(1)]] var tint_symbol_7 : f32; [[stage(fragment)]] fn frag_main() -> void { const tint_symbol_9 : Interface = Interface(tint_symbol_7); var x : f32 = tint_symbol_9.value; } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(SpirvTest, HandleEntryPointIOTypes_StructLayoutDecorations) { auto* src = R"( [[block]] struct FragmentInput { [[size(16), location(1)]] value : f32; [[builtin(frag_coord)]] [[align(32)]] coord : vec4; }; struct FragmentOutput { [[size(16), location(1)]] value : f32; }; [[stage(fragment)]] fn frag_main(inputs : FragmentInput) -> FragmentOutput { return FragmentOutput(inputs.coord.x * inputs.value); } )"; auto* expect = R"( [[block]] struct FragmentInput { [[size(16)]] value : f32; [[align(32)]] coord : vec4; }; struct FragmentOutput { [[size(16)]] value : f32; }; [[location(1)]] var tint_symbol_5 : f32; [[builtin(frag_coord)]] var tint_symbol_6 : vec4; [[location(1)]] var tint_symbol_7 : f32; fn tint_symbol_8(tint_symbol_9 : FragmentOutput) -> void { tint_symbol_7 = tint_symbol_9.value; } [[stage(fragment)]] fn frag_main() -> void { const tint_symbol_11 : FragmentInput = FragmentInput(tint_symbol_5, tint_symbol_6); tint_symbol_8(FragmentOutput((tint_symbol_11.coord.x * tint_symbol_11.value))); return; } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(SpirvTest, HandleEntryPointIOTypes_WithPrivateGlobalVariable) { // Test with a global variable to ensure that symbols are cloned correctly. // crbug.com/tint/701 auto* src = R"( var x : f32; struct VertexOutput { [[builtin(position)]] Position : vec4; }; [[stage(vertex)]] fn main() -> VertexOutput { return VertexOutput(vec4()); } )"; auto* expect = R"( var x : f32; struct VertexOutput { Position : vec4; }; [[builtin(position)]] var tint_symbol_4 : vec4; fn tint_symbol_5(tint_symbol_6 : VertexOutput) -> void { tint_symbol_4 = tint_symbol_6.Position; } [[stage(vertex)]] fn main() -> void { tint_symbol_5(VertexOutput(vec4())); return; } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(SpirvTest, HandleSampleMaskBuiltins_Basic) { auto* src = R"( [[builtin(sample_index)]] var sample_index : u32; [[builtin(sample_mask_in)]] var mask_in : u32; [[builtin(sample_mask_out)]] var mask_out : u32; [[stage(fragment)]] fn main() -> void { mask_out = mask_in; } )"; auto* expect = R"( [[builtin(sample_index)]] var sample_index : u32; [[builtin(sample_mask_in)]] var mask_in : array; [[builtin(sample_mask_out)]] var mask_out : array; [[stage(fragment)]] fn main() -> void { mask_out[0] = mask_in[0]; } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(SpirvTest, HandleSampleMaskBuiltins_FunctionArg) { auto* src = R"( [[builtin(sample_mask_in)]] var mask_in : u32; [[builtin(sample_mask_out)]] var mask_out : u32; fn filter(mask: u32) -> u32 { return (mask & 3u); } fn set_mask(input : u32) -> void { mask_out = input; } [[stage(fragment)]] fn main() -> void { set_mask(filter(mask_in)); } )"; auto* expect = R"( [[builtin(sample_mask_in)]] var mask_in : array; [[builtin(sample_mask_out)]] var mask_out : array; fn filter(mask : u32) -> u32 { return (mask & 3u); } fn set_mask(input : u32) -> void { mask_out[0] = input; } [[stage(fragment)]] fn main() -> void { set_mask(filter(mask_in[0])); } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(SpirvTest, AddEmptyEntryPoint) { auto* src = R"()"; auto* expect = R"( [[stage(compute)]] fn _tint_unused_entry_point() -> void { } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } // Test that different transforms within the sanitizer interact correctly. TEST_F(SpirvTest, MultipleTransforms) { auto* src = R"( [[stage(fragment)]] fn main([[builtin(sample_index)]] sample_index : u32, [[builtin(sample_mask_in)]] mask_in : u32) -> [[builtin(sample_mask_out)]] u32 { return mask_in; } )"; auto* expect = R"( [[builtin(sample_index)]] var tint_symbol_3 : u32; [[builtin(sample_mask_in)]] var tint_symbol_1 : array; [[builtin(sample_mask_out)]] var tint_symbol_2 : array; fn tint_symbol_4(tint_symbol_5 : u32) -> void { tint_symbol_2[0] = tint_symbol_5; } [[stage(fragment)]] fn main() -> void { tint_symbol_4(tint_symbol_1[0]); return; } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } } // namespace } // namespace transform } // namespace tint