| // Copyright 2021 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/core/intrinsic/table.h" |
| |
| #include <utility> |
| |
| #include "gmock/gmock.h" |
| #include "src/tint/lang/core/intrinsic/dialect.h" |
| #include "src/tint/lang/core/intrinsic/table_data.h" |
| #include "src/tint/lang/core/type/atomic.h" |
| #include "src/tint/lang/core/type/depth_multisampled_texture.h" |
| #include "src/tint/lang/core/type/depth_texture.h" |
| #include "src/tint/lang/core/type/external_texture.h" |
| #include "src/tint/lang/core/type/helper_test.h" |
| #include "src/tint/lang/core/type/multisampled_texture.h" |
| #include "src/tint/lang/core/type/sampled_texture.h" |
| #include "src/tint/lang/core/type/storage_texture.h" |
| #include "src/tint/lang/core/type/texture_dimension.h" |
| |
| namespace tint::core::intrinsic { |
| namespace { |
| |
| using ::testing::HasSubstr; |
| |
| class CoreIntrinsicTableTest : public testing::Test, public ProgramBuilder { |
| public: |
| Table<Dialect> table{Types(), Symbols()}; |
| }; |
| |
| TEST_F(CoreIntrinsicTableTest, MatchF32) { |
| auto* f32 = create<type::F32>(); |
| auto result = table.Lookup(BuiltinFn::kCos, Empty, Vector{f32}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, f32); |
| ASSERT_EQ(result->parameters.Length(), 1u); |
| EXPECT_EQ(result->parameters[0].type, f32); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchF32) { |
| auto* i32 = create<type::I32>(); |
| auto result = table.Lookup(BuiltinFn::kCos, Empty, Vector{i32}, EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| ASSERT_THAT(result.Failure(), HasSubstr("no matching call")); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchU32) { |
| auto* f32 = create<type::F32>(); |
| auto* u32 = create<type::U32>(); |
| auto* vec2f = create<type::Vector>(f32, 2u); |
| auto result = |
| table.Lookup(BuiltinFn::kUnpack2X16Float, Empty, Vector{u32}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, vec2f); |
| ASSERT_EQ(result->parameters.Length(), 1u); |
| EXPECT_EQ(result->parameters[0].type, u32); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchU32) { |
| auto* f32 = create<type::F32>(); |
| auto result = |
| table.Lookup(BuiltinFn::kUnpack2X16Float, Empty, Vector{f32}, EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| ASSERT_THAT(result.Failure(), HasSubstr("no matching call")); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchI32) { |
| auto* f32 = create<type::F32>(); |
| auto* i32 = create<type::I32>(); |
| auto* vec4f = create<type::Vector>(f32, 4u); |
| auto* tex = create<type::SampledTexture>(type::TextureDimension::k1d, f32); |
| auto result = table.Lookup(BuiltinFn::kTextureLoad, Empty, Vector{tex, i32, i32}, |
| EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, vec4f); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, tex); |
| EXPECT_EQ(result->parameters[0].usage, ParameterUsage::kTexture); |
| EXPECT_EQ(result->parameters[1].type, i32); |
| EXPECT_EQ(result->parameters[1].usage, ParameterUsage::kCoords); |
| EXPECT_EQ(result->parameters[2].type, i32); |
| EXPECT_EQ(result->parameters[2].usage, ParameterUsage::kLevel); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchI32) { |
| auto* f32 = create<type::F32>(); |
| auto* tex = create<type::SampledTexture>(type::TextureDimension::k1d, f32); |
| auto result = |
| table.Lookup(BuiltinFn::kTextureLoad, Empty, Vector{tex, f32}, EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| ASSERT_THAT(result.Failure(), HasSubstr("no matching call")); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchIU32AsI32) { |
| auto* i32 = create<type::I32>(); |
| auto result = |
| table.Lookup(BuiltinFn::kCountOneBits, Empty, Vector{i32}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, i32); |
| ASSERT_EQ(result->parameters.Length(), 1u); |
| EXPECT_EQ(result->parameters[0].type, i32); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchIU32AsU32) { |
| auto* u32 = create<type::U32>(); |
| auto result = |
| table.Lookup(BuiltinFn::kCountOneBits, Empty, Vector{u32}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, u32); |
| ASSERT_EQ(result->parameters.Length(), 1u); |
| EXPECT_EQ(result->parameters[0].type, u32); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchIU32) { |
| auto* f32 = create<type::F32>(); |
| auto result = |
| table.Lookup(BuiltinFn::kCountOneBits, Empty, Vector{f32}, EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| ASSERT_THAT(result.Failure(), HasSubstr("no matching call")); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchFIU32AsI32) { |
| auto* i32 = create<type::I32>(); |
| auto result = |
| table.Lookup(BuiltinFn::kClamp, Empty, Vector{i32, i32, i32}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, i32); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, i32); |
| EXPECT_EQ(result->parameters[1].type, i32); |
| EXPECT_EQ(result->parameters[2].type, i32); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchFIU32AsU32) { |
| auto* u32 = create<type::U32>(); |
| auto result = |
| table.Lookup(BuiltinFn::kClamp, Empty, Vector{u32, u32, u32}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, u32); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, u32); |
| EXPECT_EQ(result->parameters[1].type, u32); |
| EXPECT_EQ(result->parameters[2].type, u32); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchFIU32AsF32) { |
| auto* f32 = create<type::F32>(); |
| auto result = |
| table.Lookup(BuiltinFn::kClamp, Empty, Vector{f32, f32, f32}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, f32); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, f32); |
| EXPECT_EQ(result->parameters[1].type, f32); |
| EXPECT_EQ(result->parameters[2].type, f32); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchFIU32) { |
| auto* bool_ = create<type::Bool>(); |
| auto result = table.Lookup(BuiltinFn::kClamp, Empty, Vector{bool_, bool_, bool_}, |
| EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| ASSERT_THAT(result.Failure(), HasSubstr("no matching call")); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchBool) { |
| auto* f32 = create<type::F32>(); |
| auto* bool_ = create<type::Bool>(); |
| auto result = table.Lookup(BuiltinFn::kSelect, Empty, Vector{f32, f32, bool_}, |
| EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, f32); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, f32); |
| EXPECT_EQ(result->parameters[1].type, f32); |
| EXPECT_EQ(result->parameters[2].type, bool_); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchBool) { |
| auto* f32 = create<type::F32>(); |
| auto result = |
| table.Lookup(BuiltinFn::kSelect, Empty, Vector{f32, f32, f32}, EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| ASSERT_THAT(result.Failure(), HasSubstr("no matching call")); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchPointer) { |
| auto* i32 = create<type::I32>(); |
| auto* atomic_i32 = create<type::Atomic>(i32); |
| auto* ptr = create<type::Pointer>(AddressSpace::kWorkgroup, atomic_i32, Access::kReadWrite); |
| auto result = |
| table.Lookup(BuiltinFn::kAtomicLoad, Empty, Vector{ptr}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, i32); |
| ASSERT_EQ(result->parameters.Length(), 1u); |
| EXPECT_EQ(result->parameters[0].type, ptr); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchPointer) { |
| auto* i32 = create<type::I32>(); |
| auto* atomic_i32 = create<type::Atomic>(i32); |
| auto result = |
| table.Lookup(BuiltinFn::kAtomicLoad, Empty, Vector{atomic_i32}, EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| ASSERT_THAT(result.Failure(), HasSubstr("no matching call")); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchArray) { |
| auto* arr = |
| create<type::Array>(create<type::U32>(), create<type::RuntimeArrayCount>(), 4u, 4u, 4u, 4u); |
| auto* arr_ptr = create<type::Pointer>(AddressSpace::kStorage, arr, Access::kReadWrite); |
| auto result = |
| table.Lookup(BuiltinFn::kArrayLength, Empty, Vector{arr_ptr}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_TRUE(result->return_type->Is<type::U32>()); |
| ASSERT_EQ(result->parameters.Length(), 1u); |
| auto* param_type = result->parameters[0].type; |
| ASSERT_TRUE(param_type->Is<type::Pointer>()); |
| EXPECT_TRUE(param_type->As<type::Pointer>()->StoreType()->Is<type::Array>()); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchArray) { |
| auto* f32 = create<type::F32>(); |
| auto result = |
| table.Lookup(BuiltinFn::kArrayLength, Empty, Vector{f32}, EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| ASSERT_THAT(result.Failure(), HasSubstr("no matching call")); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchSampler) { |
| auto* f32 = create<type::F32>(); |
| auto* vec2f = create<type::Vector>(f32, 2u); |
| auto* vec4f = create<type::Vector>(f32, 4u); |
| auto* tex = create<type::SampledTexture>(type::TextureDimension::k2d, f32); |
| auto* sampler = create<type::Sampler>(type::SamplerKind::kSampler); |
| auto result = table.Lookup(BuiltinFn::kTextureSample, Empty, Vector{tex, sampler, vec2f}, |
| EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, vec4f); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, tex); |
| EXPECT_EQ(result->parameters[0].usage, ParameterUsage::kTexture); |
| EXPECT_EQ(result->parameters[1].type, sampler); |
| EXPECT_EQ(result->parameters[1].usage, ParameterUsage::kSampler); |
| EXPECT_EQ(result->parameters[2].type, vec2f); |
| EXPECT_EQ(result->parameters[2].usage, ParameterUsage::kCoords); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchSampler) { |
| auto* f32 = create<type::F32>(); |
| auto* vec2f = create<type::Vector>(f32, 2u); |
| auto* tex = create<type::SampledTexture>(type::TextureDimension::k2d, f32); |
| auto result = table.Lookup(BuiltinFn::kTextureSample, Empty, Vector{tex, f32, vec2f}, |
| EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| ASSERT_THAT(result.Failure(), HasSubstr("no matching call")); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchSampledTexture) { |
| auto* i32 = create<type::I32>(); |
| auto* f32 = create<type::F32>(); |
| auto* vec2i = create<type::Vector>(i32, 2u); |
| auto* vec4f = create<type::Vector>(f32, 4u); |
| auto* tex = create<type::SampledTexture>(type::TextureDimension::k2d, f32); |
| auto result = table.Lookup(BuiltinFn::kTextureLoad, Empty, Vector{tex, vec2i, i32}, |
| EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, vec4f); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, tex); |
| EXPECT_EQ(result->parameters[0].usage, ParameterUsage::kTexture); |
| EXPECT_EQ(result->parameters[1].type, vec2i); |
| EXPECT_EQ(result->parameters[1].usage, ParameterUsage::kCoords); |
| EXPECT_EQ(result->parameters[2].type, i32); |
| EXPECT_EQ(result->parameters[2].usage, ParameterUsage::kLevel); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchMultisampledTexture) { |
| auto* i32 = create<type::I32>(); |
| auto* f32 = create<type::F32>(); |
| auto* vec2i = create<type::Vector>(i32, 2u); |
| auto* vec4f = create<type::Vector>(f32, 4u); |
| auto* tex = create<type::MultisampledTexture>(type::TextureDimension::k2d, f32); |
| auto result = table.Lookup(BuiltinFn::kTextureLoad, Empty, Vector{tex, vec2i, i32}, |
| EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, vec4f); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, tex); |
| EXPECT_EQ(result->parameters[0].usage, ParameterUsage::kTexture); |
| EXPECT_EQ(result->parameters[1].type, vec2i); |
| EXPECT_EQ(result->parameters[1].usage, ParameterUsage::kCoords); |
| EXPECT_EQ(result->parameters[2].type, i32); |
| EXPECT_EQ(result->parameters[2].usage, ParameterUsage::kSampleIndex); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchDepthTexture) { |
| auto* f32 = create<type::F32>(); |
| auto* i32 = create<type::I32>(); |
| auto* vec2i = create<type::Vector>(i32, 2u); |
| auto* tex = create<type::DepthTexture>(type::TextureDimension::k2d); |
| auto result = table.Lookup(BuiltinFn::kTextureLoad, Empty, Vector{tex, vec2i, i32}, |
| EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, f32); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, tex); |
| EXPECT_EQ(result->parameters[0].usage, ParameterUsage::kTexture); |
| EXPECT_EQ(result->parameters[1].type, vec2i); |
| EXPECT_EQ(result->parameters[1].usage, ParameterUsage::kCoords); |
| EXPECT_EQ(result->parameters[2].type, i32); |
| EXPECT_EQ(result->parameters[2].usage, ParameterUsage::kLevel); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchDepthMultisampledTexture) { |
| auto* f32 = create<type::F32>(); |
| auto* i32 = create<type::I32>(); |
| auto* vec2i = create<type::Vector>(i32, 2u); |
| auto* tex = create<type::DepthMultisampledTexture>(type::TextureDimension::k2d); |
| auto result = table.Lookup(BuiltinFn::kTextureLoad, Empty, Vector{tex, vec2i, i32}, |
| EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, f32); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, tex); |
| EXPECT_EQ(result->parameters[0].usage, ParameterUsage::kTexture); |
| EXPECT_EQ(result->parameters[1].type, vec2i); |
| EXPECT_EQ(result->parameters[1].usage, ParameterUsage::kCoords); |
| EXPECT_EQ(result->parameters[2].type, i32); |
| EXPECT_EQ(result->parameters[2].usage, ParameterUsage::kSampleIndex); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchExternalTexture) { |
| auto* f32 = create<type::F32>(); |
| auto* i32 = create<type::I32>(); |
| auto* vec2i = create<type::Vector>(i32, 2u); |
| auto* vec4f = create<type::Vector>(f32, 4u); |
| auto* tex = create<type::ExternalTexture>(); |
| auto result = table.Lookup(BuiltinFn::kTextureLoad, Empty, Vector{tex, vec2i}, |
| EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, vec4f); |
| ASSERT_EQ(result->parameters.Length(), 2u); |
| EXPECT_EQ(result->parameters[0].type, tex); |
| EXPECT_EQ(result->parameters[0].usage, ParameterUsage::kTexture); |
| EXPECT_EQ(result->parameters[1].type, vec2i); |
| EXPECT_EQ(result->parameters[1].usage, ParameterUsage::kCoords); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchWOStorageTexture) { |
| auto* f32 = create<type::F32>(); |
| auto* i32 = create<type::I32>(); |
| auto* vec2i = create<type::Vector>(i32, 2u); |
| auto* vec4f = create<type::Vector>(f32, 4u); |
| auto* subtype = type::StorageTexture::SubtypeFor(TexelFormat::kR32Float, Types()); |
| auto* tex = create<type::StorageTexture>(type::TextureDimension::k2d, TexelFormat::kR32Float, |
| Access::kWrite, subtype); |
| |
| auto result = table.Lookup(BuiltinFn::kTextureStore, Empty, Vector{tex, vec2i, vec4f}, |
| EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_TRUE(result->return_type->Is<type::Void>()); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, tex); |
| EXPECT_EQ(result->parameters[0].usage, ParameterUsage::kTexture); |
| EXPECT_EQ(result->parameters[1].type, vec2i); |
| EXPECT_EQ(result->parameters[1].usage, ParameterUsage::kCoords); |
| EXPECT_EQ(result->parameters[2].type, vec4f); |
| EXPECT_EQ(result->parameters[2].usage, ParameterUsage::kValue); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchTexture) { |
| auto* f32 = create<type::F32>(); |
| auto* i32 = create<type::I32>(); |
| auto* vec2i = create<type::Vector>(i32, 2u); |
| auto result = table.Lookup(BuiltinFn::kTextureLoad, Empty, Vector{f32, vec2i}, |
| EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| ASSERT_THAT(result.Failure(), HasSubstr("no matching call")); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchTemplateType) { |
| auto* f32 = create<type::F32>(); |
| auto result = |
| table.Lookup(BuiltinFn::kClamp, Empty, Vector{f32, f32, f32}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, f32); |
| EXPECT_EQ(result->parameters[0].type, f32); |
| EXPECT_EQ(result->parameters[1].type, f32); |
| EXPECT_EQ(result->parameters[2].type, f32); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchTemplateType) { |
| auto* f32 = create<type::F32>(); |
| auto* u32 = create<type::U32>(); |
| auto result = |
| table.Lookup(BuiltinFn::kClamp, Empty, Vector{f32, u32, f32}, EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| ASSERT_THAT(result.Failure(), HasSubstr("no matching call")); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchOpenSizeVector) { |
| auto* f32 = create<type::F32>(); |
| auto* vec2f = create<type::Vector>(f32, 2u); |
| auto result = table.Lookup(BuiltinFn::kClamp, Empty, Vector{vec2f, vec2f, vec2f}, |
| EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, vec2f); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, vec2f); |
| EXPECT_EQ(result->parameters[1].type, vec2f); |
| EXPECT_EQ(result->parameters[2].type, vec2f); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchOpenSizeVector) { |
| auto* f32 = create<type::F32>(); |
| auto* u32 = create<type::U32>(); |
| auto* vec2f = create<type::Vector>(f32, 2u); |
| auto result = table.Lookup(BuiltinFn::kClamp, Empty, Vector{vec2f, u32, vec2f}, |
| EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| ASSERT_THAT(result.Failure(), HasSubstr("no matching call")); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchOpenSizeMatrix) { |
| auto* f32 = create<type::F32>(); |
| auto* vec3f = create<type::Vector>(f32, 3u); |
| auto* mat3x3f = create<type::Matrix>(vec3f, 3u); |
| auto result = |
| table.Lookup(BuiltinFn::kDeterminant, Empty, Vector{mat3x3f}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, f32); |
| ASSERT_EQ(result->parameters.Length(), 1u); |
| EXPECT_EQ(result->parameters[0].type, mat3x3f); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchOpenSizeMatrix) { |
| auto* f32 = create<type::F32>(); |
| auto* vec2f = create<type::Vector>(f32, 2u); |
| auto* mat3x2f = create<type::Matrix>(vec2f, 3u); |
| auto result = |
| table.Lookup(BuiltinFn::kDeterminant, Empty, Vector{mat3x2f}, EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| ASSERT_THAT(result.Failure(), HasSubstr("no matching call")); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchDifferentArgsElementType_Builtin_ConstantEval) { |
| auto* f32 = create<type::F32>(); |
| auto* bool_ = create<type::Bool>(); |
| auto result = table.Lookup(BuiltinFn::kSelect, Empty, Vector{f32, f32, bool_}, |
| EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_NE(result->const_eval_fn, nullptr); |
| EXPECT_EQ(result->return_type, f32); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, f32); |
| EXPECT_EQ(result->parameters[1].type, f32); |
| EXPECT_EQ(result->parameters[2].type, bool_); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchDifferentArgsElementType_Builtin_RuntimeEval) { |
| auto* f32 = create<type::F32>(); |
| auto result = table.Lookup(BuiltinFn::kSelect, Empty, Vector{f32, f32, create<type::Bool>()}, |
| EvaluationStage::kRuntime); |
| ASSERT_EQ(result, Success); |
| EXPECT_NE(result->const_eval_fn, nullptr); |
| EXPECT_TRUE(result->return_type->Is<type::F32>()); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_TRUE(result->parameters[0].type->Is<type::F32>()); |
| EXPECT_TRUE(result->parameters[1].type->Is<type::F32>()); |
| EXPECT_TRUE(result->parameters[2].type->Is<type::Bool>()); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchDifferentArgsElementType_Binary_ConstantEval) { |
| auto* i32 = create<type::I32>(); |
| auto* u32 = create<type::U32>(); |
| auto result = table.Lookup(BinaryOp::kShiftLeft, i32, u32, EvaluationStage::kConstant, false); |
| ASSERT_EQ(result, Success); |
| ASSERT_NE(result->const_eval_fn, nullptr) << Diagnostics(); |
| EXPECT_EQ(result->return_type, i32); |
| EXPECT_EQ(result->parameters[0].type, i32); |
| EXPECT_EQ(result->parameters[1].type, u32); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchDifferentArgsElementType_Binary_RuntimeEval) { |
| auto* i32 = create<type::I32>(); |
| auto* u32 = create<type::U32>(); |
| auto result = table.Lookup(BinaryOp::kShiftLeft, i32, u32, EvaluationStage::kRuntime, false); |
| ASSERT_EQ(result, Success); |
| ASSERT_NE(result->const_eval_fn, nullptr) << Diagnostics(); |
| EXPECT_TRUE(result->return_type->Is<type::I32>()); |
| EXPECT_TRUE(result->parameters[0].type->Is<type::I32>()); |
| EXPECT_TRUE(result->parameters[1].type->Is<type::U32>()); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, OverloadOrderByNumberOfParameters) { |
| // None of the arguments match, so expect the overloads with 2 parameters to |
| // come first |
| auto* bool_ = create<type::Bool>(); |
| auto result = table.Lookup(BuiltinFn::kTextureDimensions, Empty, Vector{bool_, bool_}, |
| EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| ASSERT_EQ(result.Failure(), |
| R"(no matching call to textureDimensions(bool, bool) |
| |
| 27 candidate functions: |
| textureDimensions(texture: texture_1d<T>, level: L) -> u32 where: T is f32, i32 or u32, L is i32 or u32 |
| textureDimensions(texture: texture_2d<T>, level: L) -> vec2<u32> where: T is f32, i32 or u32, L is i32 or u32 |
| textureDimensions(texture: texture_2d_array<T>, level: L) -> vec2<u32> where: T is f32, i32 or u32, L is i32 or u32 |
| textureDimensions(texture: texture_3d<T>, level: L) -> vec3<u32> where: T is f32, i32 or u32, L is i32 or u32 |
| textureDimensions(texture: texture_cube<T>, level: L) -> vec2<u32> where: T is f32, i32 or u32, L is i32 or u32 |
| textureDimensions(texture: texture_cube_array<T>, level: L) -> vec2<u32> where: T is f32, i32 or u32, L is i32 or u32 |
| textureDimensions(texture: texture_depth_2d, level: L) -> vec2<u32> where: L is i32 or u32 |
| textureDimensions(texture: texture_depth_2d_array, level: L) -> vec2<u32> where: L is i32 or u32 |
| textureDimensions(texture: texture_depth_cube, level: L) -> vec2<u32> where: L is i32 or u32 |
| textureDimensions(texture: texture_depth_cube_array, level: L) -> vec2<u32> where: L is i32 or u32 |
| textureDimensions(texture: texture_1d<T>) -> u32 where: T is f32, i32 or u32 |
| textureDimensions(texture: texture_2d<T>) -> vec2<u32> where: T is f32, i32 or u32 |
| textureDimensions(texture: texture_2d_array<T>) -> vec2<u32> where: T is f32, i32 or u32 |
| textureDimensions(texture: texture_3d<T>) -> vec3<u32> where: T is f32, i32 or u32 |
| textureDimensions(texture: texture_cube<T>) -> vec2<u32> where: T is f32, i32 or u32 |
| textureDimensions(texture: texture_cube_array<T>) -> vec2<u32> where: T is f32, i32 or u32 |
| textureDimensions(texture: texture_multisampled_2d<T>) -> vec2<u32> where: T is f32, i32 or u32 |
| textureDimensions(texture: texture_depth_2d) -> vec2<u32> |
| textureDimensions(texture: texture_depth_2d_array) -> vec2<u32> |
| textureDimensions(texture: texture_depth_cube) -> vec2<u32> |
| textureDimensions(texture: texture_depth_cube_array) -> vec2<u32> |
| textureDimensions(texture: texture_depth_multisampled_2d) -> vec2<u32> |
| textureDimensions(texture: texture_storage_1d<F, A>) -> u32 |
| textureDimensions(texture: texture_storage_2d<F, A>) -> vec2<u32> |
| textureDimensions(texture: texture_storage_2d_array<F, A>) -> vec2<u32> |
| textureDimensions(texture: texture_storage_3d<F, A>) -> vec3<u32> |
| textureDimensions(texture: texture_external) -> vec2<u32> |
| )"); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, OverloadOrderByMatchingParameter) { |
| auto* tex = create<type::DepthTexture>(type::TextureDimension::k2d); |
| auto* bool_ = create<type::Bool>(); |
| auto result = table.Lookup(BuiltinFn::kTextureDimensions, Empty, Vector{tex, bool_}, |
| EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| ASSERT_EQ(result.Failure(), |
| R"(no matching call to textureDimensions(texture_depth_2d, bool) |
| |
| 27 candidate functions: |
| textureDimensions(texture: texture_depth_2d, level: L) -> vec2<u32> where: L is i32 or u32 |
| textureDimensions(texture: texture_1d<T>, level: L) -> u32 where: T is f32, i32 or u32, L is i32 or u32 |
| textureDimensions(texture: texture_2d<T>, level: L) -> vec2<u32> where: T is f32, i32 or u32, L is i32 or u32 |
| textureDimensions(texture: texture_2d_array<T>, level: L) -> vec2<u32> where: T is f32, i32 or u32, L is i32 or u32 |
| textureDimensions(texture: texture_3d<T>, level: L) -> vec3<u32> where: T is f32, i32 or u32, L is i32 or u32 |
| textureDimensions(texture: texture_cube<T>, level: L) -> vec2<u32> where: T is f32, i32 or u32, L is i32 or u32 |
| textureDimensions(texture: texture_cube_array<T>, level: L) -> vec2<u32> where: T is f32, i32 or u32, L is i32 or u32 |
| textureDimensions(texture: texture_depth_2d_array, level: L) -> vec2<u32> where: L is i32 or u32 |
| textureDimensions(texture: texture_depth_cube, level: L) -> vec2<u32> where: L is i32 or u32 |
| textureDimensions(texture: texture_depth_cube_array, level: L) -> vec2<u32> where: L is i32 or u32 |
| textureDimensions(texture: texture_depth_2d) -> vec2<u32> |
| textureDimensions(texture: texture_1d<T>) -> u32 where: T is f32, i32 or u32 |
| textureDimensions(texture: texture_2d<T>) -> vec2<u32> where: T is f32, i32 or u32 |
| textureDimensions(texture: texture_2d_array<T>) -> vec2<u32> where: T is f32, i32 or u32 |
| textureDimensions(texture: texture_3d<T>) -> vec3<u32> where: T is f32, i32 or u32 |
| textureDimensions(texture: texture_cube<T>) -> vec2<u32> where: T is f32, i32 or u32 |
| textureDimensions(texture: texture_cube_array<T>) -> vec2<u32> where: T is f32, i32 or u32 |
| textureDimensions(texture: texture_multisampled_2d<T>) -> vec2<u32> where: T is f32, i32 or u32 |
| textureDimensions(texture: texture_depth_2d_array) -> vec2<u32> |
| textureDimensions(texture: texture_depth_cube) -> vec2<u32> |
| textureDimensions(texture: texture_depth_cube_array) -> vec2<u32> |
| textureDimensions(texture: texture_depth_multisampled_2d) -> vec2<u32> |
| textureDimensions(texture: texture_storage_1d<F, A>) -> u32 |
| textureDimensions(texture: texture_storage_2d<F, A>) -> vec2<u32> |
| textureDimensions(texture: texture_storage_2d_array<F, A>) -> vec2<u32> |
| textureDimensions(texture: texture_storage_3d<F, A>) -> vec3<u32> |
| textureDimensions(texture: texture_external) -> vec2<u32> |
| )"); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchUnaryOp) { |
| auto* i32 = create<type::I32>(); |
| auto* vec3i = create<type::Vector>(i32, 3u); |
| auto result = table.Lookup(UnaryOp::kNegation, vec3i, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, vec3i); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchUnaryOp) { |
| auto* bool_ = create<type::Bool>(); |
| auto result = table.Lookup(UnaryOp::kNegation, bool_, EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| EXPECT_EQ(result.Failure(), R"(no matching overload for operator - (bool) |
| |
| 2 candidate operators: |
| operator - (T) -> T where: T is f32, i32 or f16 |
| operator - (vecN<T>) -> vecN<T> where: T is f32, i32 or f16 |
| )"); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchUnaryOp_Constant) { |
| auto* i32 = create<type::I32>(); |
| auto result = table.Lookup(UnaryOp::kNegation, i32, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, i32); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchUnaryOp_Runtime) { |
| auto* i32 = create<type::I32>(); |
| auto result = table.Lookup(UnaryOp::kNegation, i32, EvaluationStage::kRuntime); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, i32); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchBinaryOp) { |
| auto* i32 = create<type::I32>(); |
| auto* vec3i = create<type::Vector>(i32, 3u); |
| auto result = table.Lookup(BinaryOp::kMultiply, i32, vec3i, EvaluationStage::kConstant, |
| /* is_compound */ false); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, vec3i); |
| EXPECT_EQ(result->parameters[0].type, i32); |
| EXPECT_EQ(result->parameters[1].type, vec3i); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchBinaryOp) { |
| auto* f32 = create<type::F32>(); |
| auto* bool_ = create<type::Bool>(); |
| auto result = table.Lookup(BinaryOp::kMultiply, f32, bool_, EvaluationStage::kConstant, |
| /* is_compound */ false); |
| ASSERT_NE(result, Success); |
| EXPECT_EQ(result.Failure(), R"(no matching overload for operator * (f32, bool) |
| |
| 9 candidate operators: |
| operator * (T, T) -> T where: T is f32, i32, u32 or f16 |
| operator * (vecN<T>, T) -> vecN<T> where: T is f32, i32, u32 or f16 |
| operator * (T, vecN<T>) -> vecN<T> where: T is f32, i32, u32 or f16 |
| operator * (T, matNxM<T>) -> matNxM<T> where: T is f32 or f16 |
| operator * (matNxM<T>, T) -> matNxM<T> where: T is f32 or f16 |
| operator * (vecN<T>, vecN<T>) -> vecN<T> where: T is f32, i32, u32 or f16 |
| operator * (matCxR<T>, vecC<T>) -> vecR<T> where: T is f32 or f16 |
| operator * (vecR<T>, matCxR<T>) -> vecC<T> where: T is f32 or f16 |
| operator * (matKxR<T>, matCxK<T>) -> matCxR<T> where: T is f32 or f16 |
| )"); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchCompoundOp) { |
| auto* i32 = create<type::I32>(); |
| auto* vec3i = create<type::Vector>(i32, 3u); |
| auto result = table.Lookup(BinaryOp::kMultiply, i32, vec3i, EvaluationStage::kConstant, |
| /* is_compound */ true); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, vec3i); |
| EXPECT_EQ(result->parameters[0].type, i32); |
| EXPECT_EQ(result->parameters[1].type, vec3i); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchCompoundOp) { |
| auto* f32 = create<type::F32>(); |
| auto* bool_ = create<type::Bool>(); |
| auto result = table.Lookup(BinaryOp::kMultiply, f32, bool_, EvaluationStage::kConstant, |
| /* is_compound */ true); |
| ASSERT_NE(result, Success); |
| EXPECT_EQ(result.Failure(), R"(no matching overload for operator *= (f32, bool) |
| |
| 9 candidate operators: |
| operator *= (T, T) -> T where: T is f32, i32, u32 or f16 |
| operator *= (vecN<T>, T) -> vecN<T> where: T is f32, i32, u32 or f16 |
| operator *= (T, vecN<T>) -> vecN<T> where: T is f32, i32, u32 or f16 |
| operator *= (T, matNxM<T>) -> matNxM<T> where: T is f32 or f16 |
| operator *= (matNxM<T>, T) -> matNxM<T> where: T is f32 or f16 |
| operator *= (vecN<T>, vecN<T>) -> vecN<T> where: T is f32, i32, u32 or f16 |
| operator *= (matCxR<T>, vecC<T>) -> vecR<T> where: T is f32 or f16 |
| operator *= (vecR<T>, matCxR<T>) -> vecC<T> where: T is f32 or f16 |
| operator *= (matKxR<T>, matCxK<T>) -> matCxR<T> where: T is f32 or f16 |
| )"); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchTypeInitializerImplicit) { |
| auto* i32 = create<type::I32>(); |
| auto* vec3i = create<type::Vector>(i32, 3u); |
| auto result = |
| table.Lookup(CtorConv::kVec3, Empty, Vector{i32, i32, i32}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, vec3i); |
| EXPECT_TRUE(result->info->flags.Contains(OverloadFlag::kIsConstructor)); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, i32); |
| EXPECT_EQ(result->parameters[1].type, i32); |
| EXPECT_EQ(result->parameters[2].type, i32); |
| EXPECT_NE(result->const_eval_fn, nullptr); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchTypeInitializerExplicit) { |
| auto* i32 = create<type::I32>(); |
| auto* vec3i = create<type::Vector>(i32, 3u); |
| auto result = table.Lookup(CtorConv::kVec3, Vector{i32}, Vector{i32, i32, i32}, |
| EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, vec3i); |
| EXPECT_TRUE(result->info->flags.Contains(OverloadFlag::kIsConstructor)); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, i32); |
| EXPECT_EQ(result->parameters[1].type, i32); |
| EXPECT_EQ(result->parameters[2].type, i32); |
| EXPECT_NE(result->const_eval_fn, nullptr); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchTypeInitializerImplicit) { |
| auto* i32 = create<type::I32>(); |
| auto* f32 = create<type::F32>(); |
| auto result = |
| table.Lookup(CtorConv::kVec3, Empty, Vector{i32, f32, i32}, EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| EXPECT_EQ(result.Failure(), |
| R"(no matching constructor for vec3(i32, f32, i32) |
| |
| 11 candidate constructors: |
| vec3(x: T, y: T, z: T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3<T>(xy: vec2<T>, z: T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3(xy: vec2<T>, z: T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3<T>(x: T, yz: vec2<T>) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3(x: T, yz: vec2<T>) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3<T>(T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3(T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3<T>(vec3<T>) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3(vec3<T>) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3<T>() -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3<T>(x: T, y: T, z: T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| |
| 5 candidate conversions: |
| vec3<T>(vec3<U>) -> vec3<T> where: T is f32, U is i32, f16, u32 or bool |
| vec3<T>(vec3<U>) -> vec3<T> where: T is f16, U is f32, i32, u32 or bool |
| vec3<T>(vec3<U>) -> vec3<T> where: T is i32, U is f32, f16, u32 or bool |
| vec3<T>(vec3<U>) -> vec3<T> where: T is u32, U is f32, f16, i32 or bool |
| vec3<T>(vec3<U>) -> vec3<T> where: T is bool, U is f32, f16, i32 or u32 |
| )"); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchTypeInitializerExplicit) { |
| auto* i32 = create<type::I32>(); |
| auto* f32 = create<type::F32>(); |
| auto result = table.Lookup(CtorConv::kVec3, Vector{i32}, Vector{i32, f32, i32}, |
| EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| EXPECT_EQ(result.Failure(), |
| R"(no matching constructor for vec3<i32>(i32, f32, i32) |
| |
| 11 candidate constructors: |
| vec3<T>(x: T, y: T, z: T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3<T>(xy: vec2<T>, z: T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3(xy: vec2<T>, z: T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3<T>(x: T, yz: vec2<T>) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3(x: T, yz: vec2<T>) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3<T>(T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3(T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3<T>(vec3<T>) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3(vec3<T>) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3<T>() -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3(x: T, y: T, z: T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| |
| 5 candidate conversions: |
| vec3<T>(vec3<U>) -> vec3<T> where: T is f32, U is i32, f16, u32 or bool |
| vec3<T>(vec3<U>) -> vec3<T> where: T is f16, U is f32, i32, u32 or bool |
| vec3<T>(vec3<U>) -> vec3<T> where: T is i32, U is f32, f16, u32 or bool |
| vec3<T>(vec3<U>) -> vec3<T> where: T is u32, U is f32, f16, i32 or bool |
| vec3<T>(vec3<U>) -> vec3<T> where: T is bool, U is f32, f16, i32 or u32 |
| )"); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchTypeInitializerImplicitVecFromVecAbstract) { |
| auto* i32 = create<type::I32>(); |
| auto* vec3i = create<type::Vector>(i32, 3u); |
| auto result = table.Lookup(CtorConv::kVec3, Empty, Vector{vec3i}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, vec3i); |
| EXPECT_TRUE(result->info->flags.Contains(OverloadFlag::kIsConstructor)); |
| ASSERT_EQ(result->parameters.Length(), 1u); |
| EXPECT_EQ(result->parameters[0].type, vec3i); |
| EXPECT_NE(result->const_eval_fn, nullptr); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchTypeInitializerImplicitMatFromVec) { |
| auto* f32 = create<type::F32>(); |
| auto* vec2f = create<type::Vector>(f32, 2u); |
| auto* mat2x2_af = create<type::Matrix>(vec2f, 2u); |
| auto result = |
| table.Lookup(CtorConv::kMat2x2, Empty, Vector{vec2f, vec2f}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_TYPE(result->return_type, mat2x2_af); |
| EXPECT_TRUE(result->info->flags.Contains(OverloadFlag::kIsConstructor)); |
| ASSERT_EQ(result->parameters.Length(), 2u); |
| EXPECT_TYPE(result->parameters[0].type, vec2f); |
| EXPECT_TYPE(result->parameters[1].type, vec2f); |
| EXPECT_NE(result->const_eval_fn, nullptr); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchTypeInitializer_ConstantEval) { |
| auto* i32 = create<type::I32>(); |
| auto* vec3i = create<type::Vector>(i32, 3u); |
| auto result = |
| table.Lookup(CtorConv::kVec3, Empty, Vector{i32, i32, i32}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_NE(result->const_eval_fn, nullptr); |
| EXPECT_EQ(result->return_type, vec3i); |
| EXPECT_TRUE(result->info->flags.Contains(OverloadFlag::kIsConstructor)); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, i32); |
| EXPECT_EQ(result->parameters[1].type, i32); |
| EXPECT_EQ(result->parameters[2].type, i32); |
| EXPECT_NE(result->const_eval_fn, nullptr); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchTypeInitializer_RuntimeEval) { |
| auto* i32 = create<type::I32>(); |
| auto result = |
| table.Lookup(CtorConv::kVec3, Empty, Vector{i32, i32, i32}, EvaluationStage::kRuntime); |
| auto* vec3i = create<type::Vector>(i32, 3u); |
| ASSERT_EQ(result, Success); |
| EXPECT_NE(result->const_eval_fn, nullptr); |
| EXPECT_EQ(result->return_type, vec3i); |
| EXPECT_TRUE(result->info->flags.Contains(OverloadFlag::kIsConstructor)); |
| ASSERT_EQ(result->parameters.Length(), 3u); |
| EXPECT_EQ(result->parameters[0].type, i32); |
| EXPECT_EQ(result->parameters[1].type, i32); |
| EXPECT_EQ(result->parameters[2].type, i32); |
| EXPECT_NE(result->const_eval_fn, nullptr); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchTypeConversion) { |
| auto* i32 = create<type::I32>(); |
| auto* vec3i = create<type::Vector>(i32, 3u); |
| auto* f32 = create<type::F32>(); |
| auto* vec3f = create<type::Vector>(f32, 3u); |
| auto result = |
| table.Lookup(CtorConv::kVec3, Vector{i32}, Vector{vec3f}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_EQ(result->return_type, vec3i); |
| EXPECT_FALSE(result->info->flags.Contains(OverloadFlag::kIsConstructor)); |
| ASSERT_EQ(result->parameters.Length(), 1u); |
| EXPECT_EQ(result->parameters[0].type, vec3f); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MismatchTypeConversion) { |
| auto* arr = |
| create<type::Array>(create<type::U32>(), create<type::RuntimeArrayCount>(), 4u, 4u, 4u, 4u); |
| auto* f32 = create<type::F32>(); |
| auto result = |
| table.Lookup(CtorConv::kVec3, Vector{f32}, Vector{arr}, EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| EXPECT_EQ(result.Failure(), |
| R"(no matching constructor for vec3<f32>(array<u32>) |
| |
| 11 candidate constructors: |
| vec3<T>(vec3<T>) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3<T>(T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3<T>() -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3<T>(x: T, yz: vec2<T>) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3(x: T, yz: vec2<T>) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3<T>(xy: vec2<T>, z: T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3(xy: vec2<T>, z: T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3<T>(x: T, y: T, z: T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3(x: T, y: T, z: T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3(T) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| vec3(vec3<T>) -> vec3<T> where: T is f32, f16, i32, u32 or bool |
| |
| 5 candidate conversions: |
| vec3<T>(vec3<U>) -> vec3<T> where: T is f32, U is i32, f16, u32 or bool |
| vec3<T>(vec3<U>) -> vec3<T> where: T is f16, U is f32, i32, u32 or bool |
| vec3<T>(vec3<U>) -> vec3<T> where: T is i32, U is f32, f16, u32 or bool |
| vec3<T>(vec3<U>) -> vec3<T> where: T is u32, U is f32, f16, i32 or bool |
| vec3<T>(vec3<U>) -> vec3<T> where: T is bool, U is f32, f16, i32 or u32 |
| )"); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchTypeConversion_ConstantEval) { |
| auto* i32 = create<type::I32>(); |
| auto* f32 = create<type::F32>(); |
| auto* vec3i = create<type::Vector>(i32, 3u); |
| auto* vec3f = create<type::Vector>(f32, 3u); |
| auto result = |
| table.Lookup(CtorConv::kVec3, Vector{f32}, Vector{vec3i}, EvaluationStage::kConstant); |
| ASSERT_EQ(result, Success); |
| EXPECT_NE(result->const_eval_fn, nullptr); |
| // NOTE: Conversions are explicit, so there's no way to have it return abstracts |
| EXPECT_EQ(result->return_type, vec3f); |
| EXPECT_FALSE(result->info->flags.Contains(OverloadFlag::kIsConstructor)); |
| ASSERT_EQ(result->parameters.Length(), 1u); |
| EXPECT_EQ(result->parameters[0].type, vec3i); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, MatchTypeConversion_RuntimeEval) { |
| auto* i32 = create<type::I32>(); |
| auto* f32 = create<type::F32>(); |
| auto* vec3i = create<type::Vector>(i32, 3u); |
| auto* vec3f = create<type::Vector>(create<type::F32>(), 3u); |
| auto result = |
| table.Lookup(CtorConv::kVec3, Vector{f32}, Vector{vec3i}, EvaluationStage::kRuntime); |
| ASSERT_EQ(result, Success); |
| EXPECT_NE(result->const_eval_fn, nullptr); |
| EXPECT_EQ(result->return_type, vec3f); |
| EXPECT_FALSE(result->info->flags.Contains(OverloadFlag::kIsConstructor)); |
| ASSERT_EQ(result->parameters.Length(), 1u); |
| EXPECT_EQ(result->parameters[0].type, vec3i); |
| } |
| |
| TEST_F(CoreIntrinsicTableTest, Err257Arguments) { // crbug.com/1323605 |
| auto* f32 = create<type::F32>(); |
| Vector<const type::Type*, 0> arg_tys; |
| arg_tys.Resize(257, f32); |
| auto result = |
| table.Lookup(BuiltinFn::kAbs, Empty, std::move(arg_tys), EvaluationStage::kConstant); |
| ASSERT_NE(result, Success); |
| ASSERT_THAT(result.Failure(), HasSubstr("no matching call")); |
| } |
| |
| } // namespace |
| } // namespace tint::core::intrinsic |