| // 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 <string> |
| #include <vector> |
| |
| #include "gmock/gmock.h" |
| #include "src/reader/spirv/fail_stream.h" |
| #include "src/reader/spirv/function.h" |
| #include "src/reader/spirv/parser_impl.h" |
| #include "src/reader/spirv/parser_impl_test_helper.h" |
| #include "src/reader/spirv/spirv_tools_helpers_test.h" |
| |
| namespace tint { |
| namespace reader { |
| namespace spirv { |
| namespace { |
| |
| using ::testing::Eq; |
| using ::testing::HasSubstr; |
| |
| std::string Preamble() { |
| return R"( |
| %void = OpTypeVoid |
| %voidfn = OpTypeFunction %void |
| |
| %uint = OpTypeInt 32 0 |
| %int = OpTypeInt 32 1 |
| %float = OpTypeFloat 32 |
| |
| %uint_10 = OpConstant %uint 10 |
| %uint_20 = OpConstant %uint 20 |
| %uint_3 = OpConstant %uint 3 |
| %uint_4 = OpConstant %uint 4 |
| %uint_5 = OpConstant %uint 5 |
| %int_30 = OpConstant %int 30 |
| %int_40 = OpConstant %int 40 |
| %float_50 = OpConstant %float 50 |
| %float_60 = OpConstant %float 60 |
| %float_70 = OpConstant %float 70 |
| |
| %v2uint = OpTypeVector %uint 2 |
| %v3uint = OpTypeVector %uint 3 |
| %v4uint = OpTypeVector %uint 4 |
| %v2int = OpTypeVector %int 2 |
| %v2float = OpTypeVector %float 2 |
| |
| %m3v2float = OpTypeMatrix %v2float 3 |
| %m3v2float_0 = OpConstantNull %m3v2float |
| |
| %s_v2f_u_i = OpTypeStruct %v2float %uint %int |
| %a_u_5 = OpTypeArray %uint %uint_5 |
| |
| %v2uint_3_4 = OpConstantComposite %v2uint %uint_3 %uint_4 |
| %v2uint_4_3 = OpConstantComposite %v2uint %uint_4 %uint_3 |
| %v2float_50_60 = OpConstantComposite %v2float %float_50 %float_60 |
| %v2float_60_50 = OpConstantComposite %v2float %float_60 %float_50 |
| %v2float_70_70 = OpConstantComposite %v2float %float_70 %float_70 |
| )"; |
| } |
| |
| using SpvParserTest_Composite_Construct = SpvParserTest; |
| |
| TEST_F(SpvParserTest_Composite_Construct, Vector) { |
| const auto assembly = Preamble() + R"( |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %1 = OpCompositeConstruct %v2uint %uint_10 %uint_20 |
| %2 = OpCompositeConstruct %v2int %int_30 %int_40 |
| %3 = OpCompositeConstruct %v2float %float_50 %float_60 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_TRUE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"(VariableDeclStatement{ |
| VariableConst{ |
| x_1 |
| none |
| __vec_2__u32 |
| { |
| TypeConstructor[not set]{ |
| __vec_2__u32 |
| ScalarConstructor[not set]{10} |
| ScalarConstructor[not set]{20} |
| } |
| } |
| } |
| } |
| VariableDeclStatement{ |
| VariableConst{ |
| x_2 |
| none |
| __vec_2__i32 |
| { |
| TypeConstructor[not set]{ |
| __vec_2__i32 |
| ScalarConstructor[not set]{30} |
| ScalarConstructor[not set]{40} |
| } |
| } |
| } |
| } |
| VariableDeclStatement{ |
| VariableConst{ |
| x_3 |
| none |
| __vec_2__f32 |
| { |
| TypeConstructor[not set]{ |
| __vec_2__f32 |
| ScalarConstructor[not set]{50.000000} |
| ScalarConstructor[not set]{60.000000} |
| } |
| } |
| } |
| })")) << ToString(fe.ast_body()); |
| } |
| |
| TEST_F(SpvParserTest_Composite_Construct, Matrix) { |
| const auto assembly = Preamble() + R"( |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %1 = OpCompositeConstruct %m3v2float %v2float_50_60 %v2float_60_50 %v2float_70_70 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_TRUE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"( |
| VariableConst{ |
| x_1 |
| none |
| __mat_2_3__f32 |
| { |
| TypeConstructor[not set]{ |
| __mat_2_3__f32 |
| TypeConstructor[not set]{ |
| __vec_2__f32 |
| ScalarConstructor[not set]{50.000000} |
| ScalarConstructor[not set]{60.000000} |
| } |
| TypeConstructor[not set]{ |
| __vec_2__f32 |
| ScalarConstructor[not set]{60.000000} |
| ScalarConstructor[not set]{50.000000} |
| } |
| TypeConstructor[not set]{ |
| __vec_2__f32 |
| ScalarConstructor[not set]{70.000000} |
| ScalarConstructor[not set]{70.000000} |
| } |
| } |
| } |
| })")) |
| << ToString(fe.ast_body()); |
| } |
| |
| TEST_F(SpvParserTest_Composite_Construct, Array) { |
| const auto assembly = Preamble() + R"( |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %1 = OpCompositeConstruct %a_u_5 %uint_10 %uint_20 %uint_3 %uint_4 %uint_5 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_TRUE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"( |
| VariableConst{ |
| x_1 |
| none |
| __array__u32_5 |
| { |
| TypeConstructor[not set]{ |
| __array__u32_5 |
| ScalarConstructor[not set]{10} |
| ScalarConstructor[not set]{20} |
| ScalarConstructor[not set]{3} |
| ScalarConstructor[not set]{4} |
| ScalarConstructor[not set]{5} |
| } |
| } |
| })")) |
| << ToString(fe.ast_body()); |
| } |
| |
| TEST_F(SpvParserTest_Composite_Construct, Struct) { |
| const auto assembly = Preamble() + R"( |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %1 = OpCompositeConstruct %s_v2f_u_i %v2float_50_60 %uint_5 %int_30 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_TRUE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"( |
| VariableConst{ |
| x_1 |
| none |
| __struct_S |
| { |
| TypeConstructor[not set]{ |
| __struct_S |
| TypeConstructor[not set]{ |
| __vec_2__f32 |
| ScalarConstructor[not set]{50.000000} |
| ScalarConstructor[not set]{60.000000} |
| } |
| ScalarConstructor[not set]{5} |
| ScalarConstructor[not set]{30} |
| } |
| } |
| })")) |
| << ToString(fe.ast_body()); |
| } |
| |
| using SpvParserTest_CompositeExtract = SpvParserTest; |
| |
| TEST_F(SpvParserTest_CompositeExtract, Vector) { |
| const auto assembly = Preamble() + R"( |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %1 = OpCompositeExtract %float %v2float_50_60 1 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_TRUE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"( |
| VariableConst{ |
| x_1 |
| none |
| __f32 |
| { |
| MemberAccessor[not set]{ |
| TypeConstructor[not set]{ |
| __vec_2__f32 |
| ScalarConstructor[not set]{50.000000} |
| ScalarConstructor[not set]{60.000000} |
| } |
| Identifier[not set]{y} |
| } |
| } |
| })")) |
| << ToString(fe.ast_body()); |
| } |
| |
| TEST_F(SpvParserTest_CompositeExtract, Vector_IndexTooBigError) { |
| const auto assembly = Preamble() + R"( |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %1 = OpCompositeExtract %float %v2float_50_60 900 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_FALSE(fe.EmitBody()); |
| EXPECT_THAT(p->error(), Eq("CompositeExtract %1 index value 900 is out of " |
| "bounds for vector of 2 elements")); |
| } |
| |
| TEST_F(SpvParserTest_CompositeExtract, Matrix) { |
| const auto assembly = Preamble() + R"( |
| %ptr = OpTypePointer Function %m3v2float |
| |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %var = OpVariable %ptr Function |
| %1 = OpLoad %m3v2float %var |
| %2 = OpCompositeExtract %v2float %1 2 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_TRUE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"( |
| VariableConst{ |
| x_2 |
| none |
| __vec_2__f32 |
| { |
| ArrayAccessor[not set]{ |
| Identifier[not set]{x_1} |
| ScalarConstructor[not set]{2} |
| } |
| } |
| })")) |
| << ToString(fe.ast_body()); |
| } |
| |
| TEST_F(SpvParserTest_CompositeExtract, Matrix_IndexTooBigError) { |
| const auto assembly = Preamble() + R"( |
| %ptr = OpTypePointer Function %m3v2float |
| |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %var = OpVariable %ptr Function |
| %1 = OpLoad %m3v2float %var |
| %2 = OpCompositeExtract %v2float %1 3 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_FALSE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(p->error(), Eq("CompositeExtract %2 index value 3 is out of " |
| "bounds for matrix of 3 elements")); |
| } |
| |
| TEST_F(SpvParserTest_CompositeExtract, Matrix_Vector) { |
| const auto assembly = Preamble() + R"( |
| %ptr = OpTypePointer Function %m3v2float |
| |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %var = OpVariable %ptr Function |
| %1 = OpLoad %m3v2float %var |
| %2 = OpCompositeExtract %float %1 2 1 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_TRUE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"( |
| VariableConst{ |
| x_2 |
| none |
| __f32 |
| { |
| MemberAccessor[not set]{ |
| ArrayAccessor[not set]{ |
| Identifier[not set]{x_1} |
| ScalarConstructor[not set]{2} |
| } |
| Identifier[not set]{y} |
| } |
| } |
| })")) |
| << ToString(fe.ast_body()); |
| } |
| |
| TEST_F(SpvParserTest_CompositeExtract, Array) { |
| const auto assembly = Preamble() + R"( |
| %ptr = OpTypePointer Function %a_u_5 |
| |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %var = OpVariable %ptr Function |
| %1 = OpLoad %a_u_5 %var |
| %2 = OpCompositeExtract %uint %1 3 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_TRUE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"( |
| VariableConst{ |
| x_2 |
| none |
| __u32 |
| { |
| ArrayAccessor[not set]{ |
| Identifier[not set]{x_1} |
| ScalarConstructor[not set]{3} |
| } |
| } |
| })")) |
| << ToString(fe.ast_body()); |
| } |
| |
| TEST_F(SpvParserTest_CompositeExtract, RuntimeArray_IsError) { |
| const auto assembly = Preamble() + R"( |
| %rtarr = OpTypeRuntimeArray %uint |
| %ptr = OpTypePointer Function %rtarr |
| |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %var = OpVariable %ptr Function |
| %1 = OpLoad %rtarr %var |
| %2 = OpCompositeExtract %uint %1 3 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_FALSE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(p->error(), Eq("can't do OpCompositeExtract on a runtime array")); |
| } |
| |
| TEST_F(SpvParserTest_CompositeExtract, Struct) { |
| const auto assembly = Preamble() + R"( |
| %ptr = OpTypePointer Function %s_v2f_u_i |
| |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %var = OpVariable %ptr Function |
| %1 = OpLoad %s_v2f_u_i %var |
| %2 = OpCompositeExtract %int %1 2 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_TRUE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"( |
| VariableConst{ |
| x_2 |
| none |
| __i32 |
| { |
| MemberAccessor[not set]{ |
| Identifier[not set]{x_1} |
| Identifier[not set]{field2} |
| } |
| } |
| })")) |
| << ToString(fe.ast_body()); |
| } |
| |
| TEST_F(SpvParserTest_CompositeExtract, Struct_DifferOnlyInMemberName) { |
| const auto assembly = |
| R"( |
| OpMemberName %s0 0 "algo" |
| OpMemberName %s1 0 "rithm" |
| )" + Preamble() + |
| R"( |
| %s0 = OpTypeStruct %uint |
| %s1 = OpTypeStruct %uint |
| %ptr0 = OpTypePointer Function %s0 |
| %ptr1 = OpTypePointer Function %s1 |
| |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %var0 = OpVariable %ptr0 Function |
| %var1 = OpVariable %ptr1 Function |
| %1 = OpLoad %s0 %var0 |
| %2 = OpCompositeExtract %uint %1 0 |
| %3 = OpLoad %s1 %var1 |
| %4 = OpCompositeExtract %uint %3 0 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_TRUE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"( |
| VariableConst{ |
| x_2 |
| none |
| __u32 |
| { |
| MemberAccessor[not set]{ |
| Identifier[not set]{x_1} |
| Identifier[not set]{algo} |
| } |
| } |
| })")) |
| << ToString(fe.ast_body()); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"( |
| VariableConst{ |
| x_4 |
| none |
| __u32 |
| { |
| MemberAccessor[not set]{ |
| Identifier[not set]{x_3} |
| Identifier[not set]{rithm} |
| } |
| } |
| })")) |
| << ToString(fe.ast_body()); |
| } |
| |
| TEST_F(SpvParserTest_CompositeExtract, Struct_IndexTooBigError) { |
| const auto assembly = Preamble() + R"( |
| %ptr = OpTypePointer Function %s_v2f_u_i |
| |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %var = OpVariable %ptr Function |
| %1 = OpLoad %s_v2f_u_i %var |
| %2 = OpCompositeExtract %int %1 40 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_FALSE(fe.EmitBody()); |
| EXPECT_THAT(p->error(), Eq("CompositeExtract %2 index value 40 is out of " |
| "bounds for structure %25 having 3 members")); |
| } |
| |
| TEST_F(SpvParserTest_CompositeExtract, Struct_Array_Matrix_Vector) { |
| const auto assembly = Preamble() + R"( |
| %a_mat = OpTypeArray %m3v2float %uint_3 |
| %s = OpTypeStruct %uint %a_mat |
| %ptr = OpTypePointer Function %s |
| |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %var = OpVariable %ptr Function |
| %1 = OpLoad %s %var |
| %2 = OpCompositeExtract %float %1 1 2 0 1 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_TRUE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"( |
| VariableConst{ |
| x_2 |
| none |
| __f32 |
| { |
| MemberAccessor[not set]{ |
| ArrayAccessor[not set]{ |
| ArrayAccessor[not set]{ |
| MemberAccessor[not set]{ |
| Identifier[not set]{x_1} |
| Identifier[not set]{field1} |
| } |
| ScalarConstructor[not set]{2} |
| } |
| ScalarConstructor[not set]{0} |
| } |
| Identifier[not set]{y} |
| } |
| } |
| })")) |
| << ToString(fe.ast_body()); |
| } |
| |
| using SpvParserTest_CopyObject = SpvParserTest; |
| |
| TEST_F(SpvParserTest_CopyObject, Scalar) { |
| const auto assembly = Preamble() + R"( |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %1 = OpCopyObject %uint %uint_3 |
| %2 = OpCopyObject %uint %1 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_TRUE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"(VariableDeclStatement{ |
| VariableConst{ |
| x_1 |
| none |
| __u32 |
| { |
| ScalarConstructor[not set]{3} |
| } |
| } |
| } |
| VariableDeclStatement{ |
| VariableConst{ |
| x_2 |
| none |
| __u32 |
| { |
| Identifier[not set]{x_1} |
| } |
| } |
| })")) << ToString(fe.ast_body()); |
| } |
| |
| TEST_F(SpvParserTest_CopyObject, Pointer) { |
| const auto assembly = Preamble() + R"( |
| %ptr = OpTypePointer Function %uint |
| |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %10 = OpVariable %ptr Function |
| %1 = OpCopyObject %ptr %10 |
| %2 = OpCopyObject %ptr %1 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_TRUE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"(VariableDeclStatement{ |
| VariableConst{ |
| x_1 |
| none |
| __ptr_function__u32 |
| { |
| Identifier[not set]{x_10} |
| } |
| } |
| } |
| VariableDeclStatement{ |
| VariableConst{ |
| x_2 |
| none |
| __ptr_function__u32 |
| { |
| Identifier[not set]{x_1} |
| } |
| } |
| })")) << ToString(fe.ast_body()); |
| } |
| |
| using SpvParserTest_VectorShuffle = SpvParserTest; |
| |
| TEST_F(SpvParserTest_VectorShuffle, FunctionScopeOperands_UseBoth) { |
| // Note that variables are generated for the vector operands. |
| const auto assembly = Preamble() + R"( |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %1 = OpCopyObject %v2uint %v2uint_3_4 |
| %2 = OpIAdd %v2uint %v2uint_4_3 %v2uint_3_4 |
| %10 = OpVectorShuffle %v4uint %1 %2 3 2 1 0 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_TRUE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"(VariableConst{ |
| x_10 |
| none |
| __vec_4__u32 |
| { |
| TypeConstructor[not set]{ |
| __vec_4__u32 |
| MemberAccessor[not set]{ |
| Identifier[not set]{x_2} |
| Identifier[not set]{y} |
| } |
| MemberAccessor[not set]{ |
| Identifier[not set]{x_2} |
| Identifier[not set]{x} |
| } |
| MemberAccessor[not set]{ |
| Identifier[not set]{x_1} |
| Identifier[not set]{y} |
| } |
| MemberAccessor[not set]{ |
| Identifier[not set]{x_1} |
| Identifier[not set]{x} |
| } |
| } |
| } |
| } |
| })")) << ToString(fe.ast_body()); |
| } |
| |
| TEST_F(SpvParserTest_VectorShuffle, ConstantOperands_UseBoth) { |
| const auto assembly = Preamble() + R"( |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %10 = OpVectorShuffle %v4uint %v2uint_3_4 %v2uint_4_3 3 2 1 0 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_TRUE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"(VariableConst{ |
| x_10 |
| none |
| __vec_4__u32 |
| { |
| TypeConstructor[not set]{ |
| __vec_4__u32 |
| MemberAccessor[not set]{ |
| TypeConstructor[not set]{ |
| __vec_2__u32 |
| ScalarConstructor[not set]{4} |
| ScalarConstructor[not set]{3} |
| } |
| Identifier[not set]{y} |
| } |
| MemberAccessor[not set]{ |
| TypeConstructor[not set]{ |
| __vec_2__u32 |
| ScalarConstructor[not set]{4} |
| ScalarConstructor[not set]{3} |
| } |
| Identifier[not set]{x} |
| } |
| MemberAccessor[not set]{ |
| TypeConstructor[not set]{ |
| __vec_2__u32 |
| ScalarConstructor[not set]{3} |
| ScalarConstructor[not set]{4} |
| } |
| Identifier[not set]{y} |
| } |
| MemberAccessor[not set]{ |
| TypeConstructor[not set]{ |
| __vec_2__u32 |
| ScalarConstructor[not set]{3} |
| ScalarConstructor[not set]{4} |
| } |
| Identifier[not set]{x} |
| } |
| } |
| } |
| })")) |
| << ToString(fe.ast_body()); |
| } |
| |
| TEST_F(SpvParserTest_VectorShuffle, ConstantOperands_AllOnesMapToNull) { |
| const auto assembly = Preamble() + R"( |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %1 = OpCopyObject %v2uint %v2uint_4_3 |
| %10 = OpVectorShuffle %v2uint %1 %1 0xFFFFFFFF 1 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_TRUE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"(VariableConst{ |
| x_10 |
| none |
| __vec_2__u32 |
| { |
| TypeConstructor[not set]{ |
| __vec_2__u32 |
| ScalarConstructor[not set]{0} |
| MemberAccessor[not set]{ |
| Identifier[not set]{x_1} |
| Identifier[not set]{y} |
| } |
| } |
| } |
| })")) |
| << ToString(fe.ast_body()); |
| } |
| |
| TEST_F(SpvParserTest_VectorShuffle, IndexTooBig_IsError) { |
| const auto assembly = Preamble() + R"( |
| %100 = OpFunction %void None %voidfn |
| %entry = OpLabel |
| %10 = OpVectorShuffle %v4uint %v2uint_3_4 %v2uint_4_3 9 2 1 0 |
| OpReturn |
| OpFunctionEnd |
| )"; |
| |
| auto* p = parser(test::Assemble(assembly)); |
| ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << assembly; |
| FunctionEmitter fe(p, *spirv_function(100)); |
| EXPECT_FALSE(fe.EmitBody()) << p->error(); |
| EXPECT_THAT(p->error(), |
| Eq("invalid vectorshuffle ID %10: index too large: 9")); |
| } |
| |
| } // namespace |
| } // namespace spirv |
| } // namespace reader |
| } // namespace tint |