diff options
| author | Josh Wolfe <thejoshwolfe@gmail.com> | 2017-03-31 08:41:00 -0700 |
|---|---|---|
| committer | Josh Wolfe <thejoshwolfe@gmail.com> | 2017-03-31 08:54:03 -0700 |
| commit | 36a015741db9fbf748fe9ab082f78c5d8663024c (patch) | |
| tree | 9ed34ffe3f8b4bd3d274bc21d5db6c1d5e336051 /src | |
| parent | d5a6cdb03f874ab53928a67330ebfab1e64fe111 (diff) | |
| download | zig-36a015741db9fbf748fe9ab082f78c5d8663024c.tar.gz zig-36a015741db9fbf748fe9ab082f78c5d8663024c.zip | |
clean up analysis of {blocks}
* Don't insert void statements all over the place. {} now stays as
{} instead of {{}}, and {;} becomes {} instead of {{};{}}.
* Ensure final statement is always the return value statement, or
the block is empty. This means {label:} becomes {label:{}}.
Diffstat (limited to 'src')
| -rw-r--r-- | src/all_types.hpp | 3 | ||||
| -rw-r--r-- | src/ir.cpp | 12 | ||||
| -rw-r--r-- | src/parser.cpp | 18 |
3 files changed, 23 insertions, 10 deletions
diff --git a/src/all_types.hpp b/src/all_types.hpp index 7cf410faa4..e682255602 100644 --- a/src/all_types.hpp +++ b/src/all_types.hpp @@ -403,6 +403,9 @@ struct AstNodeParamDecl { }; struct AstNodeBlock { + // the final statement is the returned expression. + // if there are no statements, the returned expression is void. + // the final statement is never a label. ZigList<AstNode *> statements; }; diff --git a/src/ir.cpp b/src/ir.cpp index 06f761c8b0..1fdb8f2e3b 100644 --- a/src/ir.cpp +++ b/src/ir.cpp @@ -3269,6 +3269,11 @@ static IrInstruction *ir_gen_block(IrBuilder *irb, Scope *parent_scope, AstNode fn_entry->def_scope = scope_block; } + if (block_node->data.block.statements.length == 0) { + // {} + return ir_mark_gen(ir_build_const_void(irb, child_scope, block_node)); + } + IrInstruction *return_value = nullptr; for (size_t i = 0; i < block_node->data.block.statements.length; i += 1) { AstNode *statement_node = block_node->data.block.statements.at(i); @@ -3292,7 +3297,8 @@ static IrInstruction *ir_gen_block(IrBuilder *irb, Scope *parent_scope, AstNode scope_block->label_table.put(label_name, label); } - if (!return_value || !instr_is_unreachable(return_value)) { + if (!(return_value && instr_is_unreachable(return_value))) { + // fall through into new labeled basic block IrInstruction *is_comptime = ir_mark_gen(ir_build_const_bool(irb, child_scope, statement_node, ir_should_inline(irb->exec, child_scope))); ir_mark_gen(ir_build_br(irb, child_scope, statement_node, label_block, is_comptime)); @@ -3321,8 +3327,8 @@ static IrInstruction *ir_gen_block(IrBuilder *irb, Scope *parent_scope, AstNode } } - if (!return_value) - return_value = ir_mark_gen(ir_build_const_void(irb, child_scope, block_node)); + // labels are never the last statement + assert(return_value != nullptr); if (!instr_is_unreachable(return_value)) ir_gen_defers_for_block(irb, child_scope, outer_block_scope, false, false); 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; |
