// 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/resolver/resolver.h" #include "gmock/gmock.h" #include "src/resolver/resolver_test_helper.h" #include "src/sem/block_statement.h" #include "src/sem/for_loop_statement.h" #include "src/sem/if_statement.h" #include "src/sem/loop_statement.h" #include "src/sem/switch_statement.h" namespace tint { namespace resolver { namespace { using ResolverCompoundStatementTest = ResolverTest; TEST_F(ResolverCompoundStatementTest, FunctionBlock) { // fn F() { // var x : 32; // } auto* stmt = Decl(Var("x", ty.i32())); auto* f = Func("F", {}, ty.void_(), {stmt}); ASSERT_TRUE(r()->Resolve()) << r()->error(); auto* s = Sem().Get(stmt); ASSERT_NE(s, nullptr); ASSERT_NE(s->Block(), nullptr); ASSERT_TRUE(s->Block()->Is()); EXPECT_EQ(s->Block(), s->FindFirstParent()); EXPECT_EQ(s->Block(), s->FindFirstParent()); EXPECT_EQ(s->Block()->As()->Function(), f); EXPECT_EQ(s->Block()->Parent(), nullptr); } TEST_F(ResolverCompoundStatementTest, Block) { // fn F() { // { // var x : 32; // } // } auto* stmt = Decl(Var("x", ty.i32())); auto* block = Block(stmt); auto* f = Func("F", {}, ty.void_(), {block}); ASSERT_TRUE(r()->Resolve()) << r()->error(); { auto* s = Sem().Get(block); ASSERT_NE(s, nullptr); EXPECT_TRUE(s->Is()); EXPECT_EQ(s, s->Block()); EXPECT_EQ(s->Parent(), s->FindFirstParent()); } { auto* s = Sem().Get(stmt); ASSERT_NE(s, nullptr); ASSERT_NE(s->Block(), nullptr); EXPECT_EQ(s->Block(), s->FindFirstParent()); EXPECT_EQ(s->Block()->Parent(), s->FindFirstParent()); ASSERT_TRUE(s->Block()->Parent()->Is()); EXPECT_EQ( s->Block()->Parent()->As()->Function(), f); EXPECT_EQ(s->Block()->Parent()->Parent(), nullptr); } } TEST_F(ResolverCompoundStatementTest, Loop) { // fn F() { // loop { // stmt_a; // continuing { // stmt_b; // } // } // } auto* stmt_a = Ignore(1); auto* stmt_b = Ignore(1); auto* loop = Loop(Block(stmt_a), Block(stmt_b)); auto* f = Func("F", {}, ty.void_(), {loop}); ASSERT_TRUE(r()->Resolve()) << r()->error(); { auto* s = Sem().Get(loop); ASSERT_NE(s, nullptr); EXPECT_TRUE(s->Is()); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent(), s->Block()); } { auto* s = Sem().Get(stmt_a); ASSERT_NE(s, nullptr); ASSERT_NE(s->Block(), nullptr); EXPECT_EQ(s->Parent(), s->Block()); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent()->Parent(), s->FindFirstParent()); EXPECT_TRUE(Is(s->Parent()->Parent())); EXPECT_EQ(s->Parent()->Parent()->Parent(), s->FindFirstParent()); EXPECT_TRUE( Is(s->Parent()->Parent()->Parent())); EXPECT_EQ(s->FindFirstParent()->Function(), f); EXPECT_EQ(s->Parent()->Parent()->Parent()->Parent(), nullptr); } { auto* s = Sem().Get(stmt_b); ASSERT_NE(s, nullptr); ASSERT_NE(s->Block(), nullptr); EXPECT_EQ(s->Parent(), s->Block()); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_TRUE(Is(s->Parent())); EXPECT_EQ(s->Parent()->Parent(), s->FindFirstParent()); EXPECT_TRUE(Is(s->Parent()->Parent())); EXPECT_EQ(s->Parent()->Parent()->Parent(), s->FindFirstParent()); EXPECT_TRUE(Is(s->Parent()->Parent()->Parent())); EXPECT_EQ(s->Parent()->Parent()->Parent()->Parent(), s->FindFirstParent()); EXPECT_TRUE(Is( s->Parent()->Parent()->Parent()->Parent())); EXPECT_EQ(s->FindFirstParent()->Function(), f); EXPECT_EQ(s->Parent()->Parent()->Parent()->Parent()->Parent(), nullptr); } } TEST_F(ResolverCompoundStatementTest, ForLoop) { // fn F() { // for (var i : u32; true; i = i + 1u) { // return; // } // } auto* init = Decl(Var("i", ty.u32())); auto* cond = Expr(true); auto* cont = Assign("i", Add("i", 1u)); auto* stmt = Return(); auto* body = Block(stmt); auto* for_ = For(init, cond, cont, body); auto* f = Func("F", {}, ty.void_(), {for_}); ASSERT_TRUE(r()->Resolve()) << r()->error(); { auto* s = Sem().Get(for_); ASSERT_NE(s, nullptr); EXPECT_TRUE(s->Is()); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent(), s->Block()); } { auto* s = Sem().Get(init); ASSERT_NE(s, nullptr); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_TRUE(Is(s->Parent())); EXPECT_EQ(s->Block(), s->FindFirstParent()); EXPECT_TRUE(Is(s->Parent()->Parent())); } { // Condition expression's statement is the for-loop itself auto* e = Sem().Get(cond); ASSERT_NE(e, nullptr); auto* s = e->Stmt(); ASSERT_NE(s, nullptr); ASSERT_TRUE(Is(s)); ASSERT_NE(s->Parent(), nullptr); EXPECT_EQ(s->Parent(), s->Block()); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_TRUE(Is(s->Block())); } { auto* s = Sem().Get(cont); ASSERT_NE(s, nullptr); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_TRUE(Is(s->Parent())); EXPECT_EQ(s->Block(), s->FindFirstParent()); EXPECT_TRUE(Is(s->Parent()->Parent())); } { auto* s = Sem().Get(stmt); ASSERT_NE(s, nullptr); ASSERT_NE(s->Block(), nullptr); EXPECT_EQ(s->Parent(), s->Block()); EXPECT_EQ(s->Block(), s->FindFirstParent()); EXPECT_TRUE(Is(s->Parent()->Parent())); EXPECT_EQ(s->Block()->Parent(), s->FindFirstParent()); ASSERT_TRUE( Is(s->Block()->Parent()->Parent())); EXPECT_EQ(s->Block()->Parent()->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Block() ->Parent() ->Parent() ->As() ->Function(), f); EXPECT_EQ(s->Block()->Parent()->Parent()->Parent(), nullptr); } } TEST_F(ResolverCompoundStatementTest, If) { // fn F() { // if (cond_a) { // stat_a; // } elseif (cond_b) { // stat_b; // } else { // stat_c; // } // } auto* cond_a = Expr(true); auto* stmt_a = Ignore(1); auto* cond_b = Expr(true); auto* stmt_b = Ignore(1); auto* stmt_c = Ignore(1); auto* if_stmt = If(cond_a, Block(stmt_a), Else(cond_b, Block(stmt_b)), Else(nullptr, Block(stmt_c))); WrapInFunction(if_stmt); ASSERT_TRUE(r()->Resolve()) << r()->error(); { auto* s = Sem().Get(if_stmt); ASSERT_NE(s, nullptr); EXPECT_TRUE(s->Is()); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent(), s->Block()); } { auto* e = Sem().Get(cond_a); ASSERT_NE(e, nullptr); auto* s = e->Stmt(); ASSERT_NE(s, nullptr); EXPECT_TRUE(s->Is()); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent(), s->Block()); } { auto* s = Sem().Get(stmt_a); ASSERT_NE(s, nullptr); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent(), s->Block()); EXPECT_EQ(s->Parent()->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent()->Parent()->Parent(), s->FindFirstParent()); } { auto* e = Sem().Get(cond_b); ASSERT_NE(e, nullptr); auto* s = e->Stmt(); ASSERT_NE(s, nullptr); EXPECT_TRUE(s->Is()); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent()->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent()->Parent(), s->Block()); } { auto* s = Sem().Get(stmt_b); ASSERT_NE(s, nullptr); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent(), s->Block()); EXPECT_EQ(s->Parent()->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent()->Parent()->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent()->Parent()->Parent()->Parent(), s->FindFirstParent()); } { auto* s = Sem().Get(stmt_c); ASSERT_NE(s, nullptr); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent(), s->Block()); EXPECT_EQ(s->Parent()->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent()->Parent()->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent()->Parent()->Parent()->Parent(), s->FindFirstParent()); } } TEST_F(ResolverCompoundStatementTest, Switch) { // fn F() { // switch (expr) { // case 1: { // stmt_a; // } // case 2: { // stmt_b; // } // default: { // stmt_c; // } // } // } auto* expr = Expr(5); auto* stmt_a = Ignore(1); auto* stmt_b = Ignore(1); auto* stmt_c = Ignore(1); auto* swi = Switch(expr, Case(Literal(1), Block(stmt_a)), Case(Literal(2), Block(stmt_b)), DefaultCase(Block(stmt_c))); WrapInFunction(swi); ASSERT_TRUE(r()->Resolve()) << r()->error(); { auto* s = Sem().Get(swi); ASSERT_NE(s, nullptr); EXPECT_TRUE(s->Is()); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent(), s->Block()); } { auto* e = Sem().Get(expr); ASSERT_NE(e, nullptr); auto* s = e->Stmt(); ASSERT_NE(s, nullptr); EXPECT_TRUE(s->Is()); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent(), s->Block()); } { auto* s = Sem().Get(stmt_a); ASSERT_NE(s, nullptr); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent(), s->Block()); EXPECT_EQ(s->Parent()->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent()->Parent()->Parent(), s->FindFirstParent()); } { auto* s = Sem().Get(stmt_b); ASSERT_NE(s, nullptr); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent(), s->Block()); EXPECT_EQ(s->Parent()->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent()->Parent()->Parent(), s->FindFirstParent()); } { auto* s = Sem().Get(stmt_c); ASSERT_NE(s, nullptr); EXPECT_EQ(s->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent(), s->Block()); EXPECT_EQ(s->Parent()->Parent(), s->FindFirstParent()); EXPECT_EQ(s->Parent()->Parent()->Parent(), s->FindFirstParent()); } } } // namespace } // namespace resolver } // namespace tint