// 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 "gtest/gtest.h"
#include "spirv/unified1/GLSL.std.450.h"
#include "src/ast/case_statement.h"
#include "src/ast/fallthrough_statement.h"
#include "src/ast/return_statement.h"
#include "src/ast/scalar_constructor_expression.h"
#include "src/ast/sint_literal.h"
#include "src/ast/switch_statement.h"
#include "src/ast/type/alias_type.h"
#include "src/ast/type/f32_type.h"
#include "src/ast/type/i32_type.h"
#include "src/ast/type/u32_type.h"
#include "src/ast/uint_literal.h"
#include "src/ast/variable.h"
#include "src/ast/variable_decl_statement.h"
#include "src/type_determiner.h"
#include "src/validator/validator_impl.h"
#include "src/validator/validator_test_helper.h"

namespace tint {
namespace {

class ValidateControlBlockTest : public ValidatorTestHelper,
                                 public testing::Test {};

TEST_F(ValidateControlBlockTest, SwitchSelectorExpressionNoneIntegerType_Fail) {
  // var a : f32 = 3.14;
  // switch (a) {
  //   default: {}
  // }
  ast::type::F32Type f32;
  auto* var = create<ast::Variable>("a", ast::StorageClass::kNone, &f32);
  var->set_constructor(create<ast::ScalarConstructorExpression>(
      create<ast::SintLiteral>(&f32, 3.14f)));

  auto* cond =
      create<ast::IdentifierExpression>(Source{Source::Location{12, 34}}, "a");
  ast::CaseSelectorList default_csl;
  auto* block_default = create<ast::BlockStatement>();
  ast::CaseStatementList body;
  body.push_back(create<ast::CaseStatement>(default_csl, block_default));

  auto* block = create<ast::BlockStatement>();
  block->append(create<ast::VariableDeclStatement>(var));
  block->append(create<ast::SwitchStatement>(cond, body));

  EXPECT_TRUE(td()->DetermineStatements(block)) << td()->error();
  EXPECT_FALSE(v()->ValidateStatements(block));
  EXPECT_EQ(v()->error(),
            "12:34: v-0025: switch statement selector expression must be "
            "of a scalar integer type");
}

TEST_F(ValidateControlBlockTest, SwitchWithoutDefault_Fail) {
  // var a : i32 = 2;
  // switch (a) {
  //   case 1: {}
  // }
  ast::type::I32Type i32;
  auto* var = create<ast::Variable>("a", ast::StorageClass::kNone, &i32);
  var->set_constructor(create<ast::ScalarConstructorExpression>(
      create<ast::SintLiteral>(&i32, 2)));

  auto* cond = create<ast::IdentifierExpression>("a");
  ast::CaseSelectorList csl;
  csl.push_back(create<ast::SintLiteral>(&i32, 1));
  ast::CaseStatementList body;
  body.push_back(
      create<ast::CaseStatement>(csl, create<ast::BlockStatement>()));

  auto* block = create<ast::BlockStatement>();
  block->append(create<ast::VariableDeclStatement>(var));
  block->append(create<ast::SwitchStatement>(Source{Source::Location{12, 34}},
                                             cond, body));

  EXPECT_TRUE(td()->DetermineStatements(block)) << td()->error();
  EXPECT_FALSE(v()->ValidateStatements(block));
  EXPECT_EQ(v()->error(),
            "12:34: v-0008: switch statement must have exactly one default "
            "clause");
}

TEST_F(ValidateControlBlockTest, SwitchWithTwoDefault_Fail) {
  // var a : i32 = 2;
  // switch (a) {
  //   default: {}
  //   case 1: {}
  //   default: {}
  // }
  ast::type::I32Type i32;
  auto* var = create<ast::Variable>("a", ast::StorageClass::kNone, &i32);
  var->set_constructor(create<ast::ScalarConstructorExpression>(
      create<ast::SintLiteral>(&i32, 2)));

  ast::CaseStatementList switch_body;
  auto* cond = create<ast::IdentifierExpression>("a");

  ast::CaseSelectorList default_csl_1;
  auto* block_default_1 = create<ast::BlockStatement>();
  switch_body.push_back(
      create<ast::CaseStatement>(default_csl_1, block_default_1));

  ast::CaseSelectorList csl_case_1;
  csl_case_1.push_back(create<ast::SintLiteral>(&i32, 1));
  auto* block_case_1 = create<ast::BlockStatement>();
  switch_body.push_back(create<ast::CaseStatement>(csl_case_1, block_case_1));

  ast::CaseSelectorList default_csl_2;
  auto* block_default_2 = create<ast::BlockStatement>();
  switch_body.push_back(
      create<ast::CaseStatement>(default_csl_2, block_default_2));

  auto* block = create<ast::BlockStatement>();
  block->append(create<ast::VariableDeclStatement>(var));
  block->append(create<ast::SwitchStatement>(Source{Source::Location{12, 34}},
                                             cond, switch_body));

  EXPECT_TRUE(td()->DetermineStatements(block)) << td()->error();
  EXPECT_FALSE(v()->ValidateStatements(block));
  EXPECT_EQ(v()->error(),
            "12:34: v-0008: switch statement must have exactly one default "
            "clause");
}

TEST_F(ValidateControlBlockTest,
       SwitchConditionTypeMustMatchSelectorType2_Fail) {
  // var a : i32 = 2;
  // switch (a) {
  //   case 1: {}
  //   default: {}
  // }
  ast::type::U32Type u32;
  ast::type::I32Type i32;
  auto* var = create<ast::Variable>("a", ast::StorageClass::kNone, &i32);
  var->set_constructor(create<ast::ScalarConstructorExpression>(
      create<ast::SintLiteral>(&i32, 2)));

  ast::CaseStatementList switch_body;
  auto* cond = create<ast::IdentifierExpression>("a");

  ast::CaseSelectorList csl;
  csl.push_back(create<ast::UintLiteral>(&u32, 1));
  switch_body.push_back(create<ast::CaseStatement>(
      Source{Source::Location{12, 34}}, csl, create<ast::BlockStatement>()));

  ast::CaseSelectorList default_csl;
  auto* block_default = create<ast::BlockStatement>();
  switch_body.push_back(create<ast::CaseStatement>(default_csl, block_default));

  auto* block = create<ast::BlockStatement>();
  block->append(create<ast::VariableDeclStatement>(var));
  block->append(create<ast::SwitchStatement>(cond, switch_body));

  EXPECT_TRUE(td()->DetermineStatements(block)) << td()->error();
  EXPECT_FALSE(v()->ValidateStatements(block));
  EXPECT_EQ(v()->error(),
            "12:34: v-0026: the case selector values must have the same "
            "type as the selector expression.");
}

TEST_F(ValidateControlBlockTest,
       SwitchConditionTypeMustMatchSelectorType_Fail) {
  // var a : u32 = 2;
  // switch (a) {
  //   case -1: {}
  //   default: {}
  // }
  ast::type::U32Type u32;
  ast::type::I32Type i32;
  auto* var = create<ast::Variable>("a", ast::StorageClass::kNone, &u32);
  var->set_constructor(create<ast::ScalarConstructorExpression>(
      create<ast::UintLiteral>(&u32, 2)));

  ast::CaseStatementList switch_body;
  auto* cond = create<ast::IdentifierExpression>("a");

  ast::CaseSelectorList csl;
  csl.push_back(create<ast::SintLiteral>(&i32, -1));
  switch_body.push_back(create<ast::CaseStatement>(
      Source{Source::Location{12, 34}}, csl, create<ast::BlockStatement>()));

  ast::CaseSelectorList default_csl;
  auto* block_default = create<ast::BlockStatement>();
  switch_body.push_back(create<ast::CaseStatement>(default_csl, block_default));

  auto* block = create<ast::BlockStatement>();
  block->append(create<ast::VariableDeclStatement>(var));
  block->append(create<ast::SwitchStatement>(cond, switch_body));

  EXPECT_TRUE(td()->DetermineStatements(block)) << td()->error();
  EXPECT_FALSE(v()->ValidateStatements(block));
  EXPECT_EQ(v()->error(),
            "12:34: v-0026: the case selector values must have the same "
            "type as the selector expression.");
}

TEST_F(ValidateControlBlockTest, NonUniqueCaseSelectorValueUint_Fail) {
  // var a : u32 = 3;
  // switch (a) {
  //   case 0: {}
  //   case 2, 2: {}
  //   default: {}
  // }
  ast::type::U32Type u32;
  auto* var = create<ast::Variable>("a", ast::StorageClass::kNone, &u32);
  var->set_constructor(create<ast::ScalarConstructorExpression>(
      create<ast::UintLiteral>(&u32, 3)));

  ast::CaseStatementList switch_body;
  auto* cond = create<ast::IdentifierExpression>("a");

  ast::CaseSelectorList csl_1;
  csl_1.push_back(create<ast::UintLiteral>(&u32, 0));
  switch_body.push_back(
      create<ast::CaseStatement>(csl_1, create<ast::BlockStatement>()));

  ast::CaseSelectorList csl_2;
  csl_2.push_back(create<ast::UintLiteral>(&u32, 2));
  csl_2.push_back(create<ast::UintLiteral>(&u32, 2));
  switch_body.push_back(create<ast::CaseStatement>(
      Source{Source::Location{12, 34}}, csl_2, create<ast::BlockStatement>()));

  ast::CaseSelectorList default_csl;
  auto* block_default = create<ast::BlockStatement>();
  switch_body.push_back(create<ast::CaseStatement>(default_csl, block_default));

  auto* block = create<ast::BlockStatement>();
  block->append(create<ast::VariableDeclStatement>(var));
  block->append(create<ast::SwitchStatement>(cond, switch_body));

  EXPECT_TRUE(td()->DetermineStatements(block)) << td()->error();
  EXPECT_FALSE(v()->ValidateStatements(block));
  EXPECT_EQ(v()->error(),
            "12:34: v-0027: a literal value must not appear more than once "
            "in the case selectors for a switch statement: '2'");
}

TEST_F(ValidateControlBlockTest, NonUniqueCaseSelectorValueSint_Fail) {
  // var a : i32 = 2;
  // switch (a) {
  //   case 10: {}
  //   case 0,1,2,10: {}
  //   default: {}
  // }
  ast::type::I32Type i32;
  auto* var = create<ast::Variable>("a", ast::StorageClass::kNone, &i32);
  var->set_constructor(create<ast::ScalarConstructorExpression>(
      create<ast::SintLiteral>(&i32, 2)));

  ast::CaseStatementList switch_body;
  auto* cond = create<ast::IdentifierExpression>("a");

  ast::CaseSelectorList csl_1;
  csl_1.push_back(create<ast::SintLiteral>(&i32, 10));
  switch_body.push_back(
      create<ast::CaseStatement>(csl_1, create<ast::BlockStatement>()));

  ast::CaseSelectorList csl_2;
  csl_2.push_back(create<ast::SintLiteral>(&i32, 0));
  csl_2.push_back(create<ast::SintLiteral>(&i32, 1));
  csl_2.push_back(create<ast::SintLiteral>(&i32, 2));
  csl_2.push_back(create<ast::SintLiteral>(&i32, 10));
  switch_body.push_back(create<ast::CaseStatement>(
      Source{Source::Location{12, 34}}, csl_2, create<ast::BlockStatement>()));

  ast::CaseSelectorList default_csl;
  auto* block_default = create<ast::BlockStatement>();
  switch_body.push_back(create<ast::CaseStatement>(default_csl, block_default));

  auto* block = create<ast::BlockStatement>();
  block->append(create<ast::VariableDeclStatement>(var));
  block->append(create<ast::SwitchStatement>(cond, switch_body));

  EXPECT_TRUE(td()->DetermineStatements(block)) << td()->error();
  EXPECT_FALSE(v()->ValidateStatements(block));
  EXPECT_EQ(v()->error(),
            "12:34: v-0027: a literal value must not appear more than once in "
            "the case selectors for a switch statement: '10'");
}

TEST_F(ValidateControlBlockTest, LastClauseLastStatementIsFallthrough_Fail) {
  // var a : i32 = 2;
  // switch (a) {
  //   default: { fallthrough; }
  // }
  ast::type::I32Type i32;
  auto* var = create<ast::Variable>("a", ast::StorageClass::kNone, &i32);
  var->set_constructor(create<ast::ScalarConstructorExpression>(
      create<ast::SintLiteral>(&i32, 2)));

  auto* cond = create<ast::IdentifierExpression>("a");
  ast::CaseSelectorList default_csl;
  auto* block_default = create<ast::BlockStatement>();
  block_default->append(
      create<ast::FallthroughStatement>(Source{Source::Location{12, 34}}));
  ast::CaseStatementList body;
  body.push_back(create<ast::CaseStatement>(default_csl, block_default));

  auto* block = create<ast::BlockStatement>();
  block->append(create<ast::VariableDeclStatement>(var));
  block->append(create<ast::SwitchStatement>(cond, body));

  EXPECT_TRUE(td()->DetermineStatements(block)) << td()->error();
  EXPECT_FALSE(v()->ValidateStatements(block));
  EXPECT_EQ(v()->error(),
            "12:34: v-0028: a fallthrough statement must not appear as the "
            "last statement in last clause of a switch");
}

TEST_F(ValidateControlBlockTest, SwitchCase_Pass) {
  // var a : i32 = 2;
  // switch (a) {
  //   default: {}
  //   case 5: {}
  // }
  ast::type::I32Type i32;
  auto* var = create<ast::Variable>("a", ast::StorageClass::kNone, &i32);
  var->set_constructor(create<ast::ScalarConstructorExpression>(
      create<ast::SintLiteral>(&i32, 2)));

  auto* cond = create<ast::IdentifierExpression>("a");
  ast::CaseSelectorList default_csl;
  auto* block_default = create<ast::BlockStatement>();
  ast::CaseStatementList body;
  body.push_back(create<ast::CaseStatement>(Source{Source::Location{12, 34}},
                                            default_csl, block_default));
  ast::CaseSelectorList case_csl;
  case_csl.push_back(create<ast::SintLiteral>(&i32, 5));
  auto* block_case = create<ast::BlockStatement>();
  body.push_back(create<ast::CaseStatement>(case_csl, block_case));

  auto* block = create<ast::BlockStatement>();
  block->append(create<ast::VariableDeclStatement>(var));
  block->append(create<ast::SwitchStatement>(cond, body));

  EXPECT_TRUE(td()->DetermineStatements(block)) << td()->error();
  EXPECT_TRUE(v()->ValidateStatements(block)) << v()->error();
}

TEST_F(ValidateControlBlockTest, SwitchCaseAlias_Pass) {
  // type MyInt = u32;
  // var v: MyInt;
  // switch(v){
  //   default: {}
  // }

  ast::type::U32Type u32;
  ast::type::AliasType my_int{"MyInt", &u32};

  auto* var = create<ast::Variable>("a", ast::StorageClass::kNone, &my_int);
  var->set_constructor(create<ast::ScalarConstructorExpression>(
      create<ast::SintLiteral>(&u32, 2)));

  auto* cond = create<ast::IdentifierExpression>("a");
  ast::CaseSelectorList default_csl;
  auto* block_default = create<ast::BlockStatement>();
  ast::CaseStatementList body;
  body.push_back(create<ast::CaseStatement>(Source{Source::Location{12, 34}},
                                            default_csl, block_default));

  auto* block = create<ast::BlockStatement>();
  block->append(create<ast::VariableDeclStatement>(var));
  block->append(create<ast::SwitchStatement>(cond, body));

  mod()->AddConstructedType(&my_int);

  EXPECT_TRUE(td()->DetermineStatements(block)) << td()->error();
  EXPECT_TRUE(v()->ValidateStatements(block)) << v()->error();
}

}  // namespace
}  // namespace tint
