| // Copyright 2025 The Dawn & Tint Authors |
| // |
| // Redistribution and use in source and binary forms, with or without |
| // modification, are permitted provided that the following conditions are met: |
| // |
| // 1. Redistributions of source code must retain the above copyright notice, this |
| // list of conditions and the following disclaimer. |
| // |
| // 2. Redistributions in binary form must reproduce the above copyright notice, |
| // this list of conditions and the following disclaimer in the documentation |
| // and/or other materials provided with the distribution. |
| // |
| // 3. Neither the name of the copyright holder nor the names of its |
| // contributors may be used to endorse or promote products derived from |
| // this software without specific prior written permission. |
| // |
| // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" |
| // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
| // DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE |
| // FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL |
| // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR |
| // SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER |
| // CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, |
| // OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| |
| #include "src/tint/lang/spirv/reader/parser/helper_test.h" |
| |
| namespace tint::spirv::reader { |
| namespace { |
| |
| struct SpirvLogicalParam { |
| std::string spv_name; |
| std::string wgsl_name; |
| }; |
| [[maybe_unused]] inline std::ostream& operator<<(std::ostream& out, SpirvLogicalParam c) { |
| out << c.spv_name; |
| return out; |
| } |
| |
| using SpirvParser_FOrdLogicalTest = SpirvParserTestWithParam<SpirvLogicalParam>; |
| TEST_P(SpirvParser_FOrdLogicalTest, Scalar) { |
| auto params = GetParam(); |
| EXPECT_IR(R"( |
| OpCapability Shader |
| OpCapability Float16 |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint GLCompute %main "main" |
| OpExecutionMode %main LocalSize 1 1 1 |
| %void = OpTypeVoid |
| %bool = OpTypeBool |
| %f32 = OpTypeFloat 32 |
| %one = OpConstant %f32 1 |
| %two = OpConstant %f32 2 |
| %v2bool = OpTypeVector %bool 2 |
| %v2float = OpTypeVector %f32 2 |
| %v2one = OpConstantComposite %v2float %one %one |
| %v2two = OpConstantComposite %v2float %two %two |
| %ep_type = OpTypeFunction %void |
| %main = OpFunction %void None %ep_type |
| %main_start = OpLabel |
| %1 = OpFOrd)" + |
| params.spv_name + R"( %bool %one %two |
| OpReturn |
| OpFunctionEnd |
| )", |
| R"( |
| %main = @compute @workgroup_size(1u, 1u, 1u) func():void { |
| $B1: { |
| %2:bool = )" + params.wgsl_name + |
| R"( 1.0f, 2.0f |
| ret |
| } |
| } |
| )"); |
| } |
| |
| TEST_P(SpirvParser_FOrdLogicalTest, Vector) { |
| auto params = GetParam(); |
| EXPECT_IR(R"( |
| OpCapability Shader |
| OpCapability Float16 |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint GLCompute %main "main" |
| OpExecutionMode %main LocalSize 1 1 1 |
| %void = OpTypeVoid |
| %bool = OpTypeBool |
| %f32 = OpTypeFloat 32 |
| %one = OpConstant %f32 1 |
| %two = OpConstant %f32 2 |
| %v2bool = OpTypeVector %bool 2 |
| %v2float = OpTypeVector %f32 2 |
| %v2one = OpConstantComposite %v2float %one %one |
| %v2two = OpConstantComposite %v2float %two %two |
| %ep_type = OpTypeFunction %void |
| %main = OpFunction %void None %ep_type |
| %main_start = OpLabel |
| %1 = OpFOrd)" + |
| params.spv_name + R"( %v2bool %v2one %v2two |
| OpReturn |
| OpFunctionEnd |
| )", |
| R"( |
| %main = @compute @workgroup_size(1u, 1u, 1u) func():void { |
| $B1: { |
| %2:vec2<bool> = )" + |
| params.wgsl_name + R"( vec2<f32>(1.0f), vec2<f32>(2.0f) |
| ret |
| } |
| } |
| )"); |
| } |
| |
| INSTANTIATE_TEST_SUITE_P(SpirvParserTest, |
| SpirvParser_FOrdLogicalTest, |
| testing::Values(SpirvLogicalParam{"Equal", "eq"}, |
| SpirvLogicalParam{"NotEqual", "neq"}, |
| SpirvLogicalParam{"GreaterThan", "gt"}, |
| SpirvLogicalParam{"GreaterThanEqual", "gte"}, |
| SpirvLogicalParam{"LessThan", "lt"}, |
| SpirvLogicalParam{"LessThanEqual", "lte"})); |
| |
| using SpirvParser_FUnordLogicalTest = SpirvParserTestWithParam<SpirvLogicalParam>; |
| TEST_P(SpirvParser_FUnordLogicalTest, FOrd_Scalar) { |
| auto params = GetParam(); |
| EXPECT_IR(R"( |
| OpCapability Shader |
| OpCapability Float16 |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint GLCompute %main "main" |
| OpExecutionMode %main LocalSize 1 1 1 |
| %void = OpTypeVoid |
| %bool = OpTypeBool |
| %f32 = OpTypeFloat 32 |
| %one = OpConstant %f32 1 |
| %two = OpConstant %f32 2 |
| %v2bool = OpTypeVector %bool 2 |
| %v2float = OpTypeVector %f32 2 |
| %v2one = OpConstantComposite %v2float %one %one |
| %v2two = OpConstantComposite %v2float %two %two |
| %ep_type = OpTypeFunction %void |
| %main = OpFunction %void None %ep_type |
| %main_start = OpLabel |
| %1 = OpFUnord)" + |
| params.spv_name + R"( %bool %one %two |
| OpReturn |
| OpFunctionEnd |
| )", |
| R"( |
| %main = @compute @workgroup_size(1u, 1u, 1u) func():void { |
| $B1: { |
| %2:bool = )" + params.wgsl_name + |
| R"( 1.0f, 2.0f |
| %3:bool = not %2 |
| ret |
| } |
| } |
| )"); |
| } |
| |
| TEST_P(SpirvParser_FUnordLogicalTest, FOrd_Vector) { |
| auto params = GetParam(); |
| EXPECT_IR(R"( |
| OpCapability Shader |
| OpCapability Float16 |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint GLCompute %main "main" |
| OpExecutionMode %main LocalSize 1 1 1 |
| %void = OpTypeVoid |
| %bool = OpTypeBool |
| %f32 = OpTypeFloat 32 |
| %one = OpConstant %f32 1 |
| %two = OpConstant %f32 2 |
| %v2bool = OpTypeVector %bool 2 |
| %v2float = OpTypeVector %f32 2 |
| %v2one = OpConstantComposite %v2float %one %one |
| %v2two = OpConstantComposite %v2float %two %two |
| %ep_type = OpTypeFunction %void |
| %main = OpFunction %void None %ep_type |
| %main_start = OpLabel |
| %1 = OpFUnord)" + |
| params.spv_name + R"( %v2bool %v2one %v2two |
| OpReturn |
| OpFunctionEnd |
| )", |
| R"( |
| %main = @compute @workgroup_size(1u, 1u, 1u) func():void { |
| $B1: { |
| %2:vec2<bool> = )" + |
| params.wgsl_name + R"( vec2<f32>(1.0f), vec2<f32>(2.0f) |
| %3:vec2<bool> = not %2 |
| ret |
| } |
| } |
| )"); |
| } |
| |
| INSTANTIATE_TEST_SUITE_P(SpirvParserTest, |
| SpirvParser_FUnordLogicalTest, |
| testing::Values(SpirvLogicalParam{"Equal", "neq"}, |
| SpirvLogicalParam{"NotEqual", "eq"}, |
| SpirvLogicalParam{"GreaterThan", "lte"}, |
| SpirvLogicalParam{"GreaterThanEqual", "lt"}, |
| SpirvLogicalParam{"LessThan", "gte"}, |
| SpirvLogicalParam{"LessThanEqual", "gt"})); |
| |
| TEST_F(SpirvParserTest, IEqual_Scalar_SignedSigned) { |
| EXPECT_IR(R"( |
| OpCapability Shader |
| OpCapability Float16 |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint GLCompute %main "main" |
| OpExecutionMode %main LocalSize 1 1 1 |
| %void = OpTypeVoid |
| %bool = OpTypeBool |
| %int = OpTypeInt 32 1 |
| %uint = OpTypeInt 32 0 |
| %v2bool = OpTypeVector %bool 2 |
| %v2int = OpTypeVector %int 2 |
| %v2uint = OpTypeVector %uint 2 |
| %one = OpConstant %int 1 |
| %two = OpConstant %int 2 |
| %eight = OpConstant %uint 8 |
| %nine = OpConstant %uint 9 |
| %v2one = OpConstantComposite %v2int %one %one |
| %v2two = OpConstantComposite %v2int %two %two |
| %v2eight = OpConstantComposite %v2uint %eight %eight |
| %v2nine = OpConstantComposite %v2uint %nine %nine |
| %ep_type = OpTypeFunction %void |
| %main = OpFunction %void None %ep_type |
| %main_start = OpLabel |
| %1 = OpIEqual %bool %one %two |
| OpReturn |
| OpFunctionEnd |
| )", |
| R"( |
| %main = @compute @workgroup_size(1u, 1u, 1u) func():void { |
| $B1: { |
| %2:bool = spirv.equal 1i, 2i |
| ret |
| } |
| } |
| )"); |
| } |
| |
| TEST_F(SpirvParserTest, IEqual_Scalar_SignedUnsigned) { |
| EXPECT_IR(R"( |
| OpCapability Shader |
| OpCapability Float16 |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint GLCompute %main "main" |
| OpExecutionMode %main LocalSize 1 1 1 |
| %void = OpTypeVoid |
| %bool = OpTypeBool |
| %int = OpTypeInt 32 1 |
| %uint = OpTypeInt 32 0 |
| %v2bool = OpTypeVector %bool 2 |
| %v2int = OpTypeVector %int 2 |
| %v2uint = OpTypeVector %uint 2 |
| %one = OpConstant %int 1 |
| %two = OpConstant %int 2 |
| %eight = OpConstant %uint 8 |
| %nine = OpConstant %uint 9 |
| %v2one = OpConstantComposite %v2int %one %one |
| %v2two = OpConstantComposite %v2int %two %two |
| %v2eight = OpConstantComposite %v2uint %eight %eight |
| %v2nine = OpConstantComposite %v2uint %nine %nine |
| %ep_type = OpTypeFunction %void |
| %main = OpFunction %void None %ep_type |
| %main_start = OpLabel |
| %1 = OpIEqual %bool %one %eight |
| OpReturn |
| OpFunctionEnd |
| )", |
| R"( |
| %main = @compute @workgroup_size(1u, 1u, 1u) func():void { |
| $B1: { |
| %2:bool = spirv.equal 1i, 8u |
| ret |
| } |
| } |
| )"); |
| } |
| |
| TEST_F(SpirvParserTest, IEqual_Scalar_UnsignedSigned) { |
| EXPECT_IR(R"( |
| OpCapability Shader |
| OpCapability Float16 |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint GLCompute %main "main" |
| OpExecutionMode %main LocalSize 1 1 1 |
| %void = OpTypeVoid |
| %bool = OpTypeBool |
| %int = OpTypeInt 32 1 |
| %uint = OpTypeInt 32 0 |
| %v2bool = OpTypeVector %bool 2 |
| %v2int = OpTypeVector %int 2 |
| %v2uint = OpTypeVector %uint 2 |
| %one = OpConstant %int 1 |
| %two = OpConstant %int 2 |
| %eight = OpConstant %uint 8 |
| %nine = OpConstant %uint 9 |
| %v2one = OpConstantComposite %v2int %one %one |
| %v2two = OpConstantComposite %v2int %two %two |
| %v2eight = OpConstantComposite %v2uint %eight %eight |
| %v2nine = OpConstantComposite %v2uint %nine %nine |
| %ep_type = OpTypeFunction %void |
| %main = OpFunction %void None %ep_type |
| %main_start = OpLabel |
| %1 = OpIEqual %bool %eight %one |
| OpReturn |
| OpFunctionEnd |
| )", |
| R"( |
| %main = @compute @workgroup_size(1u, 1u, 1u) func():void { |
| $B1: { |
| %2:bool = spirv.equal 8u, 1i |
| ret |
| } |
| } |
| )"); |
| } |
| |
| TEST_F(SpirvParserTest, IEqual_Scalar_UnsignedUnsigned) { |
| EXPECT_IR(R"( |
| OpCapability Shader |
| OpCapability Float16 |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint GLCompute %main "main" |
| OpExecutionMode %main LocalSize 1 1 1 |
| %void = OpTypeVoid |
| %bool = OpTypeBool |
| %int = OpTypeInt 32 1 |
| %uint = OpTypeInt 32 0 |
| %v2bool = OpTypeVector %bool 2 |
| %v2int = OpTypeVector %int 2 |
| %v2uint = OpTypeVector %uint 2 |
| %one = OpConstant %int 1 |
| %two = OpConstant %int 2 |
| %eight = OpConstant %uint 8 |
| %nine = OpConstant %uint 9 |
| %v2one = OpConstantComposite %v2int %one %one |
| %v2two = OpConstantComposite %v2int %two %two |
| %v2eight = OpConstantComposite %v2uint %eight %eight |
| %v2nine = OpConstantComposite %v2uint %nine %nine |
| %ep_type = OpTypeFunction %void |
| %main = OpFunction %void None %ep_type |
| %main_start = OpLabel |
| %1 = OpIEqual %bool %eight %nine |
| OpReturn |
| OpFunctionEnd |
| )", |
| R"( |
| %main = @compute @workgroup_size(1u, 1u, 1u) func():void { |
| $B1: { |
| %2:bool = spirv.equal 8u, 9u |
| ret |
| } |
| } |
| )"); |
| } |
| |
| TEST_F(SpirvParserTest, IEqual_Vector_SignedSigned) { |
| EXPECT_IR(R"( |
| OpCapability Shader |
| OpCapability Float16 |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint GLCompute %main "main" |
| OpExecutionMode %main LocalSize 1 1 1 |
| %void = OpTypeVoid |
| %bool = OpTypeBool |
| %int = OpTypeInt 32 1 |
| %uint = OpTypeInt 32 0 |
| %v2bool = OpTypeVector %bool 2 |
| %v2int = OpTypeVector %int 2 |
| %v2uint = OpTypeVector %uint 2 |
| %one = OpConstant %int 1 |
| %two = OpConstant %int 2 |
| %eight = OpConstant %uint 8 |
| %nine = OpConstant %uint 9 |
| %v2one = OpConstantComposite %v2int %one %one |
| %v2two = OpConstantComposite %v2int %two %two |
| %v2eight = OpConstantComposite %v2uint %eight %eight |
| %v2nine = OpConstantComposite %v2uint %nine %nine |
| %ep_type = OpTypeFunction %void |
| %main = OpFunction %void None %ep_type |
| %main_start = OpLabel |
| %1 = OpIEqual %v2bool %v2one %v2two |
| OpReturn |
| OpFunctionEnd |
| )", |
| R"( |
| %main = @compute @workgroup_size(1u, 1u, 1u) func():void { |
| $B1: { |
| %2:vec2<bool> = spirv.equal vec2<i32>(1i), vec2<i32>(2i) |
| ret |
| } |
| } |
| )"); |
| } |
| |
| TEST_F(SpirvParserTest, IEqual_Vector_SignedUnsigned) { |
| EXPECT_IR(R"( |
| OpCapability Shader |
| OpCapability Float16 |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint GLCompute %main "main" |
| OpExecutionMode %main LocalSize 1 1 1 |
| %void = OpTypeVoid |
| %bool = OpTypeBool |
| %int = OpTypeInt 32 1 |
| %uint = OpTypeInt 32 0 |
| %v2bool = OpTypeVector %bool 2 |
| %v2int = OpTypeVector %int 2 |
| %v2uint = OpTypeVector %uint 2 |
| %one = OpConstant %int 1 |
| %two = OpConstant %int 2 |
| %eight = OpConstant %uint 8 |
| %nine = OpConstant %uint 9 |
| %v2one = OpConstantComposite %v2int %one %one |
| %v2two = OpConstantComposite %v2int %two %two |
| %v2eight = OpConstantComposite %v2uint %eight %eight |
| %v2nine = OpConstantComposite %v2uint %nine %nine |
| %ep_type = OpTypeFunction %void |
| %main = OpFunction %void None %ep_type |
| %main_start = OpLabel |
| %1 = OpIEqual %v2bool %v2one %v2eight |
| OpReturn |
| OpFunctionEnd |
| )", |
| R"( |
| %main = @compute @workgroup_size(1u, 1u, 1u) func():void { |
| $B1: { |
| %2:vec2<bool> = spirv.equal vec2<i32>(1i), vec2<u32>(8u) |
| ret |
| } |
| } |
| )"); |
| } |
| |
| TEST_F(SpirvParserTest, IEqual_Vector_UnsignedSigned) { |
| EXPECT_IR(R"( |
| OpCapability Shader |
| OpCapability Float16 |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint GLCompute %main "main" |
| OpExecutionMode %main LocalSize 1 1 1 |
| %void = OpTypeVoid |
| %bool = OpTypeBool |
| %int = OpTypeInt 32 1 |
| %uint = OpTypeInt 32 0 |
| %v2bool = OpTypeVector %bool 2 |
| %v2int = OpTypeVector %int 2 |
| %v2uint = OpTypeVector %uint 2 |
| %one = OpConstant %int 1 |
| %two = OpConstant %int 2 |
| %eight = OpConstant %uint 8 |
| %nine = OpConstant %uint 9 |
| %v2one = OpConstantComposite %v2int %one %one |
| %v2two = OpConstantComposite %v2int %two %two |
| %v2eight = OpConstantComposite %v2uint %eight %eight |
| %v2nine = OpConstantComposite %v2uint %nine %nine |
| %ep_type = OpTypeFunction %void |
| %main = OpFunction %void None %ep_type |
| %main_start = OpLabel |
| %1 = OpIEqual %v2bool %v2eight %v2one |
| OpReturn |
| OpFunctionEnd |
| )", |
| R"( |
| %main = @compute @workgroup_size(1u, 1u, 1u) func():void { |
| $B1: { |
| %2:vec2<bool> = spirv.equal vec2<u32>(8u), vec2<i32>(1i) |
| ret |
| } |
| } |
| )"); |
| } |
| |
| TEST_F(SpirvParserTest, IEqual_Vector_UnsignedUnsigned) { |
| EXPECT_IR(R"( |
| OpCapability Shader |
| OpCapability Float16 |
| OpMemoryModel Logical GLSL450 |
| OpEntryPoint GLCompute %main "main" |
| OpExecutionMode %main LocalSize 1 1 1 |
| %void = OpTypeVoid |
| %bool = OpTypeBool |
| %int = OpTypeInt 32 1 |
| %uint = OpTypeInt 32 0 |
| %v2bool = OpTypeVector %bool 2 |
| %v2int = OpTypeVector %int 2 |
| %v2uint = OpTypeVector %uint 2 |
| %one = OpConstant %int 1 |
| %two = OpConstant %int 2 |
| %eight = OpConstant %uint 8 |
| %nine = OpConstant %uint 9 |
| %v2one = OpConstantComposite %v2int %one %one |
| %v2two = OpConstantComposite %v2int %two %two |
| %v2eight = OpConstantComposite %v2uint %eight %eight |
| %v2nine = OpConstantComposite %v2uint %nine %nine |
| %ep_type = OpTypeFunction %void |
| %main = OpFunction %void None %ep_type |
| %main_start = OpLabel |
| %1 = OpIEqual %v2bool %v2eight %v2nine |
| OpReturn |
| OpFunctionEnd |
| )", |
| R"( |
| %main = @compute @workgroup_size(1u, 1u, 1u) func():void { |
| $B1: { |
| %2:vec2<bool> = spirv.equal vec2<u32>(8u), vec2<u32>(9u) |
| ret |
| } |
| } |
| )"); |
| } |
| |
| } // namespace |
| } // namespace tint::spirv::reader |