1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-05-28 07:15:07 +00:00

LibJS: Consume semicolon at the end of a statement

A bunch of code was relying on this not happenind, in particular the
parsing of "for" statements. Reorganized things so they work again.
This commit is contained in:
Andreas Kling 2020-03-23 19:08:32 +01:00
parent fbb9e1b715
commit df524203b2
3 changed files with 32 additions and 15 deletions

View file

@ -181,6 +181,7 @@ NonnullRefPtr<Program> Parser::parse_program()
NonnullRefPtr<Statement> Parser::parse_statement()
{
auto statement = [this]() -> NonnullRefPtr<Statement> {
switch (m_current_token.type()) {
case TokenType::Function:
return parse_function_node<FunctionDeclaration>();
@ -197,17 +198,16 @@ NonnullRefPtr<Statement> Parser::parse_statement()
case TokenType::If:
return parse_if_statement();
default:
if (match_expression()) {
auto statement = adopt(*new ExpressionStatement(parse_expression(0)));
if (match(TokenType::Semicolon))
consume();
return statement;
}
if (match_expression())
return adopt(*new ExpressionStatement(parse_expression(0)));
m_has_errors = true;
expected("statement (missing switch case)");
consume();
return create_ast_node<ErrorStatement>();
}
} }();
if (match(TokenType::Semicolon))
consume();
return statement;
}
NonnullRefPtr<Expression> Parser::parse_primary_expression()
@ -394,7 +394,7 @@ NonnullRefPtr<Expression> Parser::parse_secondary_expression(NonnullRefPtr<Expre
return create_ast_node<MemberExpression>(move(lhs), parse_expression(min_precedence, associativity));
case TokenType::BracketOpen: {
consume(TokenType::BracketOpen);
auto expression = create_ast_node<MemberExpression>(move(lhs), parse_expression(min_precedence, associativity), true);
auto expression = create_ast_node<MemberExpression>(move(lhs), parse_expression(0), true);
consume(TokenType::BracketClose);
return expression;
}
@ -539,12 +539,17 @@ NonnullRefPtr<ForStatement> Parser::parse_for_statement()
consume(TokenType::ParenOpen);
RefPtr<Statement> init;
RefPtr<ASTNode> init;
switch (m_current_token.type()) {
case TokenType::Semicolon:
break;
default:
init = parse_statement();
if (match_expression())
init = parse_expression(0);
else if (match_variable_declaration())
init = parse_variable_declaration();
else
ASSERT_NOT_REACHED();
break;
}
@ -582,6 +587,18 @@ bool Parser::match(TokenType type) const
return m_current_token.type() == type;
}
bool Parser::match_variable_declaration() const
{
switch (m_current_token.type()) {
case TokenType::Var:
case TokenType::Let:
case TokenType::Const:
return true;
default:
return false;
}
}
bool Parser::match_expression() const
{
auto type = m_current_token.type();