aboutsummaryrefslogtreecommitdiff
path: root/src/parser.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/parser.cpp')
-rw-r--r--src/parser.cpp18
1 files changed, 11 insertions, 7 deletions
diff --git a/src/parser.cpp b/src/parser.cpp
index 0da247c051..bbaee219b7 100644
--- a/src/parser.cpp
+++ b/src/parser.cpp
@@ -2094,16 +2094,14 @@ static AstNode *ast_parse_block(ParseContext *pc, size_t *token_index, bool mand
AstNode *node = ast_create_node(pc, NodeTypeBlock, last_token);
- // {} -> {void}
- // {;} -> {void;void}
- // {2} -> {2}
- // {2;} -> {2;void}
- // {;2} -> {void;2}
for (;;) {
AstNode *statement_node = ast_parse_label(pc, token_index, false);
+ bool need_implicit_final_void_statement = false;
bool semicolon_expected;
if (statement_node) {
semicolon_expected = false;
+ // if a label is the last thing in a block, add a void statement.
+ need_implicit_final_void_statement = true;
} else {
statement_node = ast_parse_variable_declaration_expr(pc, token_index, false, VisibModPrivate);
if (!statement_node) {
@@ -2117,17 +2115,23 @@ static AstNode *ast_parse_block(ParseContext *pc, size_t *token_index, bool mand
if (!statement_node) {
statement_node = ast_parse_non_block_expr(pc, token_index, false);
if (!statement_node) {
- statement_node = ast_create_void_expr(pc, last_token);
+ // final semicolon means add a void statement.
+ need_implicit_final_void_statement = true;
}
}
}
}
- node->data.block.statements.append(statement_node);
+ if (statement_node)
+ node->data.block.statements.append(statement_node);
last_token = &pc->tokens->at(*token_index);
if (last_token->id == TokenIdRBrace) {
*token_index += 1;
+ if (node->data.block.statements.length > 0 && need_implicit_final_void_statement) {
+ node->data.block.statements.append(ast_create_void_expr(pc, last_token));
+ }
+
return node;
} else if (!semicolon_expected) {
continue;