@@ -1878,36 +1878,6 @@ static AstNode *ast_parse_switch_expr(ParseContext *pc, size_t *token_index, boo
1878
1878
}
1879
1879
}
1880
1880
1881
- static bool statement_has_block_body (AstNode *node) {
1882
- switch (node->type ) {
1883
- case NodeTypeIfBoolExpr:
1884
- if (node->data .if_bool_expr .else_node )
1885
- return statement_has_block_body (node->data .if_bool_expr .else_node );
1886
- return node->data .if_bool_expr .then_block ->type == NodeTypeBlock;
1887
- case NodeTypeIfVarExpr:
1888
- if (node->data .if_var_expr .else_node )
1889
- return statement_has_block_body (node->data .if_var_expr .else_node );
1890
- return node->data .if_var_expr .then_block ->type == NodeTypeBlock;
1891
- case NodeTypeTryExpr:
1892
- if (node->data .try_expr .else_node )
1893
- return statement_has_block_body (node->data .try_expr .else_node );
1894
- return node->data .try_expr .then_node ->type == NodeTypeBlock;
1895
- case NodeTypeWhileExpr:
1896
- return node->data .while_expr .body ->type == NodeTypeBlock;
1897
- case NodeTypeForExpr:
1898
- return node->data .for_expr .body ->type == NodeTypeBlock;
1899
- case NodeTypeSwitchExpr:
1900
- case NodeTypeBlock:
1901
- return true ;
1902
- case NodeTypeCompTime:
1903
- return node->data .comptime_expr .expr ->type == NodeTypeBlock;
1904
- case NodeTypeDefer:
1905
- return node->data .defer .expr ->type == NodeTypeBlock;
1906
- default :
1907
- return false ;
1908
- }
1909
- }
1910
-
1911
1881
/*
1912
1882
BlockExpression(body) = Block | IfExpression(body) | TryExpression(body) | WhileExpression(body) | ForExpression(body) | SwitchExpression | CompTimeExpression(body)
1913
1883
*/
@@ -2120,9 +2090,35 @@ static AstNode *ast_parse_label(ParseContext *pc, size_t *token_index, bool mand
2120
2090
return node;
2121
2091
}
2122
2092
2123
- static AstNode *ast_create_void_expr (ParseContext *pc, Token *token) {
2124
- AstNode *node = ast_create_node (pc, NodeTypeBlock, token);
2125
- return node;
2093
+ static bool statement_terminates_without_semicolon (AstNode *node) {
2094
+ switch (node->type ) {
2095
+ case NodeTypeIfBoolExpr:
2096
+ if (node->data .if_bool_expr .else_node )
2097
+ return statement_terminates_without_semicolon (node->data .if_bool_expr .else_node );
2098
+ return node->data .if_bool_expr .then_block ->type == NodeTypeBlock;
2099
+ case NodeTypeIfVarExpr:
2100
+ if (node->data .if_var_expr .else_node )
2101
+ return statement_terminates_without_semicolon (node->data .if_var_expr .else_node );
2102
+ return node->data .if_var_expr .then_block ->type == NodeTypeBlock;
2103
+ case NodeTypeTryExpr:
2104
+ if (node->data .try_expr .else_node )
2105
+ return statement_terminates_without_semicolon (node->data .try_expr .else_node );
2106
+ return node->data .try_expr .then_node ->type == NodeTypeBlock;
2107
+ case NodeTypeWhileExpr:
2108
+ return node->data .while_expr .body ->type == NodeTypeBlock;
2109
+ case NodeTypeForExpr:
2110
+ return node->data .for_expr .body ->type == NodeTypeBlock;
2111
+ case NodeTypeCompTime:
2112
+ return node->data .comptime_expr .expr ->type == NodeTypeBlock;
2113
+ case NodeTypeDefer:
2114
+ return node->data .defer .expr ->type == NodeTypeBlock;
2115
+ case NodeTypeSwitchExpr:
2116
+ case NodeTypeBlock:
2117
+ case NodeTypeLabel:
2118
+ return true ;
2119
+ default :
2120
+ return false ;
2121
+ }
2126
2122
}
2127
2123
2128
2124
/*
@@ -2145,56 +2141,36 @@ static AstNode *ast_parse_block(ParseContext *pc, size_t *token_index, bool mand
2145
2141
2146
2142
for (;;) {
2147
2143
AstNode *statement_node = ast_parse_label (pc, token_index, false );
2148
- bool need_implicit_final_void_statement = false ;
2144
+ if (!statement_node)
2145
+ statement_node = ast_parse_variable_declaration_expr (pc, token_index, false , VisibModPrivate);
2146
+ if (!statement_node)
2147
+ statement_node = ast_parse_defer_expr (pc, token_index);
2148
+ if (!statement_node)
2149
+ statement_node = ast_parse_block_expr (pc, token_index, false );
2150
+ if (!statement_node)
2151
+ statement_node = ast_parse_expression (pc, token_index, false );
2152
+
2149
2153
bool semicolon_expected = true ;
2150
2154
if (statement_node) {
2151
- // label
2152
- semicolon_expected = false ;
2153
- // if a label is the last thing in a block, add a void statement.
2154
- need_implicit_final_void_statement = true ;
2155
- } else {
2156
- statement_node = ast_parse_variable_declaration_expr (pc, token_index, false , VisibModPrivate);
2157
- if (!statement_node) {
2158
- statement_node = ast_parse_defer_expr (pc, token_index);
2159
- if (statement_node) {
2160
- // defer
2161
- if (statement_has_block_body (statement_node)) {
2162
- // don't let defer be the last statement in a block
2163
- need_implicit_final_void_statement = true ;
2164
- semicolon_expected = false ;
2165
- } else {
2166
- // defer without a block body requires a semicolon
2167
- Token *token = &pc->tokens ->at (*token_index);
2168
- ast_expect_token (pc, token, TokenIdSemicolon);
2169
- }
2170
- } else {
2171
- statement_node = ast_parse_block_expr (pc, token_index, false );
2172
- if (statement_node) {
2173
- // block expr
2174
- if (statement_has_block_body (statement_node))
2175
- semicolon_expected = false ;
2176
- } else {
2177
- statement_node = ast_parse_expression (pc, token_index, false );
2178
- if (!statement_node) {
2179
- // no statement.
2180
- // final semicolon means add a void statement.
2181
- need_implicit_final_void_statement = true ;
2182
- }
2183
- }
2155
+ node->data .block .statements .append (statement_node);
2156
+ if (statement_terminates_without_semicolon (statement_node)) {
2157
+ semicolon_expected = false ;
2158
+ } else {
2159
+ if (statement_node->type == NodeTypeDefer) {
2160
+ // defer without a block body requires a semicolon
2161
+ Token *token = &pc->tokens ->at (*token_index);
2162
+ ast_expect_token (pc, token, TokenIdSemicolon);
2184
2163
}
2185
2164
}
2186
2165
}
2187
- if (statement_node)
2188
- node->data .block .statements .append (statement_node);
2166
+
2167
+ node->data .block .last_statement_is_result_expression = statement_node && !(
2168
+ statement_node->type == NodeTypeLabel ||
2169
+ statement_node->type == NodeTypeDefer);
2189
2170
2190
2171
last_token = &pc->tokens ->at (*token_index);
2191
2172
if (last_token->id == TokenIdRBrace) {
2192
2173
*token_index += 1 ;
2193
-
2194
- if (node->data .block .statements .length > 0 && need_implicit_final_void_statement) {
2195
- node->data .block .statements .append (ast_create_void_expr (pc, last_token));
2196
- }
2197
-
2198
2174
return node;
2199
2175
} else if (!semicolon_expected) {
2200
2176
continue ;
0 commit comments