| // 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 "src/transform/bound_array_accessors.h" |
| |
| #include "src/transform/test_helper.h" |
| |
| namespace tint { |
| namespace transform { |
| namespace { |
| |
| using BoundArrayAccessorsTest = TransformTest; |
| |
| TEST_F(BoundArrayAccessorsTest, Ptrs_Clamp) { |
| auto* src = R"( |
| var<private> a : array<f32, 3>; |
| |
| let c : u32 = 1u; |
| |
| fn f() { |
| let b : f32 = a[c]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : array<f32, 3>; |
| |
| let c : u32 = 1u; |
| |
| fn f() { |
| let b : f32 = a[min(u32(c), 2u)]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Array_Idx_Nested_Scalar) { |
| auto* src = R"( |
| var<private> a : array<f32, 3>; |
| |
| var<private> b : array<f32, 5>; |
| |
| var<private> i : u32; |
| |
| fn f() { |
| var c : f32 = a[ b[i] ]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : array<f32, 3>; |
| |
| var<private> b : array<f32, 5>; |
| |
| var<private> i : u32; |
| |
| fn f() { |
| var c : f32 = a[min(u32(b[min(u32(i), 4u)]), 2u)]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Array_Idx_Scalar) { |
| auto* src = R"( |
| var<private> a : array<f32, 3>; |
| |
| fn f() { |
| var b : f32 = a[1]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : array<f32, 3>; |
| |
| fn f() { |
| var b : f32 = a[1]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Array_Idx_Expr) { |
| auto* src = R"( |
| var<private> a : array<f32, 3>; |
| |
| var<private> c : i32; |
| |
| fn f() { |
| var b : f32 = a[c + 2 - 3]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : array<f32, 3>; |
| |
| var<private> c : i32; |
| |
| fn f() { |
| var b : f32 = a[min(u32(((c + 2) - 3)), 2u)]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Array_Idx_Negative) { |
| auto* src = R"( |
| var<private> a : array<f32, 3>; |
| |
| fn f() { |
| var b : f32 = a[-1]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : array<f32, 3>; |
| |
| fn f() { |
| var b : f32 = a[0]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Array_Idx_OutOfBounds) { |
| auto* src = R"( |
| var<private> a : array<f32, 3>; |
| |
| fn f() { |
| var b : f32 = a[3]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : array<f32, 3>; |
| |
| fn f() { |
| var b : f32 = a[2]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Vector_Idx_Scalar) { |
| auto* src = R"( |
| var<private> a : vec3<f32>; |
| |
| fn f() { |
| var b : f32 = a[1]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : vec3<f32>; |
| |
| fn f() { |
| var b : f32 = a[1]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Vector_Idx_Expr) { |
| auto* src = R"( |
| var<private> a : vec3<f32>; |
| |
| var<private> c : i32; |
| |
| fn f() { |
| var b : f32 = a[c + 2 - 3]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : vec3<f32>; |
| |
| var<private> c : i32; |
| |
| fn f() { |
| var b : f32 = a[min(u32(((c + 2) - 3)), 2u)]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Vector_Swizzle_Idx_Scalar) { |
| auto* src = R"( |
| var<private> a : vec3<f32>; |
| |
| fn f() { |
| var b : f32 = a.xy[2]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : vec3<f32>; |
| |
| fn f() { |
| var b : f32 = a.xy[1]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Vector_Swizzle_Idx_Var) { |
| auto* src = R"( |
| var<private> a : vec3<f32>; |
| |
| var<private> c : i32; |
| |
| fn f() { |
| var b : f32 = a.xy[c]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : vec3<f32>; |
| |
| var<private> c : i32; |
| |
| fn f() { |
| var b : f32 = a.xy[min(u32(c), 1u)]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| TEST_F(BoundArrayAccessorsTest, Vector_Swizzle_Idx_Expr) { |
| auto* src = R"( |
| var<private> a : vec3<f32>; |
| |
| var<private> c : i32; |
| |
| fn f() { |
| var b : f32 = a.xy[c + 2 - 3]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : vec3<f32>; |
| |
| var<private> c : i32; |
| |
| fn f() { |
| var b : f32 = a.xy[min(u32(((c + 2) - 3)), 1u)]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Vector_Idx_Negative) { |
| auto* src = R"( |
| var<private> a : vec3<f32>; |
| |
| fn f() { |
| var b : f32 = a[-1]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : vec3<f32>; |
| |
| fn f() { |
| var b : f32 = a[0]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Vector_Idx_OutOfBounds) { |
| auto* src = R"( |
| var<private> a : vec3<f32>; |
| |
| fn f() { |
| var b : f32 = a[3]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : vec3<f32>; |
| |
| fn f() { |
| var b : f32 = a[2]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Matrix_Idx_Scalar) { |
| auto* src = R"( |
| var<private> a : mat3x2<f32>; |
| |
| fn f() { |
| var b : f32 = a[2][1]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : mat3x2<f32>; |
| |
| fn f() { |
| var b : f32 = a[2][1]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Matrix_Idx_Expr_Column) { |
| auto* src = R"( |
| var<private> a : mat3x2<f32>; |
| |
| var<private> c : i32; |
| |
| fn f() { |
| var b : f32 = a[c + 2 - 3][1]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : mat3x2<f32>; |
| |
| var<private> c : i32; |
| |
| fn f() { |
| var b : f32 = a[min(u32(((c + 2) - 3)), 2u)][1]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Matrix_Idx_Expr_Row) { |
| auto* src = R"( |
| var<private> a : mat3x2<f32>; |
| |
| var<private> c : i32; |
| |
| fn f() { |
| var b : f32 = a[1][c + 2 - 3]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : mat3x2<f32>; |
| |
| var<private> c : i32; |
| |
| fn f() { |
| var b : f32 = a[1][min(u32(((c + 2) - 3)), 1u)]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Matrix_Idx_Negative_Column) { |
| auto* src = R"( |
| var<private> a : mat3x2<f32>; |
| |
| fn f() { |
| var b : f32 = a[-1][1]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : mat3x2<f32>; |
| |
| fn f() { |
| var b : f32 = a[0][1]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Matrix_Idx_Negative_Row) { |
| auto* src = R"( |
| var<private> a : mat3x2<f32>; |
| |
| fn f() { |
| var b : f32 = a[2][-1]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : mat3x2<f32>; |
| |
| fn f() { |
| var b : f32 = a[2][0]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Matrix_Idx_OutOfBounds_Column) { |
| auto* src = R"( |
| var<private> a : mat3x2<f32>; |
| |
| fn f() { |
| var b : f32 = a[5][1]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : mat3x2<f32>; |
| |
| fn f() { |
| var b : f32 = a[2][1]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, Matrix_Idx_OutOfBounds_Row) { |
| auto* src = R"( |
| var<private> a : mat3x2<f32>; |
| |
| fn f() { |
| var b : f32 = a[2][5]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| var<private> a : mat3x2<f32>; |
| |
| fn f() { |
| var b : f32 = a[2][1]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| // TODO(dsinclair): Implement when constant_id exists |
| TEST_F(BoundArrayAccessorsTest, DISABLED_Vector_Constant_Id_Clamps) { |
| // [[override(1300)]] let idx : i32; |
| // var a : vec3<f32> |
| // var b : f32 = a[idx] |
| // |
| // ->var b : f32 = a[min(u32(idx), 2)] |
| } |
| |
| // TODO(dsinclair): Implement when constant_id exists |
| TEST_F(BoundArrayAccessorsTest, DISABLED_Array_Constant_Id_Clamps) { |
| // [[override(1300)]] let idx : i32; |
| // var a : array<f32, 4> |
| // var b : f32 = a[idx] |
| // |
| // -> var b : f32 = a[min(u32(idx), 3)] |
| } |
| |
| // TODO(dsinclair): Implement when constant_id exists |
| TEST_F(BoundArrayAccessorsTest, DISABLED_Matrix_Column_Constant_Id_Clamps) { |
| // [[override(1300)]] let idx : i32; |
| // var a : mat3x2<f32> |
| // var b : f32 = a[idx][1] |
| // |
| // -> var b : f32 = a[min(u32(idx), 2)][1] |
| } |
| |
| // TODO(dsinclair): Implement when constant_id exists |
| TEST_F(BoundArrayAccessorsTest, DISABLED_Matrix_Row_Constant_Id_Clamps) { |
| // [[override(1300)]] let idx : i32; |
| // var a : mat3x2<f32> |
| // var b : f32 = a[1][idx] |
| // |
| // -> var b : f32 = a[1][min(u32(idx), 0, 1)] |
| } |
| |
| TEST_F(BoundArrayAccessorsTest, RuntimeArray_Clamps) { |
| auto* src = R"( |
| [[block]] |
| struct S { |
| a : f32; |
| b : array<f32>; |
| }; |
| [[group(0), binding(0)]] var<storage, read> s : S; |
| |
| fn f() { |
| var d : f32 = s.b[25]; |
| } |
| )"; |
| |
| auto* expect = R"( |
| [[block]] |
| struct S { |
| a : f32; |
| b : array<f32>; |
| }; |
| |
| [[group(0), binding(0)]] var<storage, read> s : S; |
| |
| fn f() { |
| var d : f32 = s.b[min(u32(25), (arrayLength(s.b) - 1u))]; |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| // TODO(dsinclair): Clamp atomics when available. |
| TEST_F(BoundArrayAccessorsTest, DISABLED_Atomics_Clamp) { |
| FAIL(); |
| } |
| |
| // TODO(dsinclair): Clamp texture coord values. Depends on: |
| // https://github.com/gpuweb/gpuweb/issues/1107 |
| TEST_F(BoundArrayAccessorsTest, DISABLED_TextureCoord_Clamp) { |
| FAIL(); |
| } |
| |
| // TODO(dsinclair): Test for scoped variables when Lexical Scopes implemented |
| TEST_F(BoundArrayAccessorsTest, DISABLED_Scoped_Variable) { |
| // var a : array<f32, 3>; |
| // var i : u32; |
| // { |
| // var a : array<f32, 5>; |
| // var b : f32 = a[i]; |
| // } |
| // var c : f32 = a[i]; |
| // |
| // -> var b : f32 = a[min(u32(i), 4)]; |
| // var c : f32 = a[min(u32(i), 2)]; |
| FAIL(); |
| } |
| |
| // Check that existing use of min() and arrayLength() do not get renamed. |
| TEST_F(BoundArrayAccessorsTest, DontRenameSymbols) { |
| auto* src = R"( |
| [[block]] |
| struct S { |
| a : f32; |
| b : array<f32>; |
| }; |
| |
| [[group(0), binding(0)]] var<storage, read> s : S; |
| |
| let c : u32 = 1u; |
| |
| fn f() { |
| let b : f32 = s.b[c]; |
| let x : i32 = min(1, 2); |
| let y : u32 = arrayLength(s.b); |
| } |
| )"; |
| |
| auto* expect = R"( |
| [[block]] |
| struct S { |
| a : f32; |
| b : array<f32>; |
| }; |
| |
| [[group(0), binding(0)]] var<storage, read> s : S; |
| |
| let c : u32 = 1u; |
| |
| fn f() { |
| let b : f32 = s.b[min(u32(c), (arrayLength(s.b) - 1u))]; |
| let x : i32 = min(1, 2); |
| let y : u32 = arrayLength(s.b); |
| } |
| )"; |
| |
| auto got = Run<BoundArrayAccessors>(src); |
| |
| EXPECT_EQ(expect, str(got)); |
| } |
| |
| } // namespace |
| } // namespace transform |
| } // namespace tint |