aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorJosh Wolfe <thejoshwolfe@gmail.com>2017-03-31 08:41:00 -0700
committerJosh Wolfe <thejoshwolfe@gmail.com>2017-03-31 08:54:03 -0700
commit36a015741db9fbf748fe9ab082f78c5d8663024c (patch)
tree9ed34ffe3f8b4bd3d274bc21d5db6c1d5e336051 /src
parentd5a6cdb03f874ab53928a67330ebfab1e64fe111 (diff)
downloadzig-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.hpp3
-rw-r--r--src/ir.cpp12
-rw-r--r--src/parser.cpp18
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;