[validation] validates if global variables have a storage class

This CL moves the global variables checks to a function
Adds tests and checks for validation rule v-0022:
Global variables must have a storage class.

Bug: tint: 6
Change-Id: I2f2cd7df6e849bfd1ddfbca35568c6fc3345efa6
Reviewed-on: https://dawn-review.googlesource.com/c/tint/+/27283
Commit-Queue: Sarah Mashayekhi <sarahmashay@google.com>
Reviewed-by: dan sinclair <dsinclair@chromium.org>
This commit is contained in:
Sarah Mashayekhi 2020-08-25 15:04:53 +00:00 committed by Commit Bot service account
parent deea295fc1
commit 4fb431c90e
3 changed files with 48 additions and 7 deletions

View File

@ -35,14 +35,9 @@ bool ValidatorImpl::Validate(const ast::Module* module) {
return false; return false;
} }
function_stack_.push_scope(); function_stack_.push_scope();
for (const auto& var : module->global_variables()) { if (!ValidateGlobalVariables(module->global_variables())) {
if (variable_stack_.has(var->name())) {
set_error(var->source(),
"v-0011: redeclared global identifier '" + var->name() + "'");
return false; return false;
} }
variable_stack_.set_global(var->name(), var.get());
}
if (!CheckImports(module)) { if (!CheckImports(module)) {
return false; return false;
} }
@ -59,6 +54,24 @@ bool ValidatorImpl::Validate(const ast::Module* module) {
return true; return true;
} }
bool ValidatorImpl::ValidateGlobalVariables(
const ast::VariableList& global_vars) {
for (const auto& var : global_vars) {
if (variable_stack_.has(var->name())) {
set_error(var->source(),
"v-0011: redeclared global identifier '" + var->name() + "'");
return false;
}
if (var->storage_class() == ast::StorageClass::kNone) {
set_error(var->source(),
"v-0022: global variables must have a storage class");
return false;
}
variable_stack_.set_global(var->name(), var.get());
}
return true;
}
bool ValidatorImpl::ValidateEntryPoints(const ast::EntryPointList& eps) { bool ValidatorImpl::ValidateEntryPoints(const ast::EntryPointList& eps) {
for (const auto& ep : eps) { for (const auto& ep : eps) {
auto* ep_ptr = ep.get(); auto* ep_ptr = ep.get();

View File

@ -53,6 +53,10 @@ class ValidatorImpl {
/// @param src the source causing the error /// @param src the source causing the error
/// @param msg the error message /// @param msg the error message
void set_error(const Source& src, const std::string& msg); void set_error(const Source& src, const std::string& msg);
/// Validate global variables
/// @param global_vars list of global variables to check
/// @returns true if the validation was successful
bool ValidateGlobalVariables(const ast::VariableList& global_vars);
/// Validates Functions /// Validates Functions
/// @param funcs the functions to check /// @param funcs the functions to check
/// @returns true if the validation was successful /// @returns true if the validation was successful

View File

@ -257,6 +257,30 @@ TEST_F(ValidatorTest, AssignIncompatibleTypesInBlockStatement_Fail) {
"12:34: v-000x: invalid assignment of '__i32' to '__f32'"); "12:34: v-000x: invalid assignment of '__i32' to '__f32'");
} }
TEST_F(ValidatorTest, GlobalVariableWithStorageClass_Pass) {
// var<in> gloabl_var: f32;
ast::type::F32Type f32;
auto global_var = std::make_unique<ast::Variable>(
Source{12, 34}, "global_var", ast::StorageClass::kInput, &f32);
mod()->AddGlobalVariable(std::move(global_var));
EXPECT_TRUE(td()->Determine()) << td()->error();
tint::ValidatorImpl v;
EXPECT_TRUE(v.Validate(mod())) << v.error();
}
TEST_F(ValidatorTest, GlobalVariableNoStorageClass_Fail) {
// var gloabl_var: f32;
ast::type::F32Type f32;
auto global_var = std::make_unique<ast::Variable>(
Source{12, 34}, "global_var", ast::StorageClass::kNone, &f32);
mod()->AddGlobalVariable(std::move(global_var));
EXPECT_TRUE(td()->Determine()) << td()->error();
tint::ValidatorImpl v;
EXPECT_FALSE(v.Validate(mod()));
EXPECT_EQ(v.error(),
"12:34: v-0022: global variables must have a storage class");
}
TEST_F(ValidatorTest, UsingUndefinedVariableGlobalVariable_Fail) { TEST_F(ValidatorTest, UsingUndefinedVariableGlobalVariable_Fail) {
// var global_var: f32 = 2.1; // var global_var: f32 = 2.1;
// fn my_func() -> f32 { // fn my_func() -> f32 {