wsgl parser: Prefix methods with `expect_`
... for those that will internally error if the grammar does not match, otherwise will always return a valid AST object. This helps identify whether the caller is expected to error or not. Bug: tint:282 Change-Id: Ied94f717526a63033f2e6c9e94fca43dbf0b8f05 Reviewed-on: https://dawn-review.googlesource.com/c/tint/+/32001 Commit-Queue: Ben Clayton <bclayton@google.com> Commit-Queue: dan sinclair <dsinclair@chromium.org> Reviewed-by: dan sinclair <dsinclair@chromium.org>
This commit is contained in:
parent
8655b62207
commit
7b750dc733
|
@ -188,7 +188,7 @@ bool ParserImpl::Parse() {
|
||||||
// : global_decl* EOF
|
// : global_decl* EOF
|
||||||
void ParserImpl::translation_unit() {
|
void ParserImpl::translation_unit() {
|
||||||
for (;;) {
|
for (;;) {
|
||||||
global_decl();
|
expect_global_decl();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
@ -206,7 +206,7 @@ void ParserImpl::translation_unit() {
|
||||||
// | type_alias SEMICOLON
|
// | type_alias SEMICOLON
|
||||||
// | struct_decl SEMICOLON
|
// | struct_decl SEMICOLON
|
||||||
// | function_decl
|
// | function_decl
|
||||||
void ParserImpl::global_decl() {
|
void ParserImpl::expect_global_decl() {
|
||||||
auto t = peek();
|
auto t = peek();
|
||||||
if (t.IsEof()) {
|
if (t.IsEof()) {
|
||||||
return;
|
return;
|
||||||
|
@ -303,7 +303,7 @@ std::unique_ptr<ast::Variable> ParserImpl::global_variable_decl(
|
||||||
}
|
}
|
||||||
|
|
||||||
if (match(Token::Type::kEqual)) {
|
if (match(Token::Type::kEqual)) {
|
||||||
auto expr = const_expr();
|
auto expr = expect_const_expr();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return nullptr;
|
return nullptr;
|
||||||
if (expr == nullptr) {
|
if (expr == nullptr) {
|
||||||
|
@ -340,7 +340,7 @@ std::unique_ptr<ast::Variable> ParserImpl::global_constant_decl() {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto init = const_expr();
|
auto init = expect_const_expr();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return nullptr;
|
return nullptr;
|
||||||
if (init == nullptr) {
|
if (init == nullptr) {
|
||||||
|
@ -896,10 +896,10 @@ ast::type::Type* ParserImpl::type_decl() {
|
||||||
return ctx_.type_mgr().Get(std::make_unique<ast::type::U32Type>());
|
return ctx_.type_mgr().Get(std::make_unique<ast::type::U32Type>());
|
||||||
}
|
}
|
||||||
if (t.IsVec2() || t.IsVec3() || t.IsVec4()) {
|
if (t.IsVec2() || t.IsVec3() || t.IsVec4()) {
|
||||||
return type_decl_vector(t);
|
return expect_type_decl_vector(t);
|
||||||
}
|
}
|
||||||
if (t.IsPtr()) {
|
if (t.IsPtr()) {
|
||||||
return type_decl_pointer(t);
|
return expect_type_decl_pointer(t);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto decos = decoration_list();
|
auto decos = decoration_list();
|
||||||
|
@ -909,7 +909,7 @@ ast::type::Type* ParserImpl::type_decl() {
|
||||||
|
|
||||||
if (match(Token::Type::kArray)) {
|
if (match(Token::Type::kArray)) {
|
||||||
auto array_decos = cast_decorations<ast::ArrayDecoration>(decos);
|
auto array_decos = cast_decorations<ast::ArrayDecoration>(decos);
|
||||||
return type_decl_array(std::move(array_decos));
|
return expect_type_decl_array(std::move(array_decos));
|
||||||
}
|
}
|
||||||
|
|
||||||
expect_decorations_consumed(decos);
|
expect_decorations_consumed(decos);
|
||||||
|
@ -917,7 +917,7 @@ ast::type::Type* ParserImpl::type_decl() {
|
||||||
if (t.IsMat2x2() || t.IsMat2x3() || t.IsMat2x4() || t.IsMat3x2() ||
|
if (t.IsMat2x2() || t.IsMat2x3() || t.IsMat2x4() || t.IsMat3x2() ||
|
||||||
t.IsMat3x3() || t.IsMat3x4() || t.IsMat4x2() || t.IsMat4x3() ||
|
t.IsMat3x3() || t.IsMat3x4() || t.IsMat4x2() || t.IsMat4x3() ||
|
||||||
t.IsMat4x4()) {
|
t.IsMat4x4()) {
|
||||||
return type_decl_matrix(t);
|
return expect_type_decl_matrix(t);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto* texture_or_sampler = texture_sampler_types();
|
auto* texture_or_sampler = texture_sampler_types();
|
||||||
|
@ -931,7 +931,7 @@ ast::type::Type* ParserImpl::type_decl() {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
ast::type::Type* ParserImpl::type_decl_pointer(Token t) {
|
ast::type::Type* ParserImpl::expect_type_decl_pointer(Token t) {
|
||||||
next(); // Consume the peek
|
next(); // Consume the peek
|
||||||
|
|
||||||
t = next();
|
t = next();
|
||||||
|
@ -972,7 +972,7 @@ ast::type::Type* ParserImpl::type_decl_pointer(Token t) {
|
||||||
std::make_unique<ast::type::PointerType>(subtype, sc));
|
std::make_unique<ast::type::PointerType>(subtype, sc));
|
||||||
}
|
}
|
||||||
|
|
||||||
ast::type::Type* ParserImpl::type_decl_vector(Token t) {
|
ast::type::Type* ParserImpl::expect_type_decl_vector(Token t) {
|
||||||
next(); // Consume the peek
|
next(); // Consume the peek
|
||||||
|
|
||||||
uint32_t count = 2;
|
uint32_t count = 2;
|
||||||
|
@ -1005,7 +1005,8 @@ ast::type::Type* ParserImpl::type_decl_vector(Token t) {
|
||||||
std::make_unique<ast::type::VectorType>(subtype, count));
|
std::make_unique<ast::type::VectorType>(subtype, count));
|
||||||
}
|
}
|
||||||
|
|
||||||
ast::type::Type* ParserImpl::type_decl_array(ast::ArrayDecorationList decos) {
|
ast::type::Type* ParserImpl::expect_type_decl_array(
|
||||||
|
ast::ArrayDecorationList decos) {
|
||||||
const char* use = "array declaration";
|
const char* use = "array declaration";
|
||||||
|
|
||||||
if (!expect(use, Token::Type::kLessThan))
|
if (!expect(use, Token::Type::kLessThan))
|
||||||
|
@ -1033,7 +1034,7 @@ ast::type::Type* ParserImpl::type_decl_array(ast::ArrayDecorationList decos) {
|
||||||
return ctx_.type_mgr().Get(std::move(ty));
|
return ctx_.type_mgr().Get(std::move(ty));
|
||||||
}
|
}
|
||||||
|
|
||||||
ast::type::Type* ParserImpl::type_decl_matrix(Token t) {
|
ast::type::Type* ParserImpl::expect_type_decl_matrix(Token t) {
|
||||||
next(); // Consume the peek
|
next(); // Consume the peek
|
||||||
|
|
||||||
uint32_t rows = 2;
|
uint32_t rows = 2;
|
||||||
|
@ -1140,7 +1141,7 @@ std::unique_ptr<ast::type::StructType> ParserImpl::struct_decl(
|
||||||
if (!expect_ident("struct declaration", &name))
|
if (!expect_ident("struct declaration", &name))
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
auto body = struct_body_decl();
|
auto body = expect_struct_body_decl();
|
||||||
if (has_error()) {
|
if (has_error()) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
@ -1152,14 +1153,14 @@ std::unique_ptr<ast::type::StructType> ParserImpl::struct_decl(
|
||||||
|
|
||||||
// struct_body_decl
|
// struct_body_decl
|
||||||
// : BRACKET_LEFT struct_member* BRACKET_RIGHT
|
// : BRACKET_LEFT struct_member* BRACKET_RIGHT
|
||||||
ast::StructMemberList ParserImpl::struct_body_decl() {
|
ast::StructMemberList ParserImpl::expect_struct_body_decl() {
|
||||||
return expect_brace_block("struct declaration", [&] {
|
return expect_brace_block("struct declaration", [&] {
|
||||||
ast::StructMemberList members;
|
ast::StructMemberList members;
|
||||||
|
|
||||||
while (!peek().IsBraceRight() && !peek().IsEof()) {
|
while (!peek().IsBraceRight() && !peek().IsEof()) {
|
||||||
auto decos = decoration_list();
|
auto decos = decoration_list();
|
||||||
|
|
||||||
auto mem = struct_member(decos);
|
auto mem = expect_struct_member(decos);
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return ast::StructMemberList{};
|
return ast::StructMemberList{};
|
||||||
if (mem == nullptr) {
|
if (mem == nullptr) {
|
||||||
|
@ -1176,7 +1177,7 @@ ast::StructMemberList ParserImpl::struct_body_decl() {
|
||||||
|
|
||||||
// struct_member
|
// struct_member
|
||||||
// : struct_member_decoration_decl+ variable_ident_decl SEMICOLON
|
// : struct_member_decoration_decl+ variable_ident_decl SEMICOLON
|
||||||
std::unique_ptr<ast::StructMember> ParserImpl::struct_member(
|
std::unique_ptr<ast::StructMember> ParserImpl::expect_struct_member(
|
||||||
ast::DecorationList& decos) {
|
ast::DecorationList& decos) {
|
||||||
auto t = peek();
|
auto t = peek();
|
||||||
|
|
||||||
|
@ -1208,7 +1209,7 @@ std::unique_ptr<ast::Function> ParserImpl::function_decl(
|
||||||
auto func_decos = cast_decorations<ast::FunctionDecoration>(decos);
|
auto func_decos = cast_decorations<ast::FunctionDecoration>(decos);
|
||||||
f->set_decorations(std::move(func_decos));
|
f->set_decorations(std::move(func_decos));
|
||||||
|
|
||||||
auto body = body_stmt();
|
auto body = expect_body_stmt();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
|
@ -1241,7 +1242,7 @@ std::unique_ptr<ast::Function> ParserImpl::function_header() {
|
||||||
if (!expect_ident(use, &name))
|
if (!expect_ident(use, &name))
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
auto params = expect_paren_block(use, [&] { return param_list(); });
|
auto params = expect_paren_block(use, [&] { return expect_param_list(); });
|
||||||
|
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
@ -1266,7 +1267,7 @@ std::unique_ptr<ast::Function> ParserImpl::function_header() {
|
||||||
// param_list
|
// param_list
|
||||||
// :
|
// :
|
||||||
// | (variable_ident_decl COMMA)* variable_ident_decl
|
// | (variable_ident_decl COMMA)* variable_ident_decl
|
||||||
ast::VariableList ParserImpl::param_list() {
|
ast::VariableList ParserImpl::expect_param_list() {
|
||||||
auto t = peek();
|
auto t = peek();
|
||||||
|
|
||||||
ast::VariableList ret;
|
ast::VariableList ret;
|
||||||
|
@ -1341,13 +1342,13 @@ std::pair<ast::Builtin, Source> ParserImpl::expect_builtin() {
|
||||||
|
|
||||||
// body_stmt
|
// body_stmt
|
||||||
// : BRACKET_LEFT statements BRACKET_RIGHT
|
// : BRACKET_LEFT statements BRACKET_RIGHT
|
||||||
std::unique_ptr<ast::BlockStatement> ParserImpl::body_stmt() {
|
std::unique_ptr<ast::BlockStatement> ParserImpl::expect_body_stmt() {
|
||||||
return expect_brace_block("", [&] { return statements(); });
|
return expect_brace_block("", [&] { return statements(); });
|
||||||
}
|
}
|
||||||
|
|
||||||
// paren_rhs_stmt
|
// paren_rhs_stmt
|
||||||
// : PAREN_LEFT logical_or_expression PAREN_RIGHT
|
// : PAREN_LEFT logical_or_expression PAREN_RIGHT
|
||||||
std::unique_ptr<ast::Expression> ParserImpl::paren_rhs_stmt() {
|
std::unique_ptr<ast::Expression> ParserImpl::expect_paren_rhs_stmt() {
|
||||||
return expect_paren_block("", [&]() -> std::unique_ptr<ast::Expression> {
|
return expect_paren_block("", [&]() -> std::unique_ptr<ast::Expression> {
|
||||||
auto expr = logical_or_expression();
|
auto expr = logical_or_expression();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
|
@ -1495,7 +1496,7 @@ std::unique_ptr<ast::Statement> ParserImpl::statement() {
|
||||||
|
|
||||||
t = peek();
|
t = peek();
|
||||||
if (t.IsBraceLeft()) {
|
if (t.IsBraceLeft()) {
|
||||||
auto body = body_stmt();
|
auto body = expect_body_stmt();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return nullptr;
|
return nullptr;
|
||||||
if (body != nullptr)
|
if (body != nullptr)
|
||||||
|
@ -1589,7 +1590,7 @@ std::unique_ptr<ast::IfStatement> ParserImpl::if_stmt() {
|
||||||
if (!match(Token::Type::kIf, &source))
|
if (!match(Token::Type::kIf, &source))
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
auto condition = paren_rhs_stmt();
|
auto condition = expect_paren_rhs_stmt();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return nullptr;
|
return nullptr;
|
||||||
if (condition == nullptr) {
|
if (condition == nullptr) {
|
||||||
|
@ -1597,7 +1598,7 @@ std::unique_ptr<ast::IfStatement> ParserImpl::if_stmt() {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto body = body_stmt();
|
auto body = expect_body_stmt();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
|
@ -1631,7 +1632,7 @@ ast::ElseStatementList ParserImpl::elseif_stmt() {
|
||||||
auto source = t.source();
|
auto source = t.source();
|
||||||
next(); // Consume the peek
|
next(); // Consume the peek
|
||||||
|
|
||||||
auto condition = paren_rhs_stmt();
|
auto condition = expect_paren_rhs_stmt();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return {};
|
return {};
|
||||||
if (condition == nullptr) {
|
if (condition == nullptr) {
|
||||||
|
@ -1639,7 +1640,7 @@ ast::ElseStatementList ParserImpl::elseif_stmt() {
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
auto body = body_stmt();
|
auto body = expect_body_stmt();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return {};
|
return {};
|
||||||
|
|
||||||
|
@ -1664,7 +1665,7 @@ std::unique_ptr<ast::ElseStatement> ParserImpl::else_stmt() {
|
||||||
auto source = t.source();
|
auto source = t.source();
|
||||||
next(); // Consume the peek
|
next(); // Consume the peek
|
||||||
|
|
||||||
auto body = body_stmt();
|
auto body = expect_body_stmt();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
|
@ -1678,7 +1679,7 @@ std::unique_ptr<ast::SwitchStatement> ParserImpl::switch_stmt() {
|
||||||
if (!match(Token::Type::kSwitch, &source))
|
if (!match(Token::Type::kSwitch, &source))
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
auto condition = paren_rhs_stmt();
|
auto condition = expect_paren_rhs_stmt();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return nullptr;
|
return nullptr;
|
||||||
if (condition == nullptr) {
|
if (condition == nullptr) {
|
||||||
|
@ -1837,7 +1838,7 @@ ForHeader::~ForHeader() = default;
|
||||||
// SEMICOLON
|
// SEMICOLON
|
||||||
// logical_or_expression? SEMICOLON
|
// logical_or_expression? SEMICOLON
|
||||||
// (assignment_stmt | func_call_stmt)?
|
// (assignment_stmt | func_call_stmt)?
|
||||||
std::unique_ptr<ForHeader> ParserImpl::for_header() {
|
std::unique_ptr<ForHeader> ParserImpl::expect_for_header() {
|
||||||
std::unique_ptr<ast::Statement> initializer = nullptr;
|
std::unique_ptr<ast::Statement> initializer = nullptr;
|
||||||
if (initializer == nullptr) {
|
if (initializer == nullptr) {
|
||||||
initializer = func_call_stmt();
|
initializer = func_call_stmt();
|
||||||
|
@ -1894,7 +1895,8 @@ std::unique_ptr<ast::Statement> ParserImpl::for_stmt() {
|
||||||
if (!match(Token::Type::kFor, &source))
|
if (!match(Token::Type::kFor, &source))
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
auto header = expect_paren_block("for loop", [&] { return for_header(); });
|
auto header =
|
||||||
|
expect_paren_block("for loop", [&] { return expect_for_header(); });
|
||||||
if (header == nullptr)
|
if (header == nullptr)
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
|
@ -1963,7 +1965,7 @@ std::unique_ptr<ast::CallStatement> ParserImpl::func_call_stmt() {
|
||||||
t = peek();
|
t = peek();
|
||||||
ast::ExpressionList params;
|
ast::ExpressionList params;
|
||||||
if (!t.IsParenRight() && !t.IsEof()) {
|
if (!t.IsParenRight() && !t.IsEof()) {
|
||||||
params = argument_expression_list();
|
params = expect_argument_expression_list();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
@ -2003,7 +2005,7 @@ std::unique_ptr<ast::BlockStatement> ParserImpl::continuing_stmt() {
|
||||||
if (!match(Token::Type::kContinuing))
|
if (!match(Token::Type::kContinuing))
|
||||||
return std::make_unique<ast::BlockStatement>();
|
return std::make_unique<ast::BlockStatement>();
|
||||||
|
|
||||||
return body_stmt();
|
return expect_body_stmt();
|
||||||
}
|
}
|
||||||
|
|
||||||
// primary_expression
|
// primary_expression
|
||||||
|
@ -2026,7 +2028,7 @@ std::unique_ptr<ast::Expression> ParserImpl::primary_expression() {
|
||||||
|
|
||||||
t = peek();
|
t = peek();
|
||||||
if (t.IsParenLeft()) {
|
if (t.IsParenLeft()) {
|
||||||
auto paren = paren_rhs_stmt();
|
auto paren = expect_paren_rhs_stmt();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
|
@ -2058,7 +2060,7 @@ std::unique_ptr<ast::Expression> ParserImpl::primary_expression() {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto params = paren_rhs_stmt();
|
auto params = expect_paren_rhs_stmt();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return nullptr;
|
return nullptr;
|
||||||
if (params == nullptr) {
|
if (params == nullptr) {
|
||||||
|
@ -2083,7 +2085,7 @@ std::unique_ptr<ast::Expression> ParserImpl::primary_expression() {
|
||||||
auto ok = expect_paren_block("type constructor", [&] {
|
auto ok = expect_paren_block("type constructor", [&] {
|
||||||
t = peek();
|
t = peek();
|
||||||
if (!t.IsParenRight() && !t.IsEof()) {
|
if (!t.IsParenRight() && !t.IsEof()) {
|
||||||
params = argument_expression_list();
|
params = expect_argument_expression_list();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -2136,7 +2138,7 @@ std::unique_ptr<ast::Expression> ParserImpl::postfix_expr(
|
||||||
t = peek();
|
t = peek();
|
||||||
ast::ExpressionList params;
|
ast::ExpressionList params;
|
||||||
if (!t.IsParenRight() && !t.IsEof()) {
|
if (!t.IsParenRight() && !t.IsEof()) {
|
||||||
params = argument_expression_list();
|
params = expect_argument_expression_list();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
@ -2176,7 +2178,7 @@ std::unique_ptr<ast::Expression> ParserImpl::postfix_expression() {
|
||||||
|
|
||||||
// argument_expression_list
|
// argument_expression_list
|
||||||
// : (logical_or_expression COMMA)* logical_or_expression
|
// : (logical_or_expression COMMA)* logical_or_expression
|
||||||
ast::ExpressionList ParserImpl::argument_expression_list() {
|
ast::ExpressionList ParserImpl::expect_argument_expression_list() {
|
||||||
auto arg = logical_or_expression();
|
auto arg = logical_or_expression();
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return {};
|
return {};
|
||||||
|
@ -2242,7 +2244,7 @@ std::unique_ptr<ast::Expression> ParserImpl::unary_expression() {
|
||||||
// | STAR unary_expression multiplicative_expr
|
// | STAR unary_expression multiplicative_expr
|
||||||
// | FORWARD_SLASH unary_expression multiplicative_expr
|
// | FORWARD_SLASH unary_expression multiplicative_expr
|
||||||
// | MODULO unary_expression multiplicative_expr
|
// | MODULO unary_expression multiplicative_expr
|
||||||
std::unique_ptr<ast::Expression> ParserImpl::multiplicative_expr(
|
std::unique_ptr<ast::Expression> ParserImpl::expect_multiplicative_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs) {
|
std::unique_ptr<ast::Expression> lhs) {
|
||||||
auto t = peek();
|
auto t = peek();
|
||||||
|
|
||||||
|
@ -2267,7 +2269,7 @@ std::unique_ptr<ast::Expression> ParserImpl::multiplicative_expr(
|
||||||
add_error(peek(), "unable to parse right side of " + name + " expression");
|
add_error(peek(), "unable to parse right side of " + name + " expression");
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return multiplicative_expr(std::make_unique<ast::BinaryExpression>(
|
return expect_multiplicative_expr(std::make_unique<ast::BinaryExpression>(
|
||||||
source, op, std::move(lhs), std::move(rhs)));
|
source, op, std::move(lhs), std::move(rhs)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2280,14 +2282,14 @@ std::unique_ptr<ast::Expression> ParserImpl::multiplicative_expression() {
|
||||||
if (lhs == nullptr)
|
if (lhs == nullptr)
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
return multiplicative_expr(std::move(lhs));
|
return expect_multiplicative_expr(std::move(lhs));
|
||||||
}
|
}
|
||||||
|
|
||||||
// additive_expr
|
// additive_expr
|
||||||
// :
|
// :
|
||||||
// | PLUS multiplicative_expression additive_expr
|
// | PLUS multiplicative_expression additive_expr
|
||||||
// | MINUS multiplicative_expression additive_expr
|
// | MINUS multiplicative_expression additive_expr
|
||||||
std::unique_ptr<ast::Expression> ParserImpl::additive_expr(
|
std::unique_ptr<ast::Expression> ParserImpl::expect_additive_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs) {
|
std::unique_ptr<ast::Expression> lhs) {
|
||||||
auto t = peek();
|
auto t = peek();
|
||||||
|
|
||||||
|
@ -2309,7 +2311,7 @@ std::unique_ptr<ast::Expression> ParserImpl::additive_expr(
|
||||||
add_error(peek(), "unable to parse right side of + expression");
|
add_error(peek(), "unable to parse right side of + expression");
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return additive_expr(std::make_unique<ast::BinaryExpression>(
|
return expect_additive_expr(std::make_unique<ast::BinaryExpression>(
|
||||||
source, op, std::move(lhs), std::move(rhs)));
|
source, op, std::move(lhs), std::move(rhs)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2322,14 +2324,14 @@ std::unique_ptr<ast::Expression> ParserImpl::additive_expression() {
|
||||||
if (lhs == nullptr)
|
if (lhs == nullptr)
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
return additive_expr(std::move(lhs));
|
return expect_additive_expr(std::move(lhs));
|
||||||
}
|
}
|
||||||
|
|
||||||
// shift_expr
|
// shift_expr
|
||||||
// :
|
// :
|
||||||
// | LESS_THAN LESS_THAN additive_expression shift_expr
|
// | LESS_THAN LESS_THAN additive_expression shift_expr
|
||||||
// | GREATER_THAN GREATER_THAN additive_expression shift_expr
|
// | GREATER_THAN GREATER_THAN additive_expression shift_expr
|
||||||
std::unique_ptr<ast::Expression> ParserImpl::shift_expr(
|
std::unique_ptr<ast::Expression> ParserImpl::expect_shift_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs) {
|
std::unique_ptr<ast::Expression> lhs) {
|
||||||
auto t = peek();
|
auto t = peek();
|
||||||
auto source = t.source();
|
auto source = t.source();
|
||||||
|
@ -2359,7 +2361,7 @@ std::unique_ptr<ast::Expression> ParserImpl::shift_expr(
|
||||||
" expression");
|
" expression");
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return shift_expr(std::make_unique<ast::BinaryExpression>(
|
return expect_shift_expr(std::make_unique<ast::BinaryExpression>(
|
||||||
source, op, std::move(lhs), std::move(rhs)));
|
source, op, std::move(lhs), std::move(rhs)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2372,7 +2374,7 @@ std::unique_ptr<ast::Expression> ParserImpl::shift_expression() {
|
||||||
if (lhs == nullptr)
|
if (lhs == nullptr)
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
return shift_expr(std::move(lhs));
|
return expect_shift_expr(std::move(lhs));
|
||||||
}
|
}
|
||||||
|
|
||||||
// relational_expr
|
// relational_expr
|
||||||
|
@ -2381,7 +2383,7 @@ std::unique_ptr<ast::Expression> ParserImpl::shift_expression() {
|
||||||
// | GREATER_THAN shift_expression relational_expr
|
// | GREATER_THAN shift_expression relational_expr
|
||||||
// | LESS_THAN_EQUAL shift_expression relational_expr
|
// | LESS_THAN_EQUAL shift_expression relational_expr
|
||||||
// | GREATER_THAN_EQUAL shift_expression relational_expr
|
// | GREATER_THAN_EQUAL shift_expression relational_expr
|
||||||
std::unique_ptr<ast::Expression> ParserImpl::relational_expr(
|
std::unique_ptr<ast::Expression> ParserImpl::expect_relational_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs) {
|
std::unique_ptr<ast::Expression> lhs) {
|
||||||
auto t = peek();
|
auto t = peek();
|
||||||
ast::BinaryOp op = ast::BinaryOp::kNone;
|
ast::BinaryOp op = ast::BinaryOp::kNone;
|
||||||
|
@ -2407,7 +2409,7 @@ std::unique_ptr<ast::Expression> ParserImpl::relational_expr(
|
||||||
add_error(peek(), "unable to parse right side of " + name + " expression");
|
add_error(peek(), "unable to parse right side of " + name + " expression");
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return relational_expr(std::make_unique<ast::BinaryExpression>(
|
return expect_relational_expr(std::make_unique<ast::BinaryExpression>(
|
||||||
source, op, std::move(lhs), std::move(rhs)));
|
source, op, std::move(lhs), std::move(rhs)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2420,14 +2422,14 @@ std::unique_ptr<ast::Expression> ParserImpl::relational_expression() {
|
||||||
if (lhs == nullptr)
|
if (lhs == nullptr)
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
return relational_expr(std::move(lhs));
|
return expect_relational_expr(std::move(lhs));
|
||||||
}
|
}
|
||||||
|
|
||||||
// equality_expr
|
// equality_expr
|
||||||
// :
|
// :
|
||||||
// | EQUAL_EQUAL relational_expression equality_expr
|
// | EQUAL_EQUAL relational_expression equality_expr
|
||||||
// | NOT_EQUAL relational_expression equality_expr
|
// | NOT_EQUAL relational_expression equality_expr
|
||||||
std::unique_ptr<ast::Expression> ParserImpl::equality_expr(
|
std::unique_ptr<ast::Expression> ParserImpl::expect_equality_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs) {
|
std::unique_ptr<ast::Expression> lhs) {
|
||||||
auto t = peek();
|
auto t = peek();
|
||||||
ast::BinaryOp op = ast::BinaryOp::kNone;
|
ast::BinaryOp op = ast::BinaryOp::kNone;
|
||||||
|
@ -2449,7 +2451,7 @@ std::unique_ptr<ast::Expression> ParserImpl::equality_expr(
|
||||||
add_error(peek(), "unable to parse right side of " + name + " expression");
|
add_error(peek(), "unable to parse right side of " + name + " expression");
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return equality_expr(std::make_unique<ast::BinaryExpression>(
|
return expect_equality_expr(std::make_unique<ast::BinaryExpression>(
|
||||||
source, op, std::move(lhs), std::move(rhs)));
|
source, op, std::move(lhs), std::move(rhs)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2462,13 +2464,13 @@ std::unique_ptr<ast::Expression> ParserImpl::equality_expression() {
|
||||||
if (lhs == nullptr)
|
if (lhs == nullptr)
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
return equality_expr(std::move(lhs));
|
return expect_equality_expr(std::move(lhs));
|
||||||
}
|
}
|
||||||
|
|
||||||
// and_expr
|
// and_expr
|
||||||
// :
|
// :
|
||||||
// | AND equality_expression and_expr
|
// | AND equality_expression and_expr
|
||||||
std::unique_ptr<ast::Expression> ParserImpl::and_expr(
|
std::unique_ptr<ast::Expression> ParserImpl::expect_and_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs) {
|
std::unique_ptr<ast::Expression> lhs) {
|
||||||
auto t = peek();
|
auto t = peek();
|
||||||
if (!t.IsAnd())
|
if (!t.IsAnd())
|
||||||
|
@ -2484,7 +2486,7 @@ std::unique_ptr<ast::Expression> ParserImpl::and_expr(
|
||||||
add_error(peek(), "unable to parse right side of & expression");
|
add_error(peek(), "unable to parse right side of & expression");
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return and_expr(std::make_unique<ast::BinaryExpression>(
|
return expect_and_expr(std::make_unique<ast::BinaryExpression>(
|
||||||
source, ast::BinaryOp::kAnd, std::move(lhs), std::move(rhs)));
|
source, ast::BinaryOp::kAnd, std::move(lhs), std::move(rhs)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2497,13 +2499,13 @@ std::unique_ptr<ast::Expression> ParserImpl::and_expression() {
|
||||||
if (lhs == nullptr)
|
if (lhs == nullptr)
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
return and_expr(std::move(lhs));
|
return expect_and_expr(std::move(lhs));
|
||||||
}
|
}
|
||||||
|
|
||||||
// exclusive_or_expr
|
// exclusive_or_expr
|
||||||
// :
|
// :
|
||||||
// | XOR and_expression exclusive_or_expr
|
// | XOR and_expression exclusive_or_expr
|
||||||
std::unique_ptr<ast::Expression> ParserImpl::exclusive_or_expr(
|
std::unique_ptr<ast::Expression> ParserImpl::expect_exclusive_or_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs) {
|
std::unique_ptr<ast::Expression> lhs) {
|
||||||
auto t = peek();
|
auto t = peek();
|
||||||
if (!t.IsXor())
|
if (!t.IsXor())
|
||||||
|
@ -2519,7 +2521,7 @@ std::unique_ptr<ast::Expression> ParserImpl::exclusive_or_expr(
|
||||||
add_error(peek(), "unable to parse right side of ^ expression");
|
add_error(peek(), "unable to parse right side of ^ expression");
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return exclusive_or_expr(std::make_unique<ast::BinaryExpression>(
|
return expect_exclusive_or_expr(std::make_unique<ast::BinaryExpression>(
|
||||||
source, ast::BinaryOp::kXor, std::move(lhs), std::move(rhs)));
|
source, ast::BinaryOp::kXor, std::move(lhs), std::move(rhs)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2532,13 +2534,13 @@ std::unique_ptr<ast::Expression> ParserImpl::exclusive_or_expression() {
|
||||||
if (lhs == nullptr)
|
if (lhs == nullptr)
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
return exclusive_or_expr(std::move(lhs));
|
return expect_exclusive_or_expr(std::move(lhs));
|
||||||
}
|
}
|
||||||
|
|
||||||
// inclusive_or_expr
|
// inclusive_or_expr
|
||||||
// :
|
// :
|
||||||
// | OR exclusive_or_expression inclusive_or_expr
|
// | OR exclusive_or_expression inclusive_or_expr
|
||||||
std::unique_ptr<ast::Expression> ParserImpl::inclusive_or_expr(
|
std::unique_ptr<ast::Expression> ParserImpl::expect_inclusive_or_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs) {
|
std::unique_ptr<ast::Expression> lhs) {
|
||||||
auto t = peek();
|
auto t = peek();
|
||||||
if (!t.IsOr())
|
if (!t.IsOr())
|
||||||
|
@ -2554,7 +2556,7 @@ std::unique_ptr<ast::Expression> ParserImpl::inclusive_or_expr(
|
||||||
add_error(peek(), "unable to parse right side of | expression");
|
add_error(peek(), "unable to parse right side of | expression");
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return inclusive_or_expr(std::make_unique<ast::BinaryExpression>(
|
return expect_inclusive_or_expr(std::make_unique<ast::BinaryExpression>(
|
||||||
source, ast::BinaryOp::kOr, std::move(lhs), std::move(rhs)));
|
source, ast::BinaryOp::kOr, std::move(lhs), std::move(rhs)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2567,13 +2569,13 @@ std::unique_ptr<ast::Expression> ParserImpl::inclusive_or_expression() {
|
||||||
if (lhs == nullptr)
|
if (lhs == nullptr)
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
return inclusive_or_expr(std::move(lhs));
|
return expect_inclusive_or_expr(std::move(lhs));
|
||||||
}
|
}
|
||||||
|
|
||||||
// logical_and_expr
|
// logical_and_expr
|
||||||
// :
|
// :
|
||||||
// | AND_AND inclusive_or_expression logical_and_expr
|
// | AND_AND inclusive_or_expression logical_and_expr
|
||||||
std::unique_ptr<ast::Expression> ParserImpl::logical_and_expr(
|
std::unique_ptr<ast::Expression> ParserImpl::expect_logical_and_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs) {
|
std::unique_ptr<ast::Expression> lhs) {
|
||||||
auto t = peek();
|
auto t = peek();
|
||||||
if (!t.IsAndAnd())
|
if (!t.IsAndAnd())
|
||||||
|
@ -2589,7 +2591,7 @@ std::unique_ptr<ast::Expression> ParserImpl::logical_and_expr(
|
||||||
add_error(peek(), "unable to parse right side of && expression");
|
add_error(peek(), "unable to parse right side of && expression");
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return logical_and_expr(std::make_unique<ast::BinaryExpression>(
|
return expect_logical_and_expr(std::make_unique<ast::BinaryExpression>(
|
||||||
source, ast::BinaryOp::kLogicalAnd, std::move(lhs), std::move(rhs)));
|
source, ast::BinaryOp::kLogicalAnd, std::move(lhs), std::move(rhs)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2602,13 +2604,13 @@ std::unique_ptr<ast::Expression> ParserImpl::logical_and_expression() {
|
||||||
if (lhs == nullptr)
|
if (lhs == nullptr)
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
return logical_and_expr(std::move(lhs));
|
return expect_logical_and_expr(std::move(lhs));
|
||||||
}
|
}
|
||||||
|
|
||||||
// logical_or_expr
|
// logical_or_expr
|
||||||
// :
|
// :
|
||||||
// | OR_OR logical_and_expression logical_or_expr
|
// | OR_OR logical_and_expression logical_or_expr
|
||||||
std::unique_ptr<ast::Expression> ParserImpl::logical_or_expr(
|
std::unique_ptr<ast::Expression> ParserImpl::expect_logical_or_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs) {
|
std::unique_ptr<ast::Expression> lhs) {
|
||||||
auto t = peek();
|
auto t = peek();
|
||||||
if (!t.IsOrOr())
|
if (!t.IsOrOr())
|
||||||
|
@ -2624,7 +2626,7 @@ std::unique_ptr<ast::Expression> ParserImpl::logical_or_expr(
|
||||||
add_error(peek(), "unable to parse right side of || expression");
|
add_error(peek(), "unable to parse right side of || expression");
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return logical_or_expr(std::make_unique<ast::BinaryExpression>(
|
return expect_logical_or_expr(std::make_unique<ast::BinaryExpression>(
|
||||||
source, ast::BinaryOp::kLogicalOr, std::move(lhs), std::move(rhs)));
|
source, ast::BinaryOp::kLogicalOr, std::move(lhs), std::move(rhs)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2637,7 +2639,7 @@ std::unique_ptr<ast::Expression> ParserImpl::logical_or_expression() {
|
||||||
if (lhs == nullptr)
|
if (lhs == nullptr)
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
return logical_or_expr(std::move(lhs));
|
return expect_logical_or_expr(std::move(lhs));
|
||||||
}
|
}
|
||||||
|
|
||||||
// assignment_stmt
|
// assignment_stmt
|
||||||
|
@ -2725,12 +2727,12 @@ std::unique_ptr<ast::Literal> ParserImpl::const_literal() {
|
||||||
// const_expr
|
// const_expr
|
||||||
// : type_decl PAREN_LEFT (const_expr COMMA)? const_expr PAREN_RIGHT
|
// : type_decl PAREN_LEFT (const_expr COMMA)? const_expr PAREN_RIGHT
|
||||||
// | const_literal
|
// | const_literal
|
||||||
std::unique_ptr<ast::ConstructorExpression> ParserImpl::const_expr() {
|
std::unique_ptr<ast::ConstructorExpression> ParserImpl::expect_const_expr() {
|
||||||
return const_expr_internal(0);
|
return expect_const_expr_internal(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::unique_ptr<ast::ConstructorExpression> ParserImpl::const_expr_internal(
|
std::unique_ptr<ast::ConstructorExpression>
|
||||||
uint32_t depth) {
|
ParserImpl::expect_const_expr_internal(uint32_t depth) {
|
||||||
auto t = peek();
|
auto t = peek();
|
||||||
|
|
||||||
if (depth > kMaxConstExprDepth) {
|
if (depth > kMaxConstExprDepth) {
|
||||||
|
@ -2744,7 +2746,7 @@ std::unique_ptr<ast::ConstructorExpression> ParserImpl::const_expr_internal(
|
||||||
if (type != nullptr) {
|
if (type != nullptr) {
|
||||||
ast::ExpressionList params;
|
ast::ExpressionList params;
|
||||||
bool ok = expect_paren_block("type constructor", [&] {
|
bool ok = expect_paren_block("type constructor", [&] {
|
||||||
auto param = const_expr_internal(depth + 1);
|
auto param = expect_const_expr_internal(depth + 1);
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return false;
|
return false;
|
||||||
if (param == nullptr) {
|
if (param == nullptr) {
|
||||||
|
@ -2753,7 +2755,7 @@ std::unique_ptr<ast::ConstructorExpression> ParserImpl::const_expr_internal(
|
||||||
}
|
}
|
||||||
params.push_back(std::move(param));
|
params.push_back(std::move(param));
|
||||||
while (match(Token::Type::kComma)) {
|
while (match(Token::Type::kComma)) {
|
||||||
param = const_expr_internal(depth + 1);
|
param = expect_const_expr_internal(depth + 1);
|
||||||
if (has_error())
|
if (has_error())
|
||||||
return false;
|
return false;
|
||||||
if (param == nullptr) {
|
if (param == nullptr) {
|
||||||
|
|
|
@ -155,8 +155,8 @@ class ParserImpl {
|
||||||
|
|
||||||
/// Parses the `translation_unit` grammar element
|
/// Parses the `translation_unit` grammar element
|
||||||
void translation_unit();
|
void translation_unit();
|
||||||
/// Parses the `global_decl` grammar element
|
/// Parses the `global_decl` grammar element, erroring on parse failure.
|
||||||
void global_decl();
|
void expect_global_decl();
|
||||||
/// Parses a `global_variable_decl` grammar element with the initial
|
/// Parses a `global_variable_decl` grammar element with the initial
|
||||||
/// `variable_decoration_list*` provided as |decos|.
|
/// `variable_decoration_list*` provided as |decos|.
|
||||||
/// @returns the variable parsed or nullptr
|
/// @returns the variable parsed or nullptr
|
||||||
|
@ -190,14 +190,16 @@ class ParserImpl {
|
||||||
/// @param decos the list of decorations for the struct declaration.
|
/// @param decos the list of decorations for the struct declaration.
|
||||||
std::unique_ptr<ast::type::StructType> struct_decl(
|
std::unique_ptr<ast::type::StructType> struct_decl(
|
||||||
ast::DecorationList& decos);
|
ast::DecorationList& decos);
|
||||||
/// Parses a `struct_body_decl` grammar element
|
/// Parses a `struct_body_decl` grammar element, erroring on parse failure.
|
||||||
/// @returns the struct members
|
/// @returns the struct members
|
||||||
ast::StructMemberList struct_body_decl();
|
ast::StructMemberList expect_struct_body_decl();
|
||||||
/// Parses a `struct_member` grammar element with the initial
|
/// Parses a `struct_member` grammar element with the initial
|
||||||
/// `struct_member_decoration_decl+` provided as |decos|.
|
/// `struct_member_decoration_decl+` provided as |decos|, erroring on parse
|
||||||
|
/// failure.
|
||||||
/// @param decos the list of decorations for the struct member.
|
/// @param decos the list of decorations for the struct member.
|
||||||
/// @returns the struct member or nullptr
|
/// @returns the struct member or nullptr
|
||||||
std::unique_ptr<ast::StructMember> struct_member(ast::DecorationList& decos);
|
std::unique_ptr<ast::StructMember> expect_struct_member(
|
||||||
|
ast::DecorationList& decos);
|
||||||
/// Parses a `function_decl` grammar element with the initial
|
/// Parses a `function_decl` grammar element with the initial
|
||||||
/// `function_decoration_decl*` provided as |decos|.
|
/// `function_decoration_decl*` provided as |decos|.
|
||||||
/// @param decos the list of decorations for the function declaration.
|
/// @param decos the list of decorations for the function declaration.
|
||||||
|
@ -233,9 +235,9 @@ class ParserImpl {
|
||||||
/// Parses a `function_header` grammar element
|
/// Parses a `function_header` grammar element
|
||||||
/// @returns the parsed function nullptr otherwise
|
/// @returns the parsed function nullptr otherwise
|
||||||
std::unique_ptr<ast::Function> function_header();
|
std::unique_ptr<ast::Function> function_header();
|
||||||
/// Parses a `param_list` grammar element
|
/// Parses a `param_list` grammar element, erroring on parse failure.
|
||||||
/// @returns the parsed variables
|
/// @returns the parsed variables
|
||||||
ast::VariableList param_list();
|
ast::VariableList expect_param_list();
|
||||||
/// Parses a `pipeline_stage` grammar element, erroring if the next token does
|
/// Parses a `pipeline_stage` grammar element, erroring if the next token does
|
||||||
/// not match a stage name.
|
/// not match a stage name.
|
||||||
/// @returns the pipeline stage or PipelineStage::kNone if none matched, along
|
/// @returns the pipeline stage or PipelineStage::kNone if none matched, along
|
||||||
|
@ -246,12 +248,12 @@ class ParserImpl {
|
||||||
/// @returns the builtin or Builtin::kNone if none matched, along with the
|
/// @returns the builtin or Builtin::kNone if none matched, along with the
|
||||||
/// source location for the stage.
|
/// source location for the stage.
|
||||||
std::pair<ast::Builtin, Source> expect_builtin();
|
std::pair<ast::Builtin, Source> expect_builtin();
|
||||||
/// Parses a `body_stmt` grammar element
|
/// Parses a `body_stmt` grammar element, erroring on parse failure.
|
||||||
/// @returns the parsed statements
|
/// @returns the parsed statements
|
||||||
std::unique_ptr<ast::BlockStatement> body_stmt();
|
std::unique_ptr<ast::BlockStatement> expect_body_stmt();
|
||||||
/// Parses a `paren_rhs_stmt` grammar element
|
/// Parses a `paren_rhs_stmt` grammar element, erroring on parse failure.
|
||||||
/// @returns the parsed element or nullptr
|
/// @returns the parsed element or nullptr
|
||||||
std::unique_ptr<ast::Expression> paren_rhs_stmt();
|
std::unique_ptr<ast::Expression> expect_paren_rhs_stmt();
|
||||||
/// Parses a `statements` grammar element
|
/// Parses a `statements` grammar element
|
||||||
/// @returns the statements parsed
|
/// @returns the statements parsed
|
||||||
std::unique_ptr<ast::BlockStatement> statements();
|
std::unique_ptr<ast::BlockStatement> statements();
|
||||||
|
@ -297,9 +299,9 @@ class ParserImpl {
|
||||||
/// Parses a `loop_stmt` grammar element
|
/// Parses a `loop_stmt` grammar element
|
||||||
/// @returns the parsed loop or nullptr
|
/// @returns the parsed loop or nullptr
|
||||||
std::unique_ptr<ast::LoopStatement> loop_stmt();
|
std::unique_ptr<ast::LoopStatement> loop_stmt();
|
||||||
/// Parses a `for_header` grammar element
|
/// Parses a `for_header` grammar element, erroring on parse failure.
|
||||||
/// @returns the parsed for header or nullptr
|
/// @returns the parsed for header or nullptr
|
||||||
std::unique_ptr<ForHeader> for_header();
|
std::unique_ptr<ForHeader> expect_for_header();
|
||||||
/// Parses a `for_stmt` grammar element
|
/// Parses a `for_stmt` grammar element
|
||||||
/// @returns the parsed for loop or nullptr
|
/// @returns the parsed for loop or nullptr
|
||||||
std::unique_ptr<ast::Statement> for_stmt();
|
std::unique_ptr<ast::Statement> for_stmt();
|
||||||
|
@ -309,15 +311,16 @@ class ParserImpl {
|
||||||
/// Parses a `const_literal` grammar element
|
/// Parses a `const_literal` grammar element
|
||||||
/// @returns the const literal parsed or nullptr if none found
|
/// @returns the const literal parsed or nullptr if none found
|
||||||
std::unique_ptr<ast::Literal> const_literal();
|
std::unique_ptr<ast::Literal> const_literal();
|
||||||
/// Parses a `const_expr` grammar element
|
/// Parses a `const_expr` grammar element, erroring on parse failure.
|
||||||
/// @returns the parsed constructor expression or nullptr on error
|
/// @returns the parsed constructor expression or nullptr on error
|
||||||
std::unique_ptr<ast::ConstructorExpression> const_expr();
|
std::unique_ptr<ast::ConstructorExpression> expect_const_expr();
|
||||||
/// Parses a `primary_expression` grammar element
|
/// Parses a `primary_expression` grammar element
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> primary_expression();
|
std::unique_ptr<ast::Expression> primary_expression();
|
||||||
/// Parses a `argument_expression_list` grammar element
|
/// Parses a `argument_expression_list` grammar element, erroring on parse
|
||||||
|
/// failure.
|
||||||
/// @returns the list of arguments
|
/// @returns the list of arguments
|
||||||
ast::ExpressionList argument_expression_list();
|
ast::ExpressionList expect_argument_expression_list();
|
||||||
/// Parses the recursive portion of the postfix_expression
|
/// Parses the recursive portion of the postfix_expression
|
||||||
/// @param prefix the left side of the expression
|
/// @param prefix the left side of the expression
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
|
@ -329,82 +332,92 @@ class ParserImpl {
|
||||||
/// Parses a `unary_expression` grammar element
|
/// Parses a `unary_expression` grammar element
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> unary_expression();
|
std::unique_ptr<ast::Expression> unary_expression();
|
||||||
/// Parses the recursive part of the `multiplicative_expression`
|
/// Parses the recursive part of the `multiplicative_expression`, erroring on
|
||||||
|
/// parse failure.
|
||||||
/// @param lhs the left side of the expression
|
/// @param lhs the left side of the expression
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> multiplicative_expr(
|
std::unique_ptr<ast::Expression> expect_multiplicative_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs);
|
std::unique_ptr<ast::Expression> lhs);
|
||||||
/// Parses the `multiplicative_expression` grammar element
|
/// Parses the `multiplicative_expression` grammar element
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> multiplicative_expression();
|
std::unique_ptr<ast::Expression> multiplicative_expression();
|
||||||
/// Parses the recursive part of the `additive_expression`
|
/// Parses the recursive part of the `additive_expression`, erroring on parse
|
||||||
|
/// failure.
|
||||||
/// @param lhs the left side of the expression
|
/// @param lhs the left side of the expression
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> additive_expr(
|
std::unique_ptr<ast::Expression> expect_additive_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs);
|
std::unique_ptr<ast::Expression> lhs);
|
||||||
/// Parses the `additive_expression` grammar element
|
/// Parses the `additive_expression` grammar element
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> additive_expression();
|
std::unique_ptr<ast::Expression> additive_expression();
|
||||||
/// Parses the recursive part of the `shift_expression`
|
/// Parses the recursive part of the `shift_expression`, erroring on parse
|
||||||
|
/// failure.
|
||||||
/// @param lhs the left side of the expression
|
/// @param lhs the left side of the expression
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> shift_expr(
|
std::unique_ptr<ast::Expression> expect_shift_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs);
|
std::unique_ptr<ast::Expression> lhs);
|
||||||
/// Parses the `shift_expression` grammar element
|
/// Parses the `shift_expression` grammar element
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> shift_expression();
|
std::unique_ptr<ast::Expression> shift_expression();
|
||||||
/// Parses the recursive part of the `relational_expression`
|
/// Parses the recursive part of the `relational_expression`, erroring on
|
||||||
|
/// parse failure.
|
||||||
/// @param lhs the left side of the expression
|
/// @param lhs the left side of the expression
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> relational_expr(
|
std::unique_ptr<ast::Expression> expect_relational_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs);
|
std::unique_ptr<ast::Expression> lhs);
|
||||||
/// Parses the `relational_expression` grammar element
|
/// Parses the `relational_expression` grammar element
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> relational_expression();
|
std::unique_ptr<ast::Expression> relational_expression();
|
||||||
/// Parses the recursive part of the `equality_expression`
|
/// Parses the recursive part of the `equality_expression`, erroring on parse
|
||||||
|
/// failure.
|
||||||
/// @param lhs the left side of the expression
|
/// @param lhs the left side of the expression
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> equality_expr(
|
std::unique_ptr<ast::Expression> expect_equality_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs);
|
std::unique_ptr<ast::Expression> lhs);
|
||||||
/// Parses the `equality_expression` grammar element
|
/// Parses the `equality_expression` grammar element
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> equality_expression();
|
std::unique_ptr<ast::Expression> equality_expression();
|
||||||
/// Parses the recursive part of the `and_expression`
|
/// Parses the recursive part of the `and_expression`, erroring on parse
|
||||||
|
/// failure.
|
||||||
/// @param lhs the left side of the expression
|
/// @param lhs the left side of the expression
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> and_expr(
|
std::unique_ptr<ast::Expression> expect_and_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs);
|
std::unique_ptr<ast::Expression> lhs);
|
||||||
/// Parses the `and_expression` grammar element
|
/// Parses the `and_expression` grammar element
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> and_expression();
|
std::unique_ptr<ast::Expression> and_expression();
|
||||||
/// Parses the recursive part of the `exclusive_or_expression`
|
/// Parses the recursive part of the `exclusive_or_expression`, erroring on
|
||||||
|
/// parse failure.
|
||||||
/// @param lhs the left side of the expression
|
/// @param lhs the left side of the expression
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> exclusive_or_expr(
|
std::unique_ptr<ast::Expression> expect_exclusive_or_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs);
|
std::unique_ptr<ast::Expression> lhs);
|
||||||
/// Parses the `exclusive_or_expression` grammar elememnt
|
/// Parses the `exclusive_or_expression` grammar elememnt
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> exclusive_or_expression();
|
std::unique_ptr<ast::Expression> exclusive_or_expression();
|
||||||
/// Parses the recursive part of the `inclusive_or_expression`
|
/// Parses the recursive part of the `inclusive_or_expression`, erroring on
|
||||||
|
/// parse failure.
|
||||||
/// @param lhs the left side of the expression
|
/// @param lhs the left side of the expression
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> inclusive_or_expr(
|
std::unique_ptr<ast::Expression> expect_inclusive_or_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs);
|
std::unique_ptr<ast::Expression> lhs);
|
||||||
/// Parses the `inclusive_or_expression` grammar element
|
/// Parses the `inclusive_or_expression` grammar element
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> inclusive_or_expression();
|
std::unique_ptr<ast::Expression> inclusive_or_expression();
|
||||||
/// Parses the recursive part of the `logical_and_expression`
|
/// Parses the recursive part of the `logical_and_expression`, erroring on
|
||||||
|
/// parse failure.
|
||||||
/// @param lhs the left side of the expression
|
/// @param lhs the left side of the expression
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> logical_and_expr(
|
std::unique_ptr<ast::Expression> expect_logical_and_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs);
|
std::unique_ptr<ast::Expression> lhs);
|
||||||
/// Parses a `logical_and_expression` grammar element
|
/// Parses a `logical_and_expression` grammar element
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> logical_and_expression();
|
std::unique_ptr<ast::Expression> logical_and_expression();
|
||||||
/// Parses the recursive part of the `logical_or_expression`
|
/// Parses the recursive part of the `logical_or_expression`, erroring on
|
||||||
|
/// parse failure.
|
||||||
/// @param lhs the left side of the expression
|
/// @param lhs the left side of the expression
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
std::unique_ptr<ast::Expression> logical_or_expr(
|
std::unique_ptr<ast::Expression> expect_logical_or_expr(
|
||||||
std::unique_ptr<ast::Expression> lhs);
|
std::unique_ptr<ast::Expression> lhs);
|
||||||
/// Parses a `logical_or_expression` grammar element
|
/// Parses a `logical_or_expression` grammar element
|
||||||
/// @returns the parsed expression or nullptr
|
/// @returns the parsed expression or nullptr
|
||||||
|
@ -528,12 +541,12 @@ class ParserImpl {
|
||||||
/// Used to ensure that all decorations are consumed.
|
/// Used to ensure that all decorations are consumed.
|
||||||
bool expect_decorations_consumed(const ast::DecorationList& list);
|
bool expect_decorations_consumed(const ast::DecorationList& list);
|
||||||
|
|
||||||
ast::type::Type* type_decl_pointer(Token t);
|
ast::type::Type* expect_type_decl_pointer(Token t);
|
||||||
ast::type::Type* type_decl_vector(Token t);
|
ast::type::Type* expect_type_decl_vector(Token t);
|
||||||
ast::type::Type* type_decl_array(ast::ArrayDecorationList decos);
|
ast::type::Type* expect_type_decl_array(ast::ArrayDecorationList decos);
|
||||||
ast::type::Type* type_decl_matrix(Token t);
|
ast::type::Type* expect_type_decl_matrix(Token t);
|
||||||
|
|
||||||
std::unique_ptr<ast::ConstructorExpression> const_expr_internal(
|
std::unique_ptr<ast::ConstructorExpression> expect_const_expr_internal(
|
||||||
uint32_t depth);
|
uint32_t depth);
|
||||||
|
|
||||||
Context& ctx_;
|
Context& ctx_;
|
||||||
|
|
|
@ -28,7 +28,7 @@ namespace {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ArgumentExpressionList_Parses) {
|
TEST_F(ParserImplTest, ArgumentExpressionList_Parses) {
|
||||||
auto* p = parser("a");
|
auto* p = parser("a");
|
||||||
auto e = p->argument_expression_list();
|
auto e = p->expect_argument_expression_list();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
|
|
||||||
ASSERT_EQ(e.size(), 1u);
|
ASSERT_EQ(e.size(), 1u);
|
||||||
|
@ -37,7 +37,7 @@ TEST_F(ParserImplTest, ArgumentExpressionList_Parses) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ArgumentExpressionList_ParsesMultiple) {
|
TEST_F(ParserImplTest, ArgumentExpressionList_ParsesMultiple) {
|
||||||
auto* p = parser("a, -33, 1+2");
|
auto* p = parser("a, -33, 1+2");
|
||||||
auto e = p->argument_expression_list();
|
auto e = p->expect_argument_expression_list();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
|
|
||||||
ASSERT_EQ(e.size(), 3u);
|
ASSERT_EQ(e.size(), 3u);
|
||||||
|
@ -48,14 +48,14 @@ TEST_F(ParserImplTest, ArgumentExpressionList_ParsesMultiple) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ArgumentExpressionList_HandlesMissingExpression) {
|
TEST_F(ParserImplTest, ArgumentExpressionList_HandlesMissingExpression) {
|
||||||
auto* p = parser("a, ");
|
auto* p = parser("a, ");
|
||||||
auto e = p->argument_expression_list();
|
auto e = p->expect_argument_expression_list();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(), "1:4: unable to parse argument expression after comma");
|
EXPECT_EQ(p->error(), "1:4: unable to parse argument expression after comma");
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ArgumentExpressionList_HandlesInvalidExpression) {
|
TEST_F(ParserImplTest, ArgumentExpressionList_HandlesInvalidExpression) {
|
||||||
auto* p = parser("if(a) {}");
|
auto* p = parser("if(a) {}");
|
||||||
auto e = p->argument_expression_list();
|
auto e = p->expect_argument_expression_list();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(), "1:1: unable to parse argument expression");
|
EXPECT_EQ(p->error(), "1:1: unable to parse argument expression");
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,7 @@ TEST_F(ParserImplTest, BodyStmt) {
|
||||||
discard;
|
discard;
|
||||||
return 1 + b / 2;
|
return 1 + b / 2;
|
||||||
})");
|
})");
|
||||||
auto e = p->body_stmt();
|
auto e = p->expect_body_stmt();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
ASSERT_EQ(e->size(), 2u);
|
ASSERT_EQ(e->size(), 2u);
|
||||||
EXPECT_TRUE(e->get(0)->IsDiscard());
|
EXPECT_TRUE(e->get(0)->IsDiscard());
|
||||||
|
@ -35,21 +35,21 @@ TEST_F(ParserImplTest, BodyStmt) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, BodyStmt_Empty) {
|
TEST_F(ParserImplTest, BodyStmt_Empty) {
|
||||||
auto* p = parser("{}");
|
auto* p = parser("{}");
|
||||||
auto e = p->body_stmt();
|
auto e = p->expect_body_stmt();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
EXPECT_EQ(e->size(), 0u);
|
EXPECT_EQ(e->size(), 0u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(ParserImplTest, BodyStmt_InvalidStmt) {
|
TEST_F(ParserImplTest, BodyStmt_InvalidStmt) {
|
||||||
auto* p = parser("{fn main() -> void {}}");
|
auto* p = parser("{fn main() -> void {}}");
|
||||||
auto e = p->body_stmt();
|
auto e = p->expect_body_stmt();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(), "1:2: expected '}'");
|
EXPECT_EQ(p->error(), "1:2: expected '}'");
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(ParserImplTest, BodyStmt_MissingRightParen) {
|
TEST_F(ParserImplTest, BodyStmt_MissingRightParen) {
|
||||||
auto* p = parser("{return;");
|
auto* p = parser("{return;");
|
||||||
auto e = p->body_stmt();
|
auto e = p->expect_body_stmt();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(), "1:9: expected '}'");
|
EXPECT_EQ(p->error(), "1:9: expected '}'");
|
||||||
}
|
}
|
||||||
|
|
|
@ -28,7 +28,7 @@ namespace {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ConstExpr_TypeDecl) {
|
TEST_F(ParserImplTest, ConstExpr_TypeDecl) {
|
||||||
auto* p = parser("vec2<f32>(1., 2.)");
|
auto* p = parser("vec2<f32>(1., 2.)");
|
||||||
auto e = p->const_expr();
|
auto e = p->expect_const_expr();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
ASSERT_NE(e, nullptr);
|
ASSERT_NE(e, nullptr);
|
||||||
ASSERT_TRUE(e->IsConstructor());
|
ASSERT_TRUE(e->IsConstructor());
|
||||||
|
@ -56,7 +56,7 @@ TEST_F(ParserImplTest, ConstExpr_TypeDecl) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ConstExpr_TypeDecl_MissingRightParen) {
|
TEST_F(ParserImplTest, ConstExpr_TypeDecl_MissingRightParen) {
|
||||||
auto* p = parser("vec2<f32>(1., 2.");
|
auto* p = parser("vec2<f32>(1., 2.");
|
||||||
auto e = p->const_expr();
|
auto e = p->expect_const_expr();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
ASSERT_EQ(e, nullptr);
|
ASSERT_EQ(e, nullptr);
|
||||||
EXPECT_EQ(p->error(), "1:17: expected ')' for type constructor");
|
EXPECT_EQ(p->error(), "1:17: expected ')' for type constructor");
|
||||||
|
@ -64,7 +64,7 @@ TEST_F(ParserImplTest, ConstExpr_TypeDecl_MissingRightParen) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ConstExpr_TypeDecl_MissingLeftParen) {
|
TEST_F(ParserImplTest, ConstExpr_TypeDecl_MissingLeftParen) {
|
||||||
auto* p = parser("vec2<f32> 1., 2.)");
|
auto* p = parser("vec2<f32> 1., 2.)");
|
||||||
auto e = p->const_expr();
|
auto e = p->expect_const_expr();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
ASSERT_EQ(e, nullptr);
|
ASSERT_EQ(e, nullptr);
|
||||||
EXPECT_EQ(p->error(), "1:11: expected '(' for type constructor");
|
EXPECT_EQ(p->error(), "1:11: expected '(' for type constructor");
|
||||||
|
@ -72,7 +72,7 @@ TEST_F(ParserImplTest, ConstExpr_TypeDecl_MissingLeftParen) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ConstExpr_TypeDecl_HangingComma) {
|
TEST_F(ParserImplTest, ConstExpr_TypeDecl_HangingComma) {
|
||||||
auto* p = parser("vec2<f32>(1.,)");
|
auto* p = parser("vec2<f32>(1.,)");
|
||||||
auto e = p->const_expr();
|
auto e = p->expect_const_expr();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
ASSERT_EQ(e, nullptr);
|
ASSERT_EQ(e, nullptr);
|
||||||
EXPECT_EQ(p->error(), "1:14: unable to parse const literal");
|
EXPECT_EQ(p->error(), "1:14: unable to parse const literal");
|
||||||
|
@ -80,7 +80,7 @@ TEST_F(ParserImplTest, ConstExpr_TypeDecl_HangingComma) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ConstExpr_TypeDecl_MissingComma) {
|
TEST_F(ParserImplTest, ConstExpr_TypeDecl_MissingComma) {
|
||||||
auto* p = parser("vec2<f32>(1. 2.");
|
auto* p = parser("vec2<f32>(1. 2.");
|
||||||
auto e = p->const_expr();
|
auto e = p->expect_const_expr();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
ASSERT_EQ(e, nullptr);
|
ASSERT_EQ(e, nullptr);
|
||||||
EXPECT_EQ(p->error(), "1:14: expected ')' for type constructor");
|
EXPECT_EQ(p->error(), "1:14: expected ')' for type constructor");
|
||||||
|
@ -88,7 +88,7 @@ TEST_F(ParserImplTest, ConstExpr_TypeDecl_MissingComma) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ConstExpr_MissingExpr) {
|
TEST_F(ParserImplTest, ConstExpr_MissingExpr) {
|
||||||
auto* p = parser("vec2<f32>()");
|
auto* p = parser("vec2<f32>()");
|
||||||
auto e = p->const_expr();
|
auto e = p->expect_const_expr();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
ASSERT_EQ(e, nullptr);
|
ASSERT_EQ(e, nullptr);
|
||||||
EXPECT_EQ(p->error(), "1:11: unable to parse const literal");
|
EXPECT_EQ(p->error(), "1:11: unable to parse const literal");
|
||||||
|
@ -96,7 +96,7 @@ TEST_F(ParserImplTest, ConstExpr_MissingExpr) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ConstExpr_InvalidExpr) {
|
TEST_F(ParserImplTest, ConstExpr_InvalidExpr) {
|
||||||
auto* p = parser("vec2<f32>(1., if(a) {})");
|
auto* p = parser("vec2<f32>(1., if(a) {})");
|
||||||
auto e = p->const_expr();
|
auto e = p->expect_const_expr();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
ASSERT_EQ(e, nullptr);
|
ASSERT_EQ(e, nullptr);
|
||||||
EXPECT_EQ(p->error(), "1:15: unable to parse const literal");
|
EXPECT_EQ(p->error(), "1:15: unable to parse const literal");
|
||||||
|
@ -104,7 +104,7 @@ TEST_F(ParserImplTest, ConstExpr_InvalidExpr) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ConstExpr_ConstLiteral) {
|
TEST_F(ParserImplTest, ConstExpr_ConstLiteral) {
|
||||||
auto* p = parser("true");
|
auto* p = parser("true");
|
||||||
auto e = p->const_expr();
|
auto e = p->expect_const_expr();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
ASSERT_NE(e, nullptr);
|
ASSERT_NE(e, nullptr);
|
||||||
ASSERT_TRUE(e->IsConstructor());
|
ASSERT_TRUE(e->IsConstructor());
|
||||||
|
@ -116,7 +116,7 @@ TEST_F(ParserImplTest, ConstExpr_ConstLiteral) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ConstExpr_ConstLiteral_Invalid) {
|
TEST_F(ParserImplTest, ConstExpr_ConstLiteral_Invalid) {
|
||||||
auto* p = parser("invalid");
|
auto* p = parser("invalid");
|
||||||
auto e = p->const_expr();
|
auto e = p->expect_const_expr();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
ASSERT_EQ(e, nullptr);
|
ASSERT_EQ(e, nullptr);
|
||||||
EXPECT_EQ(p->error(), "1:1: unknown constructed type 'invalid'");
|
EXPECT_EQ(p->error(), "1:1: unknown constructed type 'invalid'");
|
||||||
|
@ -132,7 +132,7 @@ TEST_F(ParserImplTest, ConstExpr_Recursion) {
|
||||||
out << ")";
|
out << ")";
|
||||||
}
|
}
|
||||||
auto* p = parser(out.str());
|
auto* p = parser(out.str());
|
||||||
auto e = p->const_expr();
|
auto e = p->expect_const_expr();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
ASSERT_EQ(e, nullptr);
|
ASSERT_EQ(e, nullptr);
|
||||||
EXPECT_EQ(p->error(), "1:517: max const_expr depth reached");
|
EXPECT_EQ(p->error(), "1:517: max const_expr depth reached");
|
||||||
|
|
|
@ -25,13 +25,13 @@ namespace {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_Semicolon) {
|
TEST_F(ParserImplTest, GlobalDecl_Semicolon) {
|
||||||
auto* p = parser(";");
|
auto* p = parser(";");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_GlobalVariable) {
|
TEST_F(ParserImplTest, GlobalDecl_GlobalVariable) {
|
||||||
auto* p = parser("var<out> a : vec2<i32> = vec2<i32>(1, 2);");
|
auto* p = parser("var<out> a : vec2<i32> = vec2<i32>(1, 2);");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
|
|
||||||
auto m = p->module();
|
auto m = p->module();
|
||||||
|
@ -43,21 +43,21 @@ TEST_F(ParserImplTest, GlobalDecl_GlobalVariable) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_GlobalVariable_Invalid) {
|
TEST_F(ParserImplTest, GlobalDecl_GlobalVariable_Invalid) {
|
||||||
auto* p = parser("var<out> a : vec2<invalid>;");
|
auto* p = parser("var<out> a : vec2<invalid>;");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(), "1:19: unknown constructed type 'invalid'");
|
EXPECT_EQ(p->error(), "1:19: unknown constructed type 'invalid'");
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_GlobalVariable_MissingSemicolon) {
|
TEST_F(ParserImplTest, GlobalDecl_GlobalVariable_MissingSemicolon) {
|
||||||
auto* p = parser("var<out> a : vec2<i32>");
|
auto* p = parser("var<out> a : vec2<i32>");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(), "1:23: expected ';' for variable declaration");
|
EXPECT_EQ(p->error(), "1:23: expected ';' for variable declaration");
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_GlobalConstant) {
|
TEST_F(ParserImplTest, GlobalDecl_GlobalConstant) {
|
||||||
auto* p = parser("const a : i32 = 2;");
|
auto* p = parser("const a : i32 = 2;");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
|
|
||||||
auto m = p->module();
|
auto m = p->module();
|
||||||
|
@ -69,21 +69,21 @@ TEST_F(ParserImplTest, GlobalDecl_GlobalConstant) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_GlobalConstant_Invalid) {
|
TEST_F(ParserImplTest, GlobalDecl_GlobalConstant_Invalid) {
|
||||||
auto* p = parser("const a : vec2<i32>;");
|
auto* p = parser("const a : vec2<i32>;");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(), "1:20: missing = for const declaration");
|
EXPECT_EQ(p->error(), "1:20: missing = for const declaration");
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_GlobalConstant_MissingSemicolon) {
|
TEST_F(ParserImplTest, GlobalDecl_GlobalConstant_MissingSemicolon) {
|
||||||
auto* p = parser("const a : vec2<i32> = vec2<i32>(1, 2)");
|
auto* p = parser("const a : vec2<i32> = vec2<i32>(1, 2)");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(), "1:38: expected ';' for constant declaration");
|
EXPECT_EQ(p->error(), "1:38: expected ';' for constant declaration");
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_TypeAlias) {
|
TEST_F(ParserImplTest, GlobalDecl_TypeAlias) {
|
||||||
auto* p = parser("type A = i32;");
|
auto* p = parser("type A = i32;");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
|
|
||||||
auto m = p->module();
|
auto m = p->module();
|
||||||
|
@ -97,8 +97,8 @@ TEST_F(ParserImplTest, GlobalDecl_TypeAlias_StructIdent) {
|
||||||
a : f32;
|
a : f32;
|
||||||
};
|
};
|
||||||
type B = A;)");
|
type B = A;)");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
|
|
||||||
auto m = p->module();
|
auto m = p->module();
|
||||||
|
@ -115,21 +115,21 @@ type B = A;)");
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_TypeAlias_Invalid) {
|
TEST_F(ParserImplTest, GlobalDecl_TypeAlias_Invalid) {
|
||||||
auto* p = parser("type A = invalid;");
|
auto* p = parser("type A = invalid;");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(), "1:10: unknown constructed type 'invalid'");
|
EXPECT_EQ(p->error(), "1:10: unknown constructed type 'invalid'");
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_TypeAlias_MissingSemicolon) {
|
TEST_F(ParserImplTest, GlobalDecl_TypeAlias_MissingSemicolon) {
|
||||||
auto* p = parser("type A = i32");
|
auto* p = parser("type A = i32");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(), "1:13: expected ';' for type alias");
|
EXPECT_EQ(p->error(), "1:13: expected ';' for type alias");
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_Function) {
|
TEST_F(ParserImplTest, GlobalDecl_Function) {
|
||||||
auto* p = parser("fn main() -> void { return; }");
|
auto* p = parser("fn main() -> void { return; }");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
|
|
||||||
auto m = p->module();
|
auto m = p->module();
|
||||||
|
@ -139,7 +139,7 @@ TEST_F(ParserImplTest, GlobalDecl_Function) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_Function_WithDecoration) {
|
TEST_F(ParserImplTest, GlobalDecl_Function_WithDecoration) {
|
||||||
auto* p = parser("[[workgroup_size(2)]] fn main() -> void { return; }");
|
auto* p = parser("[[workgroup_size(2)]] fn main() -> void { return; }");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
|
|
||||||
auto m = p->module();
|
auto m = p->module();
|
||||||
|
@ -149,14 +149,14 @@ TEST_F(ParserImplTest, GlobalDecl_Function_WithDecoration) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_Function_Invalid) {
|
TEST_F(ParserImplTest, GlobalDecl_Function_Invalid) {
|
||||||
auto* p = parser("fn main() -> { return; }");
|
auto* p = parser("fn main() -> { return; }");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(), "1:14: unable to determine function return type");
|
EXPECT_EQ(p->error(), "1:14: unable to determine function return type");
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_ParsesStruct) {
|
TEST_F(ParserImplTest, GlobalDecl_ParsesStruct) {
|
||||||
auto* p = parser("struct A { b: i32; c: f32;};");
|
auto* p = parser("struct A { b: i32; c: f32;};");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
|
|
||||||
auto m = p->module();
|
auto m = p->module();
|
||||||
|
@ -174,7 +174,7 @@ TEST_F(ParserImplTest, GlobalDecl_ParsesStruct) {
|
||||||
TEST_F(ParserImplTest, GlobalDecl_Struct_WithStride) {
|
TEST_F(ParserImplTest, GlobalDecl_Struct_WithStride) {
|
||||||
auto* p =
|
auto* p =
|
||||||
parser("struct A { [[offset(0)]] data: [[stride(4)]] array<f32>; };");
|
parser("struct A { [[offset(0)]] data: [[stride(4)]] array<f32>; };");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
|
|
||||||
auto m = p->module();
|
auto m = p->module();
|
||||||
|
@ -198,7 +198,7 @@ TEST_F(ParserImplTest, GlobalDecl_Struct_WithStride) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_Struct_WithDecoration) {
|
TEST_F(ParserImplTest, GlobalDecl_Struct_WithDecoration) {
|
||||||
auto* p = parser("[[block]] struct A { [[offset(0)]] data: f32; };");
|
auto* p = parser("[[block]] struct A { [[offset(0)]] data: f32; };");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
|
|
||||||
auto m = p->module();
|
auto m = p->module();
|
||||||
|
@ -216,14 +216,14 @@ TEST_F(ParserImplTest, GlobalDecl_Struct_WithDecoration) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_Struct_Invalid) {
|
TEST_F(ParserImplTest, GlobalDecl_Struct_Invalid) {
|
||||||
auto* p = parser("[[block]] A {};");
|
auto* p = parser("[[block]] A {};");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(), "1:11: expected declaration after decorations");
|
EXPECT_EQ(p->error(), "1:11: expected declaration after decorations");
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(ParserImplTest, GlobalDecl_StructMissing_Semi) {
|
TEST_F(ParserImplTest, GlobalDecl_StructMissing_Semi) {
|
||||||
auto* p = parser("[[block]] struct A {}");
|
auto* p = parser("[[block]] struct A {}");
|
||||||
p->global_decl();
|
p->expect_global_decl();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(), "1:22: expected ';' for struct declaration");
|
EXPECT_EQ(p->error(), "1:22: expected ';' for struct declaration");
|
||||||
}
|
}
|
||||||
|
|
|
@ -32,7 +32,7 @@ TEST_F(ParserImplTest, ParamList_Single) {
|
||||||
auto* i32 = tm()->Get(std::make_unique<ast::type::I32Type>());
|
auto* i32 = tm()->Get(std::make_unique<ast::type::I32Type>());
|
||||||
|
|
||||||
auto* p = parser("a : i32");
|
auto* p = parser("a : i32");
|
||||||
auto e = p->param_list();
|
auto e = p->expect_param_list();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
EXPECT_EQ(e.size(), 1u);
|
EXPECT_EQ(e.size(), 1u);
|
||||||
|
|
||||||
|
@ -52,7 +52,7 @@ TEST_F(ParserImplTest, ParamList_Multiple) {
|
||||||
auto* vec2 = tm()->Get(std::make_unique<ast::type::VectorType>(f32, 2));
|
auto* vec2 = tm()->Get(std::make_unique<ast::type::VectorType>(f32, 2));
|
||||||
|
|
||||||
auto* p = parser("a : i32, b: f32, c: vec2<f32>");
|
auto* p = parser("a : i32, b: f32, c: vec2<f32>");
|
||||||
auto e = p->param_list();
|
auto e = p->expect_param_list();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
EXPECT_EQ(e.size(), 3u);
|
EXPECT_EQ(e.size(), 3u);
|
||||||
|
|
||||||
|
@ -86,14 +86,14 @@ TEST_F(ParserImplTest, ParamList_Multiple) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ParamList_Empty) {
|
TEST_F(ParserImplTest, ParamList_Empty) {
|
||||||
auto* p = parser("");
|
auto* p = parser("");
|
||||||
auto e = p->param_list();
|
auto e = p->expect_param_list();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
EXPECT_EQ(e.size(), 0u);
|
EXPECT_EQ(e.size(), 0u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ParamList_HangingComma) {
|
TEST_F(ParserImplTest, ParamList_HangingComma) {
|
||||||
auto* p = parser("a : i32,");
|
auto* p = parser("a : i32,");
|
||||||
auto e = p->param_list();
|
auto e = p->expect_param_list();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(), "1:8: found , but no variable declaration");
|
EXPECT_EQ(p->error(), "1:8: found , but no variable declaration");
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,7 +23,7 @@ namespace {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ParenRhsStmt) {
|
TEST_F(ParserImplTest, ParenRhsStmt) {
|
||||||
auto* p = parser("(a + b)");
|
auto* p = parser("(a + b)");
|
||||||
auto e = p->paren_rhs_stmt();
|
auto e = p->expect_paren_rhs_stmt();
|
||||||
ASSERT_FALSE(p->has_error()) << p->error();
|
ASSERT_FALSE(p->has_error()) << p->error();
|
||||||
ASSERT_NE(e, nullptr);
|
ASSERT_NE(e, nullptr);
|
||||||
ASSERT_TRUE(e->IsBinary());
|
ASSERT_TRUE(e->IsBinary());
|
||||||
|
@ -31,7 +31,7 @@ TEST_F(ParserImplTest, ParenRhsStmt) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ParenRhsStmt_MissingLeftParen) {
|
TEST_F(ParserImplTest, ParenRhsStmt_MissingLeftParen) {
|
||||||
auto* p = parser("true)");
|
auto* p = parser("true)");
|
||||||
auto e = p->paren_rhs_stmt();
|
auto e = p->expect_paren_rhs_stmt();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
ASSERT_EQ(e, nullptr);
|
ASSERT_EQ(e, nullptr);
|
||||||
EXPECT_EQ(p->error(), "1:1: expected '('");
|
EXPECT_EQ(p->error(), "1:1: expected '('");
|
||||||
|
@ -39,7 +39,7 @@ TEST_F(ParserImplTest, ParenRhsStmt_MissingLeftParen) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ParenRhsStmt_MissingRightParen) {
|
TEST_F(ParserImplTest, ParenRhsStmt_MissingRightParen) {
|
||||||
auto* p = parser("(true");
|
auto* p = parser("(true");
|
||||||
auto e = p->paren_rhs_stmt();
|
auto e = p->expect_paren_rhs_stmt();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
ASSERT_EQ(e, nullptr);
|
ASSERT_EQ(e, nullptr);
|
||||||
EXPECT_EQ(p->error(), "1:6: expected ')'");
|
EXPECT_EQ(p->error(), "1:6: expected ')'");
|
||||||
|
@ -47,7 +47,7 @@ TEST_F(ParserImplTest, ParenRhsStmt_MissingRightParen) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ParenRhsStmt_InvalidExpression) {
|
TEST_F(ParserImplTest, ParenRhsStmt_InvalidExpression) {
|
||||||
auto* p = parser("(if (a() {})");
|
auto* p = parser("(if (a() {})");
|
||||||
auto e = p->paren_rhs_stmt();
|
auto e = p->expect_paren_rhs_stmt();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
ASSERT_EQ(e, nullptr);
|
ASSERT_EQ(e, nullptr);
|
||||||
EXPECT_EQ(p->error(), "1:2: unable to parse expression");
|
EXPECT_EQ(p->error(), "1:2: unable to parse expression");
|
||||||
|
@ -55,7 +55,7 @@ TEST_F(ParserImplTest, ParenRhsStmt_InvalidExpression) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, ParenRhsStmt_MissingExpression) {
|
TEST_F(ParserImplTest, ParenRhsStmt_MissingExpression) {
|
||||||
auto* p = parser("()");
|
auto* p = parser("()");
|
||||||
auto e = p->paren_rhs_stmt();
|
auto e = p->expect_paren_rhs_stmt();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
ASSERT_EQ(e, nullptr);
|
ASSERT_EQ(e, nullptr);
|
||||||
EXPECT_EQ(p->error(), "1:2: unable to parse expression");
|
EXPECT_EQ(p->error(), "1:2: unable to parse expression");
|
||||||
|
|
|
@ -27,7 +27,7 @@ TEST_F(ParserImplTest, StructBodyDecl_Parses) {
|
||||||
auto* i32 = tm()->Get(std::make_unique<ast::type::I32Type>());
|
auto* i32 = tm()->Get(std::make_unique<ast::type::I32Type>());
|
||||||
|
|
||||||
auto* p = parser("{a : i32;}");
|
auto* p = parser("{a : i32;}");
|
||||||
auto m = p->struct_body_decl();
|
auto m = p->expect_struct_body_decl();
|
||||||
ASSERT_FALSE(p->has_error());
|
ASSERT_FALSE(p->has_error());
|
||||||
ASSERT_EQ(m.size(), 1u);
|
ASSERT_EQ(m.size(), 1u);
|
||||||
|
|
||||||
|
@ -39,7 +39,7 @@ TEST_F(ParserImplTest, StructBodyDecl_Parses) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, StructBodyDecl_ParsesEmpty) {
|
TEST_F(ParserImplTest, StructBodyDecl_ParsesEmpty) {
|
||||||
auto* p = parser("{}");
|
auto* p = parser("{}");
|
||||||
auto m = p->struct_body_decl();
|
auto m = p->expect_struct_body_decl();
|
||||||
ASSERT_FALSE(p->has_error());
|
ASSERT_FALSE(p->has_error());
|
||||||
ASSERT_EQ(m.size(), 0u);
|
ASSERT_EQ(m.size(), 0u);
|
||||||
}
|
}
|
||||||
|
@ -49,7 +49,7 @@ TEST_F(ParserImplTest, StructBodyDecl_InvalidMember) {
|
||||||
{
|
{
|
||||||
[[offset(nan)]] a : i32;
|
[[offset(nan)]] a : i32;
|
||||||
})");
|
})");
|
||||||
auto m = p->struct_body_decl();
|
auto m = p->expect_struct_body_decl();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(),
|
EXPECT_EQ(p->error(),
|
||||||
"3:12: expected signed integer literal for offset decoration");
|
"3:12: expected signed integer literal for offset decoration");
|
||||||
|
@ -57,7 +57,7 @@ TEST_F(ParserImplTest, StructBodyDecl_InvalidMember) {
|
||||||
|
|
||||||
TEST_F(ParserImplTest, StructBodyDecl_MissingClosingBracket) {
|
TEST_F(ParserImplTest, StructBodyDecl_MissingClosingBracket) {
|
||||||
auto* p = parser("{a : i32;");
|
auto* p = parser("{a : i32;");
|
||||||
auto m = p->struct_body_decl();
|
auto m = p->expect_struct_body_decl();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(), "1:10: expected '}' for struct declaration");
|
EXPECT_EQ(p->error(), "1:10: expected '}' for struct declaration");
|
||||||
}
|
}
|
||||||
|
@ -68,7 +68,7 @@ TEST_F(ParserImplTest, StructBodyDecl_InvalidToken) {
|
||||||
a : i32;
|
a : i32;
|
||||||
1.23
|
1.23
|
||||||
} )");
|
} )");
|
||||||
auto m = p->struct_body_decl();
|
auto m = p->expect_struct_body_decl();
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
EXPECT_EQ(p->error(), "4:3: invalid identifier declaration");
|
EXPECT_EQ(p->error(), "4:3: invalid identifier declaration");
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,7 +30,7 @@ TEST_F(ParserImplTest, StructMember_Parses) {
|
||||||
auto* p = parser("a : i32;");
|
auto* p = parser("a : i32;");
|
||||||
auto decos = p->decoration_list();
|
auto decos = p->decoration_list();
|
||||||
EXPECT_EQ(decos.size(), 0u);
|
EXPECT_EQ(decos.size(), 0u);
|
||||||
auto m = p->struct_member(decos);
|
auto m = p->expect_struct_member(decos);
|
||||||
ASSERT_FALSE(p->has_error());
|
ASSERT_FALSE(p->has_error());
|
||||||
ASSERT_NE(m, nullptr);
|
ASSERT_NE(m, nullptr);
|
||||||
|
|
||||||
|
@ -50,7 +50,7 @@ TEST_F(ParserImplTest, StructMember_ParsesWithDecoration) {
|
||||||
auto* p = parser("[[offset(2)]] a : i32;");
|
auto* p = parser("[[offset(2)]] a : i32;");
|
||||||
auto decos = p->decoration_list();
|
auto decos = p->decoration_list();
|
||||||
EXPECT_EQ(decos.size(), 1u);
|
EXPECT_EQ(decos.size(), 1u);
|
||||||
auto m = p->struct_member(decos);
|
auto m = p->expect_struct_member(decos);
|
||||||
ASSERT_FALSE(p->has_error());
|
ASSERT_FALSE(p->has_error());
|
||||||
ASSERT_NE(m, nullptr);
|
ASSERT_NE(m, nullptr);
|
||||||
|
|
||||||
|
@ -73,7 +73,7 @@ TEST_F(ParserImplTest, StructMember_ParsesWithMultipleDecorations) {
|
||||||
[[offset(4)]] a : i32;)");
|
[[offset(4)]] a : i32;)");
|
||||||
auto decos = p->decoration_list();
|
auto decos = p->decoration_list();
|
||||||
EXPECT_EQ(decos.size(), 2u);
|
EXPECT_EQ(decos.size(), 2u);
|
||||||
auto m = p->struct_member(decos);
|
auto m = p->expect_struct_member(decos);
|
||||||
ASSERT_FALSE(p->has_error());
|
ASSERT_FALSE(p->has_error());
|
||||||
ASSERT_NE(m, nullptr);
|
ASSERT_NE(m, nullptr);
|
||||||
|
|
||||||
|
@ -94,7 +94,7 @@ TEST_F(ParserImplTest, StructMember_ParsesWithMultipleDecorations) {
|
||||||
TEST_F(ParserImplTest, StructMember_InvalidDecoration) {
|
TEST_F(ParserImplTest, StructMember_InvalidDecoration) {
|
||||||
auto* p = parser("[[offset(nan)]] a : i32;");
|
auto* p = parser("[[offset(nan)]] a : i32;");
|
||||||
auto decos = p->decoration_list();
|
auto decos = p->decoration_list();
|
||||||
auto m = p->struct_member(decos);
|
auto m = p->expect_struct_member(decos);
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
ASSERT_EQ(m, nullptr);
|
ASSERT_EQ(m, nullptr);
|
||||||
EXPECT_EQ(p->error(),
|
EXPECT_EQ(p->error(),
|
||||||
|
@ -104,7 +104,7 @@ TEST_F(ParserImplTest, StructMember_InvalidDecoration) {
|
||||||
TEST_F(ParserImplTest, StructMember_InvalidVariable) {
|
TEST_F(ParserImplTest, StructMember_InvalidVariable) {
|
||||||
auto* p = parser("[[offset(4)]] a : B;");
|
auto* p = parser("[[offset(4)]] a : B;");
|
||||||
auto decos = p->decoration_list();
|
auto decos = p->decoration_list();
|
||||||
auto m = p->struct_member(decos);
|
auto m = p->expect_struct_member(decos);
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
ASSERT_EQ(m, nullptr);
|
ASSERT_EQ(m, nullptr);
|
||||||
EXPECT_EQ(p->error(), "1:19: unknown constructed type 'B'");
|
EXPECT_EQ(p->error(), "1:19: unknown constructed type 'B'");
|
||||||
|
@ -113,7 +113,7 @@ TEST_F(ParserImplTest, StructMember_InvalidVariable) {
|
||||||
TEST_F(ParserImplTest, StructMember_MissingSemicolon) {
|
TEST_F(ParserImplTest, StructMember_MissingSemicolon) {
|
||||||
auto* p = parser("a : i32");
|
auto* p = parser("a : i32");
|
||||||
auto decos = p->decoration_list();
|
auto decos = p->decoration_list();
|
||||||
auto m = p->struct_member(decos);
|
auto m = p->expect_struct_member(decos);
|
||||||
ASSERT_TRUE(p->has_error());
|
ASSERT_TRUE(p->has_error());
|
||||||
ASSERT_EQ(m, nullptr);
|
ASSERT_EQ(m, nullptr);
|
||||||
EXPECT_EQ(p->error(), "1:8: expected ';' for struct member");
|
EXPECT_EQ(p->error(), "1:8: expected ';' for struct member");
|
||||||
|
|
Loading…
Reference in New Issue