Add tests for if statement AST
This CL adds tests for the if statement AST node.
Bug: tint:11
Change-Id: Ice3f281e1dfd72d76f43767f7b3040af862f4a58
Reviewed-on: https://dawn-review.googlesource.com/c/tint/+/16540
Reviewed-by: Sarah Mashayekhi <sarahmashay@google.com>
diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt
index 70cf7b4..d8b608e 100644
--- a/src/CMakeLists.txt
+++ b/src/CMakeLists.txt
@@ -216,6 +216,7 @@
ast/fallthrough_statement_test.cc
ast/function_test.cc
ast/identifier_expression_test.cc
+ ast/if_statement_test.cc
ast/import_test.cc
ast/int_literal_test.cc
ast/location_decoration_test.cc
diff --git a/src/ast/else_statement.cc b/src/ast/else_statement.cc
index 1ad7a02..307daec 100644
--- a/src/ast/else_statement.cc
+++ b/src/ast/else_statement.cc
@@ -53,8 +53,15 @@
void ElseStatement::to_str(std::ostream& out, size_t indent) const {
make_indent(out, indent);
out << "Else{" << std::endl;
- if (condition_ != nullptr)
- condition_->to_str(out, indent + 2);
+ if (condition_ != nullptr) {
+ make_indent(out, indent + 2);
+ out << "(" << std::endl;
+
+ condition_->to_str(out, indent + 4);
+
+ make_indent(out, indent + 2);
+ out << ")" << std::endl;
+ }
make_indent(out, indent + 2);
out << "{" << std::endl;
diff --git a/src/ast/else_statement_test.cc b/src/ast/else_statement_test.cc
index 4f20122..b9c5f1b 100644
--- a/src/ast/else_statement_test.cc
+++ b/src/ast/else_statement_test.cc
@@ -110,7 +110,9 @@
std::ostringstream out;
e.to_str(out, 2);
EXPECT_EQ(out.str(), R"( Else{
- ConstInitializer{true}
+ (
+ ConstInitializer{true}
+ )
{
Nop{}
}
diff --git a/src/ast/if_statement.cc b/src/ast/if_statement.cc
index 364b828..f000f2e 100644
--- a/src/ast/if_statement.cc
+++ b/src/ast/if_statement.cc
@@ -35,11 +35,40 @@
IfStatement::~IfStatement() = default;
bool IfStatement::IsValid() const {
- if (condition_ == nullptr)
+ if (condition_ == nullptr || !condition_->IsValid())
return false;
- if (premerge_.size() > 0 && else_statements_.size() > 1)
- return false;
+ for (const auto& stmt : body_) {
+ if (stmt == nullptr || !stmt->IsValid())
+ return false;
+ }
+
+ bool found_else = false;
+ for (const auto& el : else_statements_) {
+ // Else statement must be last
+ if (found_else)
+ return false;
+
+ if (el == nullptr || !el->IsValid())
+ return false;
+
+ if (el->condition() == nullptr)
+ found_else = true;
+ }
+
+ for (const auto& stmt : premerge_) {
+ if (stmt == nullptr || !stmt->IsValid())
+ return false;
+ }
+
+ if (premerge_.size() > 0) {
+ // Premerge only with a single else statement
+ if (else_statements_.size() != 1)
+ return false;
+ // Must be an else, not an elseif
+ if (else_statements_[0]->condition() != nullptr)
+ return false;
+ }
return true;
}
@@ -47,28 +76,43 @@
void IfStatement::to_str(std::ostream& out, size_t indent) const {
make_indent(out, indent);
out << "If{" << std::endl;
- condition_->to_str(out, indent + 2);
- out << std::endl;
+
+ // Open if conditional
+ make_indent(out, indent + 2);
+ out << "(" << std::endl;
+
+ condition_->to_str(out, indent + 4);
+
+ // Close if conditional
+ make_indent(out, indent + 2);
+ out << ")" << std::endl;
+
+ // Open if body
make_indent(out, indent + 2);
out << "{" << std::endl;
for (const auto& stmt : body_)
stmt->to_str(out, indent + 4);
+ // Close the if body
make_indent(out, indent + 2);
out << "}" << std::endl;
+ // Close the If
+ make_indent(out, indent);
+ out << "}" << std::endl;
+
for (const auto& e : else_statements_)
- e->to_str(out, indent + 2);
+ e->to_str(out, indent);
if (premerge_.size() > 0) {
- make_indent(out, indent + 2);
+ make_indent(out, indent);
out << "premerge{" << std::endl;
for (const auto& stmt : premerge_)
- stmt->to_str(out, indent + 4);
+ stmt->to_str(out, indent + 2);
- make_indent(out, indent + 2);
+ make_indent(out, indent);
out << "}" << std::endl;
}
}
diff --git a/src/ast/if_statement_test.cc b/src/ast/if_statement_test.cc
new file mode 100644
index 0000000..e9990af
--- /dev/null
+++ b/src/ast/if_statement_test.cc
@@ -0,0 +1,365 @@
+// 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/ast/if_statement.h"
+
+#include "gtest/gtest.h"
+#include "src/ast/identifier_expression.h"
+#include "src/ast/kill_statement.h"
+#include "src/ast/nop_statement.h"
+
+namespace tint {
+namespace ast {
+
+using IfStatementTest = testing::Test;
+
+TEST_F(IfStatementTest, Creation) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ auto cond_ptr = cond.get();
+ auto stmt_ptr = body[0].get();
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ EXPECT_EQ(stmt.condition(), cond_ptr);
+ ASSERT_EQ(stmt.body().size(), 1);
+ EXPECT_EQ(stmt.body()[0].get(), stmt_ptr);
+}
+
+TEST_F(IfStatementTest, Creation_WithSource) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ IfStatement stmt(Source{20, 2}, std::move(cond), std::move(body));
+ auto src = stmt.source();
+ EXPECT_EQ(src.line, 20);
+ EXPECT_EQ(src.column, 2);
+}
+
+TEST_F(IfStatementTest, IsIf) {
+ IfStatement stmt;
+ EXPECT_TRUE(stmt.IsIf());
+}
+
+TEST_F(IfStatementTest, IsValid) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ EXPECT_TRUE(stmt.IsValid());
+}
+
+TEST_F(IfStatementTest, IsValid_WithElseStatements) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ std::vector<std::unique_ptr<ElseStatement>> else_stmts;
+ else_stmts.push_back(std::make_unique<ElseStatement>());
+ else_stmts[0]->set_condition(std::make_unique<IdentifierExpression>("Ident"));
+ else_stmts.push_back(std::make_unique<ElseStatement>());
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ stmt.set_else_statements(std::move(else_stmts));
+ EXPECT_TRUE(stmt.IsValid());
+}
+
+TEST_F(IfStatementTest, IsValid_WithPremerge) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ std::vector<std::unique_ptr<ElseStatement>> else_stmts;
+ else_stmts.push_back(std::make_unique<ElseStatement>());
+
+ std::vector<std::unique_ptr<Statement>> premerge;
+ premerge.push_back(std::make_unique<NopStatement>());
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ stmt.set_else_statements(std::move(else_stmts));
+ stmt.set_premerge(std::move(premerge));
+ EXPECT_TRUE(stmt.IsValid());
+}
+
+TEST_F(IfStatementTest, IsValid_MissingCondition) {
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ IfStatement stmt(nullptr, std::move(body));
+ EXPECT_FALSE(stmt.IsValid());
+}
+
+TEST_F(IfStatementTest, IsValid_InvalidCondition) {
+ auto cond = std::make_unique<IdentifierExpression>("");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ EXPECT_FALSE(stmt.IsValid());
+}
+
+TEST_F(IfStatementTest, IsValid_NullBodyStatement) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+ body.push_back(nullptr);
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ EXPECT_FALSE(stmt.IsValid());
+}
+
+TEST_F(IfStatementTest, IsValid_InvalidBodyStatement) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+ body.push_back(std::make_unique<IfStatement>());
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ EXPECT_FALSE(stmt.IsValid());
+}
+
+TEST_F(IfStatementTest, IsValid_NullElseStatement) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ std::vector<std::unique_ptr<ElseStatement>> else_stmts;
+ else_stmts.push_back(std::make_unique<ElseStatement>());
+ else_stmts[0]->set_condition(std::make_unique<IdentifierExpression>("Ident"));
+ else_stmts.push_back(std::make_unique<ElseStatement>());
+ else_stmts.push_back(nullptr);
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ stmt.set_else_statements(std::move(else_stmts));
+ EXPECT_FALSE(stmt.IsValid());
+}
+
+TEST_F(IfStatementTest, IsValid_InvalidElseStatement) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ std::vector<std::unique_ptr<ElseStatement>> else_stmts;
+ else_stmts.push_back(std::make_unique<ElseStatement>());
+ else_stmts[0]->set_condition(std::make_unique<IdentifierExpression>(""));
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ stmt.set_else_statements(std::move(else_stmts));
+ EXPECT_FALSE(stmt.IsValid());
+}
+
+TEST_F(IfStatementTest, IsValid_NullPremergeStatement) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ std::vector<std::unique_ptr<ElseStatement>> else_stmts;
+ else_stmts.push_back(std::make_unique<ElseStatement>());
+
+ std::vector<std::unique_ptr<Statement>> premerge;
+ premerge.push_back(std::make_unique<NopStatement>());
+ premerge.push_back(nullptr);
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ stmt.set_else_statements(std::move(else_stmts));
+ stmt.set_premerge(std::move(premerge));
+ EXPECT_FALSE(stmt.IsValid());
+}
+
+TEST_F(IfStatementTest, IsValid_InvalidPremergeStatement) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ std::vector<std::unique_ptr<ElseStatement>> else_stmts;
+ else_stmts.push_back(std::make_unique<ElseStatement>());
+
+ std::vector<std::unique_ptr<Statement>> premerge;
+ premerge.push_back(std::make_unique<IfStatement>());
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ stmt.set_else_statements(std::move(else_stmts));
+ stmt.set_premerge(std::move(premerge));
+ EXPECT_FALSE(stmt.IsValid());
+}
+
+TEST_F(IfStatementTest, IsValid_PremergeWithElseIf) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ std::vector<std::unique_ptr<ElseStatement>> else_stmts;
+ else_stmts.push_back(std::make_unique<ElseStatement>());
+ else_stmts[0]->set_condition(std::make_unique<IdentifierExpression>("ident"));
+
+ std::vector<std::unique_ptr<Statement>> premerge;
+ premerge.push_back(std::make_unique<NopStatement>());
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ stmt.set_else_statements(std::move(else_stmts));
+ stmt.set_premerge(std::move(premerge));
+ EXPECT_FALSE(stmt.IsValid());
+}
+
+TEST_F(IfStatementTest, IsValid_PremergeWithoutElse) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ std::vector<std::unique_ptr<Statement>> premerge;
+ premerge.push_back(std::make_unique<NopStatement>());
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ stmt.set_premerge(std::move(premerge));
+ EXPECT_FALSE(stmt.IsValid());
+}
+
+TEST_F(IfStatementTest, IsValid_MultipleElseWiththoutCondition) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ std::vector<std::unique_ptr<ElseStatement>> else_stmts;
+ else_stmts.push_back(std::make_unique<ElseStatement>());
+ else_stmts.push_back(std::make_unique<ElseStatement>());
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ stmt.set_else_statements(std::move(else_stmts));
+ EXPECT_FALSE(stmt.IsValid());
+}
+
+TEST_F(IfStatementTest, IsValid_ElseNotLast) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ std::vector<std::unique_ptr<ElseStatement>> else_stmts;
+ else_stmts.push_back(std::make_unique<ElseStatement>());
+ else_stmts.push_back(std::make_unique<ElseStatement>());
+ else_stmts[1]->set_condition(std::make_unique<IdentifierExpression>("ident"));
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ stmt.set_else_statements(std::move(else_stmts));
+ EXPECT_FALSE(stmt.IsValid());
+}
+
+TEST_F(IfStatementTest, ToStr) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ IfStatement stmt(std::move(cond), std::move(body));
+
+ std::ostringstream out;
+ stmt.to_str(out, 2);
+ EXPECT_EQ(out.str(), R"( If{
+ (
+ Identifier{cond}
+ )
+ {
+ Nop{}
+ }
+ }
+)");
+}
+
+TEST_F(IfStatementTest, ToStr_WithElseStatements) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ std::vector<std::unique_ptr<Statement>> else_if_body;
+ else_if_body.push_back(std::make_unique<KillStatement>());
+
+ std::vector<std::unique_ptr<Statement>> else_body;
+ else_body.push_back(std::make_unique<NopStatement>());
+ else_body.push_back(std::make_unique<KillStatement>());
+
+ std::vector<std::unique_ptr<ElseStatement>> else_stmts;
+ else_stmts.push_back(std::make_unique<ElseStatement>());
+ else_stmts[0]->set_condition(std::make_unique<IdentifierExpression>("ident"));
+ else_stmts[0]->set_body(std::move(else_if_body));
+ else_stmts.push_back(std::make_unique<ElseStatement>());
+ else_stmts[1]->set_body(std::move(else_body));
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ stmt.set_else_statements(std::move(else_stmts));
+
+ std::ostringstream out;
+ stmt.to_str(out, 2);
+ EXPECT_EQ(out.str(), R"( If{
+ (
+ Identifier{cond}
+ )
+ {
+ Nop{}
+ }
+ }
+ Else{
+ (
+ Identifier{ident}
+ )
+ {
+ Kill{}
+ }
+ }
+ Else{
+ {
+ Nop{}
+ Kill{}
+ }
+ }
+)");
+}
+
+TEST_F(IfStatementTest, ToStr_WithPremerge) {
+ auto cond = std::make_unique<IdentifierExpression>("cond");
+ std::vector<std::unique_ptr<Statement>> body;
+ body.push_back(std::make_unique<NopStatement>());
+
+ std::vector<std::unique_ptr<ElseStatement>> else_stmts;
+ else_stmts.push_back(std::make_unique<ElseStatement>());
+
+ std::vector<std::unique_ptr<Statement>> premerge;
+ premerge.push_back(std::make_unique<NopStatement>());
+
+ IfStatement stmt(std::move(cond), std::move(body));
+ stmt.set_else_statements(std::move(else_stmts));
+ stmt.set_premerge(std::move(premerge));
+
+ std::ostringstream out;
+ stmt.to_str(out, 2);
+ EXPECT_EQ(out.str(), R"( If{
+ (
+ Identifier{cond}
+ )
+ {
+ Nop{}
+ }
+ }
+ Else{
+ {
+ }
+ }
+ premerge{
+ Nop{}
+ }
+)");
+}
+
+} // namespace ast
+} // namespace tint