// Copyright 2021 The Tint Authors. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include "src/tint/resolver/resolver.h" #include "src/tint/resolver/resolver_test_helper.h" #include "src/tint/sem/reference.h" #include "gmock/gmock.h" using namespace tint::number_suffixes; // NOLINT namespace tint::resolver { namespace { struct ResolverVariableTest : public resolver::TestHelper, public testing::Test {}; //////////////////////////////////////////////////////////////////////////////////////////////////// // Function-scope 'var' //////////////////////////////////////////////////////////////////////////////////////////////////// TEST_F(ResolverVariableTest, LocalVar_NoConstructor) { // struct S { i : i32; } // alias A = S; // fn F(){ // var i : i32; // var u : u32; // var f : f32; // var b : bool; // var s : S; // var a : A; // } auto* S = Structure("S", {Member("i", ty.i32())}); auto* A = Alias("A", ty.Of(S)); auto* i = Var("i", ty.i32(), ast::StorageClass::kNone); auto* u = Var("u", ty.u32(), ast::StorageClass::kNone); auto* f = Var("f", ty.f32(), ast::StorageClass::kNone); auto* b = Var("b", ty.bool_(), ast::StorageClass::kNone); auto* s = Var("s", ty.Of(S), ast::StorageClass::kNone); auto* a = Var("a", ty.Of(A), ast::StorageClass::kNone); Func("F", {}, ty.void_(), { Decl(i), Decl(u), Decl(f), Decl(b), Decl(s), Decl(a), }); ASSERT_TRUE(r()->Resolve()) << r()->error(); // `var` declarations are always of reference type ASSERT_TRUE(TypeOf(i)->Is()); ASSERT_TRUE(TypeOf(u)->Is()); ASSERT_TRUE(TypeOf(f)->Is()); ASSERT_TRUE(TypeOf(b)->Is()); ASSERT_TRUE(TypeOf(s)->Is()); ASSERT_TRUE(TypeOf(a)->Is()); EXPECT_TRUE(TypeOf(i)->As()->StoreType()->Is()); EXPECT_TRUE(TypeOf(u)->As()->StoreType()->Is()); EXPECT_TRUE(TypeOf(f)->As()->StoreType()->Is()); EXPECT_TRUE(TypeOf(b)->As()->StoreType()->Is()); EXPECT_TRUE(TypeOf(s)->As()->StoreType()->Is()); EXPECT_TRUE(TypeOf(a)->As()->StoreType()->Is()); EXPECT_EQ(Sem().Get(i)->Constructor(), nullptr); EXPECT_EQ(Sem().Get(u)->Constructor(), nullptr); EXPECT_EQ(Sem().Get(f)->Constructor(), nullptr); EXPECT_EQ(Sem().Get(b)->Constructor(), nullptr); EXPECT_EQ(Sem().Get(s)->Constructor(), nullptr); EXPECT_EQ(Sem().Get(a)->Constructor(), nullptr); } TEST_F(ResolverVariableTest, LocalVar_WithConstructor) { // struct S { i : i32; } // alias A = S; // fn F(){ // var i : i32 = 1i; // var u : u32 = 1u; // var f : f32 = 1.f; // var b : bool = true; // var s : S = S(1); // var a : A = A(1); // } auto* S = Structure("S", {Member("i", ty.i32())}); auto* A = Alias("A", ty.Of(S)); auto* i_c = Expr(1_i); auto* u_c = Expr(1_u); auto* f_c = Expr(1_f); auto* b_c = Expr(true); auto* s_c = Construct(ty.Of(S), Expr(1_i)); auto* a_c = Construct(ty.Of(A), Expr(1_i)); auto* i = Var("i", ty.i32(), ast::StorageClass::kNone, i_c); auto* u = Var("u", ty.u32(), ast::StorageClass::kNone, u_c); auto* f = Var("f", ty.f32(), ast::StorageClass::kNone, f_c); auto* b = Var("b", ty.bool_(), ast::StorageClass::kNone, b_c); auto* s = Var("s", ty.Of(S), ast::StorageClass::kNone, s_c); auto* a = Var("a", ty.Of(A), ast::StorageClass::kNone, a_c); Func("F", {}, ty.void_(), { Decl(i), Decl(u), Decl(f), Decl(b), Decl(s), Decl(a), }); ASSERT_TRUE(r()->Resolve()) << r()->error(); // `var` declarations are always of reference type ASSERT_TRUE(TypeOf(i)->Is()); ASSERT_TRUE(TypeOf(u)->Is()); ASSERT_TRUE(TypeOf(f)->Is()); ASSERT_TRUE(TypeOf(b)->Is()); ASSERT_TRUE(TypeOf(s)->Is()); ASSERT_TRUE(TypeOf(a)->Is()); EXPECT_EQ(TypeOf(i)->As()->Access(), ast::Access::kReadWrite); EXPECT_EQ(TypeOf(u)->As()->Access(), ast::Access::kReadWrite); EXPECT_EQ(TypeOf(f)->As()->Access(), ast::Access::kReadWrite); EXPECT_EQ(TypeOf(b)->As()->Access(), ast::Access::kReadWrite); EXPECT_EQ(TypeOf(s)->As()->Access(), ast::Access::kReadWrite); EXPECT_EQ(TypeOf(a)->As()->Access(), ast::Access::kReadWrite); EXPECT_TRUE(TypeOf(i)->As()->StoreType()->Is()); EXPECT_TRUE(TypeOf(u)->As()->StoreType()->Is()); EXPECT_TRUE(TypeOf(f)->As()->StoreType()->Is()); EXPECT_TRUE(TypeOf(b)->As()->StoreType()->Is()); EXPECT_TRUE(TypeOf(s)->As()->StoreType()->Is()); EXPECT_TRUE(TypeOf(a)->As()->StoreType()->Is()); EXPECT_EQ(Sem().Get(i)->Constructor()->Declaration(), i_c); EXPECT_EQ(Sem().Get(u)->Constructor()->Declaration(), u_c); EXPECT_EQ(Sem().Get(f)->Constructor()->Declaration(), f_c); EXPECT_EQ(Sem().Get(b)->Constructor()->Declaration(), b_c); EXPECT_EQ(Sem().Get(s)->Constructor()->Declaration(), s_c); EXPECT_EQ(Sem().Get(a)->Constructor()->Declaration(), a_c); } TEST_F(ResolverVariableTest, LocalVar_ShadowsAlias) { // type a = i32; // // fn F() { // var a = false; // } auto* t = Alias("a", ty.i32()); auto* v = Var("a", nullptr, Expr(false)); Func("F", {}, ty.void_(), {Decl(v)}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* type_t = Sem().Get(t); auto* local = Sem().Get(v); ASSERT_NE(local, nullptr); EXPECT_EQ(local->Shadows(), type_t); } TEST_F(ResolverVariableTest, LocalVar_ShadowsStruct) { // struct a { // m : i32; // }; // // fn F() { // var a = true; // } auto* t = Structure("a", {Member("m", ty.i32())}); auto* v = Var("a", nullptr, Expr(false)); Func("F", {}, ty.void_(), {Decl(v)}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* type_t = Sem().Get(t); auto* local = Sem().Get(v); ASSERT_NE(local, nullptr); EXPECT_EQ(local->Shadows(), type_t); } TEST_F(ResolverVariableTest, LocalVar_ShadowsFunction) { // fn a() { // var a = true; // } auto* v = Var("a", nullptr, Expr(false)); auto* f = Func("a", {}, ty.void_(), {Decl(v)}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* func = Sem().Get(f); ASSERT_NE(func, nullptr); auto* local = Sem().Get(v); ASSERT_NE(local, nullptr); EXPECT_EQ(local->Shadows(), func); } TEST_F(ResolverVariableTest, LocalVar_ShadowsGlobalVar) { // var a : i32; // // fn F() { // var a = a; // } auto* g = Global("a", ty.i32(), ast::StorageClass::kPrivate); auto* v = Var("a", nullptr, Expr("a")); Func("F", {}, ty.void_(), {Decl(v)}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* global = Sem().Get(g); auto* local = Sem().Get(v); ASSERT_NE(local, nullptr); EXPECT_EQ(local->Shadows(), global); auto* user_v = Sem().Get(local->Declaration()->constructor); ASSERT_NE(user_v, nullptr); EXPECT_EQ(user_v->Variable(), global); } TEST_F(ResolverVariableTest, LocalVar_ShadowsGlobalLet) { // let a : i32 = 1; // // fn X() { // var a = (a == 123); // } auto* g = GlobalConst("a", ty.i32(), Expr(1_i)); auto* v = Var("a", nullptr, Expr("a")); Func("F", {}, ty.void_(), {Decl(v)}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* global = Sem().Get(g); auto* local = Sem().Get(v); ASSERT_NE(local, nullptr); EXPECT_EQ(local->Shadows(), global); auto* user_v = Sem().Get(local->Declaration()->constructor); ASSERT_NE(user_v, nullptr); EXPECT_EQ(user_v->Variable(), global); } TEST_F(ResolverVariableTest, LocalVar_ShadowsLocalVar) { // fn F() { // var a : i32; // x // { // var a = a; // y // } // } auto* x = Var("a", ty.i32(), Expr(1_i)); auto* y = Var("a", nullptr, Expr("a")); Func("F", {}, ty.void_(), {Decl(x), Block(Decl(y))}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* local_x = Sem().Get(x); auto* local_y = Sem().Get(y); ASSERT_NE(local_x, nullptr); ASSERT_NE(local_y, nullptr); EXPECT_EQ(local_y->Shadows(), local_x); auto* user_y = Sem().Get(local_y->Declaration()->constructor); ASSERT_NE(user_y, nullptr); EXPECT_EQ(user_y->Variable(), local_x); } TEST_F(ResolverVariableTest, LocalVar_ShadowsLocalLet) { // fn F() { // let a = 1; // { // var a = (a == 123); // } // } auto* l = Let("a", ty.i32(), Expr(1_i)); auto* v = Var("a", nullptr, Expr("a")); Func("X", {}, ty.void_(), {Decl(l), Block(Decl(v))}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* local_l = Sem().Get(l); auto* local_v = Sem().Get(v); ASSERT_NE(local_l, nullptr); ASSERT_NE(local_v, nullptr); EXPECT_EQ(local_v->Shadows(), local_l); auto* user_v = Sem().Get(local_v->Declaration()->constructor); ASSERT_NE(user_v, nullptr); EXPECT_EQ(user_v->Variable(), local_l); } TEST_F(ResolverVariableTest, LocalVar_ShadowsParam) { // fn F(a : i32) { // { // var a = a; // } // } auto* p = Param("a", ty.i32()); auto* v = Var("a", nullptr, Expr("a")); Func("X", {p}, ty.void_(), {Block(Decl(v))}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* param = Sem().Get(p); auto* local = Sem().Get(v); ASSERT_NE(param, nullptr); ASSERT_NE(local, nullptr); EXPECT_EQ(local->Shadows(), param); auto* user_v = Sem().Get(local->Declaration()->constructor); ASSERT_NE(user_v, nullptr); EXPECT_EQ(user_v->Variable(), param); } //////////////////////////////////////////////////////////////////////////////////////////////////// // Function-scope 'let' //////////////////////////////////////////////////////////////////////////////////////////////////// TEST_F(ResolverVariableTest, LocalLet) { // struct S { i : i32; } // fn F(){ // var v : i32; // let i : i32 = 1i; // let u : u32 = 1u; // let f : f32 = 1.; // let b : bool = true; // let s : S = S(1); // let a : A = A(1); // let p : pointer = &v; // } auto* S = Structure("S", {Member("i", ty.i32())}); auto* A = Alias("A", ty.Of(S)); auto* v = Var("v", ty.i32(), ast::StorageClass::kNone); auto* i_c = Expr(1_i); auto* u_c = Expr(1_u); auto* f_c = Expr(1_f); auto* b_c = Expr(true); auto* s_c = Construct(ty.Of(S), Expr(1_i)); auto* a_c = Construct(ty.Of(A), Expr(1_i)); auto* p_c = AddressOf(v); auto* i = Let("i", ty.i32(), i_c); auto* u = Let("u", ty.u32(), u_c); auto* f = Let("f", ty.f32(), f_c); auto* b = Let("b", ty.bool_(), b_c); auto* s = Let("s", ty.Of(S), s_c); auto* a = Let("a", ty.Of(A), a_c); auto* p = Let("p", ty.pointer(ast::StorageClass::kFunction), p_c); Func("F", {}, ty.void_(), { Decl(v), Decl(i), Decl(u), Decl(f), Decl(b), Decl(s), Decl(a), Decl(p), }); ASSERT_TRUE(r()->Resolve()) << r()->error(); // `let` declarations are always of the storage type ASSERT_TRUE(TypeOf(i)->Is()); ASSERT_TRUE(TypeOf(u)->Is()); ASSERT_TRUE(TypeOf(f)->Is()); ASSERT_TRUE(TypeOf(b)->Is()); ASSERT_TRUE(TypeOf(s)->Is()); ASSERT_TRUE(TypeOf(a)->Is()); ASSERT_TRUE(TypeOf(p)->Is()); ASSERT_TRUE(TypeOf(p)->As()->StoreType()->Is()); EXPECT_EQ(Sem().Get(i)->Constructor()->Declaration(), i_c); EXPECT_EQ(Sem().Get(u)->Constructor()->Declaration(), u_c); EXPECT_EQ(Sem().Get(f)->Constructor()->Declaration(), f_c); EXPECT_EQ(Sem().Get(b)->Constructor()->Declaration(), b_c); EXPECT_EQ(Sem().Get(s)->Constructor()->Declaration(), s_c); EXPECT_EQ(Sem().Get(a)->Constructor()->Declaration(), a_c); EXPECT_EQ(Sem().Get(p)->Constructor()->Declaration(), p_c); } TEST_F(ResolverVariableTest, LocalLet_InheritsAccessFromOriginatingVariable) { // struct Inner { // arr: array; // } // struct S { // inner: Inner; // } // @group(0) @binding(0) var s : S; // fn f() { // let p = &s.inner.arr[4]; // } auto* inner = Structure("Inner", {Member("arr", ty.array())}); auto* buf = Structure("S", {Member("inner", ty.Of(inner))}); auto* storage = Global("s", ty.Of(buf), ast::StorageClass::kStorage, ast::Access::kReadWrite, ast::AttributeList{ create(0), create(0), }); auto* expr = IndexAccessor(MemberAccessor(MemberAccessor(storage, "inner"), "arr"), 4_i); auto* ptr = Let("p", nullptr, AddressOf(expr)); WrapInFunction(ptr); ASSERT_TRUE(r()->Resolve()) << r()->error(); ASSERT_TRUE(TypeOf(expr)->Is()); ASSERT_TRUE(TypeOf(ptr)->Is()); EXPECT_EQ(TypeOf(expr)->As()->Access(), ast::Access::kReadWrite); EXPECT_EQ(TypeOf(ptr)->As()->Access(), ast::Access::kReadWrite); } TEST_F(ResolverVariableTest, LocalLet_ShadowsAlias) { // type a = i32; // // fn F() { // let a = true; // } auto* t = Alias("a", ty.i32()); auto* l = Let("a", nullptr, Expr(false)); Func("F", {}, ty.void_(), {Decl(l)}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* type_t = Sem().Get(t); auto* local = Sem().Get(l); ASSERT_NE(local, nullptr); EXPECT_EQ(local->Shadows(), type_t); } TEST_F(ResolverVariableTest, LocalLet_ShadowsStruct) { // struct a { // m : i32; // }; // // fn F() { // let a = false; // } auto* t = Structure("a", {Member("m", ty.i32())}); auto* l = Let("a", nullptr, Expr(false)); Func("F", {}, ty.void_(), {Decl(l)}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* type_t = Sem().Get(t); auto* local = Sem().Get(l); ASSERT_NE(local, nullptr); EXPECT_EQ(local->Shadows(), type_t); } TEST_F(ResolverVariableTest, LocalLet_ShadowsFunction) { // fn a() { // let a = false; // } auto* l = Let("a", nullptr, Expr(false)); auto* fb = Func("a", {}, ty.void_(), {Decl(l)}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* func = Sem().Get(fb); ASSERT_NE(func, nullptr); auto* local = Sem().Get(l); ASSERT_NE(local, nullptr); EXPECT_EQ(local->Shadows(), func); } TEST_F(ResolverVariableTest, LocalLet_ShadowsGlobalVar) { // var a : i32; // // fn F() { // let a = a; // } auto* g = Global("a", ty.i32(), ast::StorageClass::kPrivate); auto* l = Let("a", nullptr, Expr("a")); Func("F", {}, ty.void_(), {Decl(l)}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* global = Sem().Get(g); auto* local = Sem().Get(l); ASSERT_NE(local, nullptr); EXPECT_EQ(local->Shadows(), global); auto* user = Sem().Get(local->Declaration()->constructor); ASSERT_NE(user, nullptr); EXPECT_EQ(user->Variable(), global); } TEST_F(ResolverVariableTest, LocalLet_ShadowsGlobalLet) { // let a : i32 = 1; // // fn F() { // let a = (a == 321); // } auto* g = GlobalConst("a", ty.i32(), Expr(1_i)); auto* l = Let("a", nullptr, Expr("a")); Func("F", {}, ty.void_(), {Decl(l)}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* global = Sem().Get(g); auto* local = Sem().Get(l); ASSERT_NE(local, nullptr); EXPECT_EQ(local->Shadows(), global); auto* user = Sem().Get(local->Declaration()->constructor); ASSERT_NE(user, nullptr); EXPECT_EQ(user->Variable(), global); } TEST_F(ResolverVariableTest, LocalLet_ShadowsLocalVar) { // fn F() { // var a : i32; // { // let a = a; // } // } auto* v = Var("a", ty.i32(), Expr(1_i)); auto* l = Let("a", nullptr, Expr("a")); Func("F", {}, ty.void_(), {Decl(v), Block(Decl(l))}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* local_v = Sem().Get(v); auto* local_l = Sem().Get(l); ASSERT_NE(local_v, nullptr); ASSERT_NE(local_l, nullptr); EXPECT_EQ(local_l->Shadows(), local_v); auto* user = Sem().Get(local_l->Declaration()->constructor); ASSERT_NE(user, nullptr); EXPECT_EQ(user->Variable(), local_v); } TEST_F(ResolverVariableTest, LocalLet_ShadowsLocalLet) { // fn X() { // let a = 1; // x // { // let a = (a == 321); // y // } // } auto* x = Let("a", ty.i32(), Expr(1_i)); auto* y = Let("a", nullptr, Expr("a")); Func("X", {}, ty.void_(), {Decl(x), Block(Decl(y))}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* local_x = Sem().Get(x); auto* local_y = Sem().Get(y); ASSERT_NE(local_x, nullptr); ASSERT_NE(local_y, nullptr); EXPECT_EQ(local_y->Shadows(), local_x); auto* user = Sem().Get(local_y->Declaration()->constructor); ASSERT_NE(user, nullptr); EXPECT_EQ(user->Variable(), local_x); } TEST_F(ResolverVariableTest, LocalLet_ShadowsParam) { // fn F(a : i32) { // { // let a = a; // } // } auto* p = Param("a", ty.i32()); auto* l = Let("a", nullptr, Expr("a")); Func("X", {p}, ty.void_(), {Block(Decl(l))}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* param = Sem().Get(p); auto* local = Sem().Get(l); ASSERT_NE(param, nullptr); ASSERT_NE(local, nullptr); EXPECT_EQ(local->Shadows(), param); auto* user = Sem().Get(local->Declaration()->constructor); ASSERT_NE(user, nullptr); EXPECT_EQ(user->Variable(), param); } //////////////////////////////////////////////////////////////////////////////////////////////////// // Module-scope 'var' //////////////////////////////////////////////////////////////////////////////////////////////////// TEST_F(ResolverVariableTest, GlobalVar_StorageClass) { // https://gpuweb.github.io/gpuweb/wgsl/#storage-class auto* buf = Structure("S", {Member("m", ty.i32())}); auto* private_ = Global("p", ty.i32(), ast::StorageClass::kPrivate); auto* workgroup = Global("w", ty.i32(), ast::StorageClass::kWorkgroup); auto* uniform = Global("ub", ty.Of(buf), ast::StorageClass::kUniform, ast::AttributeList{ create(0), create(0), }); auto* storage = Global("sb", ty.Of(buf), ast::StorageClass::kStorage, ast::AttributeList{ create(1), create(0), }); auto* handle = Global("h", ty.depth_texture(ast::TextureDimension::k2d), ast::AttributeList{ create(2), create(0), }); ASSERT_TRUE(r()->Resolve()) << r()->error(); ASSERT_TRUE(TypeOf(private_)->Is()); ASSERT_TRUE(TypeOf(workgroup)->Is()); ASSERT_TRUE(TypeOf(uniform)->Is()); ASSERT_TRUE(TypeOf(storage)->Is()); ASSERT_TRUE(TypeOf(handle)->Is()); EXPECT_EQ(TypeOf(private_)->As()->Access(), ast::Access::kReadWrite); EXPECT_EQ(TypeOf(workgroup)->As()->Access(), ast::Access::kReadWrite); EXPECT_EQ(TypeOf(uniform)->As()->Access(), ast::Access::kRead); EXPECT_EQ(TypeOf(storage)->As()->Access(), ast::Access::kRead); EXPECT_EQ(TypeOf(handle)->As()->Access(), ast::Access::kRead); } TEST_F(ResolverVariableTest, GlobalVar_ExplicitStorageClass) { // https://gpuweb.github.io/gpuweb/wgsl/#storage-class auto* buf = Structure("S", {Member("m", ty.i32())}); auto* storage = Global("sb", ty.Of(buf), ast::StorageClass::kStorage, ast::Access::kReadWrite, ast::AttributeList{ create(1), create(0), }); ASSERT_TRUE(r()->Resolve()) << r()->error(); ASSERT_TRUE(TypeOf(storage)->Is()); EXPECT_EQ(TypeOf(storage)->As()->Access(), ast::Access::kReadWrite); } //////////////////////////////////////////////////////////////////////////////////////////////////// // Function parameter //////////////////////////////////////////////////////////////////////////////////////////////////// TEST_F(ResolverVariableTest, Param_ShadowsFunction) { // fn a(a : bool) { // } auto* p = Param("a", ty.bool_()); auto* f = Func("a", {p}, ty.void_(), {}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* func = Sem().Get(f); auto* param = Sem().Get(p); ASSERT_NE(func, nullptr); ASSERT_NE(param, nullptr); EXPECT_EQ(param->Shadows(), func); } TEST_F(ResolverVariableTest, Param_ShadowsGlobalVar) { // var a : i32; // // fn F(a : bool) { // } auto* g = Global("a", ty.i32(), ast::StorageClass::kPrivate); auto* p = Param("a", ty.bool_()); Func("F", {p}, ty.void_(), {}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* global = Sem().Get(g); auto* param = Sem().Get(p); ASSERT_NE(global, nullptr); ASSERT_NE(param, nullptr); EXPECT_EQ(param->Shadows(), global); } TEST_F(ResolverVariableTest, Param_ShadowsGlobalLet) { // let a : i32 = 1; // // fn F(a : bool) { // } auto* g = GlobalConst("a", ty.i32(), Expr(1_i)); auto* p = Param("a", ty.bool_()); Func("F", {p}, ty.void_(), {}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* global = Sem().Get(g); auto* param = Sem().Get(p); ASSERT_NE(global, nullptr); ASSERT_NE(param, nullptr); EXPECT_EQ(param->Shadows(), global); } TEST_F(ResolverVariableTest, Param_ShadowsAlias) { // type a = i32; // // fn F(a : a) { // } auto* a = Alias("a", ty.i32()); auto* p = Param("a", ty.type_name("a")); Func("F", {p}, ty.void_(), {}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* alias = Sem().Get(a); auto* param = Sem().Get(p); ASSERT_NE(alias, nullptr); ASSERT_NE(param, nullptr); EXPECT_EQ(param->Shadows(), alias); EXPECT_EQ(param->Type(), alias); } } // namespace } // namespace tint::resolver