aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorgingerBill <bill@gingerbill.org>2023-09-30 15:04:17 +0100
committergingerBill <bill@gingerbill.org>2023-09-30 15:04:17 +0100
commit648b83d6ead67a291dc023f93a7262622991be2a (patch)
tree722495f871bb992e44e321d894879337dd03c20c /src
parent4e97b833126d9773ec23a0a0652beb19aab4aea4 (diff)
Add `or_break` and `or_continue` constructs
Diffstat (limited to 'src')
-rw-r--r--src/check_expr.cpp121
-rw-r--r--src/check_stmt.cpp5
-rw-r--r--src/checker.hpp1
-rw-r--r--src/llvm_backend.hpp2
-rw-r--r--src/llvm_backend_expr.cpp56
-rw-r--r--src/parser.cpp65
-rw-r--r--src/parser.hpp1
-rw-r--r--src/parser_pos.cpp6
-rw-r--r--src/tokenizer.cpp4
9 files changed, 252 insertions, 9 deletions
diff --git a/src/check_expr.cpp b/src/check_expr.cpp
index 968b6ec1e..1401793a5 100644
--- a/src/check_expr.cpp
+++ b/src/check_expr.cpp
@@ -3826,6 +3826,15 @@ gb_internal void update_untyped_expr_type(CheckerContext *c, Ast *e, Type *type,
update_untyped_expr_type(c, ore->expr, type, final);
case_end;
+ case_ast_node(obe, OrBranchExpr, e);
+ if (old->value.kind != ExactValue_Invalid) {
+ // See above note in UnaryExpr case
+ break;
+ }
+
+ update_untyped_expr_type(c, obe->expr, type, final);
+ case_end;
+
case_ast_node(oee, OrElseExpr, e);
if (old->value.kind != ExactValue_Invalid) {
// See above note in UnaryExpr case
@@ -8193,6 +8202,104 @@ gb_internal ExprKind check_or_return_expr(CheckerContext *c, Operand *o, Ast *no
return Expr_Expr;
}
+gb_internal ExprKind check_or_branch_expr(CheckerContext *c, Operand *o, Ast *node, Type *type_hint) {
+ ast_node(be, OrBranchExpr, node);
+
+ String name = be->token.string;
+ Operand x = {};
+ check_multi_expr_with_type_hint(c, &x, be->expr, type_hint);
+ if (x.mode == Addressing_Invalid) {
+ o->mode = Addressing_Value;
+ o->type = t_invalid;
+ o->expr = node;
+ return Expr_Expr;
+ }
+
+ Type *left_type = nullptr;
+ Type *right_type = nullptr;
+ check_or_return_split_types(c, &x, name, &left_type, &right_type);
+ add_type_and_value(c, be->expr, x.mode, x.type, x.value);
+
+ if (right_type == nullptr) {
+ check_or_else_expr_no_value_error(c, name, x, type_hint);
+ } else {
+ if (is_type_boolean(right_type) || type_has_nil(right_type)) {
+ // okay
+ } else {
+ gbString s = type_to_string(right_type);
+ error(node, "'%.*s' requires a boolean or nil-able type, got %s", s);
+ gb_string_free(s);
+ }
+ }
+
+ o->expr = node;
+ o->type = left_type;
+ if (left_type != nullptr) {
+ o->mode = Addressing_Value;
+ } else {
+ o->mode = Addressing_NoValue;
+ }
+
+ if (c->curr_proc_sig == nullptr) {
+ error(node, "'%.*s' can only be used within a procedure", LIT(name));
+ }
+
+ Ast *label = be->label;
+
+ switch (be->token.kind) {
+ case Token_or_break:
+ if ((c->stmt_flags & Stmt_BreakAllowed) == 0 && label == nullptr) {
+ error(be->token, "'%.*s' only allowed in non-inline loops or 'switch' statements", LIT(name));
+ }
+ break;
+ case Token_or_continue:
+ if ((c->stmt_flags & Stmt_ContinueAllowed) == 0 && label == nullptr) {
+ error(be->token, "'%.*s' only allowed in non-inline loops", LIT(name));
+ }
+ break;
+ }
+
+ if (label != nullptr) {
+ if (label->kind != Ast_Ident) {
+ error(label, "A branch statement's label name must be an identifier");
+ return Expr_Expr;
+ }
+ Ast *ident = label;
+ String name = ident->Ident.token.string;
+ Operand o = {};
+ Entity *e = check_ident(c, &o, ident, nullptr, nullptr, false);
+ if (e == nullptr) {
+ error(ident, "Undeclared label name: %.*s", LIT(name));
+ return Expr_Expr;
+ }
+ add_entity_use(c, ident, e);
+ if (e->kind != Entity_Label) {
+ error(ident, "'%.*s' is not a label", LIT(name));
+ return Expr_Expr;
+ }
+ Ast *parent = e->Label.parent;
+ GB_ASSERT(parent != nullptr);
+ switch (parent->kind) {
+ case Ast_BlockStmt:
+ case Ast_IfStmt:
+ case Ast_SwitchStmt:
+ if (be->token.kind != Token_or_break) {
+ error(label, "Label '%.*s' can only be used with 'or_break'", LIT(e->token.string));
+ }
+ break;
+ case Ast_RangeStmt:
+ case Ast_ForStmt:
+ if ((be->token.kind != Token_or_break) && (be->token.kind != Token_or_continue)) {
+ error(label, "Label '%.*s' can only be used with 'or_break' and 'or_continue'", LIT(e->token.string));
+ }
+ break;
+
+ }
+ }
+
+ return Expr_Expr;
+}
+
gb_internal void check_compound_literal_field_values(CheckerContext *c, Slice<Ast *> const &elems, Operand *o, Type *type, bool &is_constant) {
Type *bt = base_type(type);
@@ -9947,6 +10054,10 @@ gb_internal ExprKind check_expr_base_internal(CheckerContext *c, Operand *o, Ast
return check_or_return_expr(c, o, node, type_hint);
case_end;
+ case_ast_node(re, OrBranchExpr, node);
+ return check_or_branch_expr(c, o, node, type_hint);
+ case_end;
+
case_ast_node(cl, CompoundLit, node);
kind = check_compound_literal(c, o, node, type_hint);
case_end;
@@ -10513,6 +10624,16 @@ gb_internal gbString write_expr_to_string(gbString str, Ast *node, bool shorthan
str = gb_string_appendc(str, " or_return");
case_end;
+ case_ast_node(oe, OrBranchExpr, node);
+ str = write_expr_to_string(str, oe->expr, shorthand);
+ str = gb_string_append_rune(str, ' ');
+ str = string_append_token(str, oe->token);
+ if (oe->label) {
+ str = gb_string_append_rune(str, ' ');
+ str = write_expr_to_string(str, oe->label, shorthand);
+ }
+ case_end;
+
case_ast_node(pe, ParenExpr, node);
str = gb_string_append_rune(str, '(');
str = write_expr_to_string(str, pe->expr, shorthand);
diff --git a/src/check_stmt.cpp b/src/check_stmt.cpp
index 3f1b9611c..0fe44289c 100644
--- a/src/check_stmt.cpp
+++ b/src/check_stmt.cpp
@@ -102,8 +102,13 @@ gb_internal void check_stmt_list(CheckerContext *ctx, Slice<Ast *> const &stmts,
new_flags |= Stmt_FallthroughAllowed;
}
+ u32 prev_stmt_flags = ctx->stmt_flags;
+ ctx->stmt_flags = new_flags;
+
check_stmt(ctx, n, new_flags);
+ ctx->stmt_flags = prev_stmt_flags;
+
if (i+1 < max_non_constant_declaration) {
switch (n->kind) {
case Ast_ReturnStmt:
diff --git a/src/checker.hpp b/src/checker.hpp
index bf956393c..a6a5f6788 100644
--- a/src/checker.hpp
+++ b/src/checker.hpp
@@ -435,6 +435,7 @@ struct CheckerContext {
#define MAX_INLINE_FOR_DEPTH 1024ll
i64 inline_for_depth;
+ u32 stmt_flags;
bool in_enum_type;
bool collect_delayed_decls;
bool allow_polymorphic_types;
diff --git a/src/llvm_backend.hpp b/src/llvm_backend.hpp
index 13a250f00..abdeea4ba 100644
--- a/src/llvm_backend.hpp
+++ b/src/llvm_backend.hpp
@@ -535,6 +535,7 @@ gb_internal void lb_mem_zero_ptr(lbProcedure *p, LLVMValueRef ptr, Type *type, u
gb_internal void lb_emit_init_context(lbProcedure *p, lbAddr addr);
+gb_internal lbBranchBlocks lb_lookup_branch_blocks(lbProcedure *p, Ast *ident);
gb_internal lbStructFieldRemapping lb_get_struct_remapping(lbModule *m, Type *t);
gb_internal LLVMTypeRef lb_type_padding_filler(lbModule *m, i64 padding, i64 padding_align);
@@ -556,6 +557,7 @@ gb_internal LLVMTypeRef OdinLLVMGetVectorElementType(LLVMTypeRef type);
gb_internal String lb_filepath_ll_for_module(lbModule *m);
+
gb_internal LLVMTypeRef llvm_array_type(LLVMTypeRef ElementType, uint64_t ElementCount) {
#if LB_USE_NEW_PASS_SYSTEM
return LLVMArrayType2(ElementType, ElementCount);
diff --git a/src/llvm_backend_expr.cpp b/src/llvm_backend_expr.cpp
index d6e8843fa..aba129ea4 100644
--- a/src/llvm_backend_expr.cpp
+++ b/src/llvm_backend_expr.cpp
@@ -3305,6 +3305,62 @@ gb_internal lbValue lb_build_expr_internal(lbProcedure *p, Ast *expr) {
return lb_emit_or_return(p, oe->expr, tv);
case_end;
+ case_ast_node(be, OrBranchExpr, expr);
+ lbBlock *block = nullptr;
+
+ if (be->label != nullptr) {
+ lbBranchBlocks bb = lb_lookup_branch_blocks(p, be->label);
+ switch (be->token.kind) {
+ case Token_or_break: block = bb.break_; break;
+ case Token_or_continue: block = bb.continue_; break;
+ }
+ } else {
+ for (lbTargetList *t = p->target_list; t != nullptr && block == nullptr; t = t->prev) {
+ if (t->is_block) {
+ continue;
+ }
+
+ switch (be->token.kind) {
+ case Token_or_break: block = t->break_; break;
+ case Token_or_continue: block = t->continue_; break;
+ }
+ }
+ }
+
+ lbValue lhs = {};
+ lbValue rhs = {};
+ lb_emit_try_lhs_rhs(p, be->expr, tv, &lhs, &rhs);
+
+ Type *type = default_type(tv.type);
+
+ lbBlock *then = lb_create_block(p, "or_branch.then");
+ lbBlock *done = lb_create_block(p, "or_branch.done"); // NOTE(bill): Append later
+ lbBlock *else_ = lb_create_block(p, "or_branch.else");
+
+ lb_emit_if(p, lb_emit_try_has_value(p, rhs), then, else_);
+ lb_start_block(p, then);
+
+ lbValue res = {};
+ if (lhs.value) {
+ res = lb_emit_conv(p, lhs, type);
+ }
+
+ lb_emit_jump(p, done);
+ lb_start_block(p, else_);
+
+ if (lhs.value) {
+ res = lb_const_nil(p->module, type);
+ }
+
+ if (block != nullptr) {
+ lb_emit_defer_stmts(p, lbDeferExit_Branch, block);
+ }
+ lb_emit_jump(p, block);
+ lb_start_block(p, done);
+
+ return res;
+ case_end;
+
case_ast_node(ta, TypeAssertion, expr);
TokenPos pos = ast_token(expr).pos;
lbValue e = lb_build_expr(p, ta->expr);
diff --git a/src/parser.cpp b/src/parser.cpp
index a81594663..c0498b425 100644
--- a/src/parser.cpp
+++ b/src/parser.cpp
@@ -533,8 +533,13 @@ gb_internal Ast *ast_tag_expr(AstFile *f, Token token, Token name, Ast *expr) {
gb_internal Ast *ast_unary_expr(AstFile *f, Token op, Ast *expr) {
Ast *result = alloc_ast_node(f, Ast_UnaryExpr);
- if (expr && expr->kind == Ast_OrReturnExpr) {
+ if (expr) switch (expr->kind) {
+ case Ast_OrReturnExpr:
syntax_error_with_verbose(expr, "'or_return' within an unary expression not wrapped in parentheses (...)");
+ break;
+ case Ast_OrBranchExpr:
+ syntax_error_with_verbose(expr, "'or_%.*s' within an unary expression not wrapped in parentheses (...)", LIT(expr->OrBranchExpr.token.string));
+ break;
}
result->UnaryExpr.op = op;
@@ -555,11 +560,22 @@ gb_internal Ast *ast_binary_expr(AstFile *f, Token op, Ast *left, Ast *right) {
right = ast_bad_expr(f, op, op);
}
- if (left->kind == Ast_OrReturnExpr) {
+
+ if (left) switch (left->kind) {
+ case Ast_OrReturnExpr:
syntax_error_with_verbose(left, "'or_return' within a binary expression not wrapped in parentheses (...)");
+ break;
+ case Ast_OrBranchExpr:
+ syntax_error_with_verbose(left, "'or_%.*s' within a binary expression not wrapped in parentheses (...)", LIT(left->OrBranchExpr.token.string));
+ break;
}
- if (right->kind == Ast_OrReturnExpr) {
+ if (right) switch (right->kind) {
+ case Ast_OrReturnExpr:
syntax_error_with_verbose(right, "'or_return' within a binary expression not wrapped in parentheses (...)");
+ break;
+ case Ast_OrBranchExpr:
+ syntax_error_with_verbose(right, "'or_%.*s' within a binary expression not wrapped in parentheses (...)", LIT(right->OrBranchExpr.token.string));
+ break;
}
result->BinaryExpr.op = op;
@@ -800,6 +816,14 @@ gb_internal Ast *ast_or_return_expr(AstFile *f, Ast *expr, Token const &token) {
return result;
}
+gb_internal Ast *ast_or_branch_expr(AstFile *f, Ast *expr, Token const &token, Ast *label) {
+ Ast *result = alloc_ast_node(f, Ast_OrBranchExpr);
+ result->OrBranchExpr.expr = expr;
+ result->OrBranchExpr.token = token;
+ result->OrBranchExpr.label = label;
+ return result;
+}
+
gb_internal Ast *ast_type_assertion(AstFile *f, Ast *expr, Token dot, Ast *type) {
Ast *result = alloc_ast_node(f, Ast_TypeAssertion);
result->TypeAssertion.expr = expr;
@@ -1477,19 +1501,20 @@ gb_internal Token expect_operator(AstFile *f) {
// okay
} else if (prev.kind == Token_if || prev.kind == Token_when) {
// okay
- } else if (prev.kind == Token_or_else || prev.kind == Token_or_return) {
+ } else if (prev.kind == Token_or_else || prev.kind == Token_or_return ||
+ prev.kind == Token_or_break || prev.kind == Token_or_continue) {
// okay
} else if (!gb_is_between(prev.kind, Token__OperatorBegin+1, Token__OperatorEnd-1)) {
String p = token_to_string(prev);
- syntax_error(f->curr_token, "Expected an operator, got '%.*s'",
+ syntax_error(prev, "Expected an operator, got '%.*s'",
LIT(p));
} else if (!f->allow_range && is_token_range(prev)) {
String p = token_to_string(prev);
- syntax_error(f->curr_token, "Expected an non-range operator, got '%.*s'",
+ syntax_error(prev, "Expected an non-range operator, got '%.*s'",
LIT(p));
}
- if (f->curr_token.kind == Token_Ellipsis) {
- syntax_warning(f->curr_token, "'..' for ranges has now been deprecated, prefer '..='");
+ if (prev.kind == Token_Ellipsis) {
+ syntax_warning(prev, "'..' for ranges has now been deprecated, prefer '..='");
f->tokens[f->curr_token_index].flags |= TokenFlag_Replace;
}
@@ -1736,6 +1761,8 @@ gb_internal Ast *strip_or_return_expr(Ast *node) {
}
if (node->kind == Ast_OrReturnExpr) {
node = node->OrReturnExpr.expr;
+ } else if (node->kind == Ast_OrBranchExpr) {
+ node = node->OrBranchExpr.expr;
} else if (node->kind == Ast_ParenExpr) {
node = node->ParenExpr.expr;
} else {
@@ -2869,8 +2896,16 @@ gb_internal Ast *parse_call_expr(AstFile *f, Ast *operand) {
}
gb_internal void parse_check_or_return(Ast *operand, char const *msg) {
- if (operand && operand->kind == Ast_OrReturnExpr) {
+ if (operand == nullptr) {
+ return;
+ }
+ switch (operand->kind) {
+ case Ast_OrReturnExpr:
syntax_error_with_verbose(operand, "'or_return' use within %s is not wrapped in parentheses (...)", msg);
+ break;
+ case Ast_OrBranchExpr:
+ syntax_error_with_verbose(operand, "'or_%.*s' use within %s is not wrapped in parentheses (...)", msg, LIT(operand->OrBranchExpr.token.string));
+ break;
}
}
@@ -3004,6 +3039,18 @@ gb_internal Ast *parse_atom_expr(AstFile *f, Ast *operand, bool lhs) {
operand = ast_or_return_expr(f, operand, expect_token(f, Token_or_return));
break;
+ case Token_or_break:
+ case Token_or_continue:
+ {
+ Token token = advance_token(f);
+ Ast *label = nullptr;
+ if (f->curr_token.kind == Token_Ident) {
+ label = parse_ident(f);
+ }
+ operand = ast_or_branch_expr(f, operand, token, label);
+ }
+ break;
+
case Token_OpenBrace:
if (!lhs && is_literal_type(operand) && f->expr_level >= 0) {
operand = parse_literal_value(f, operand);
diff --git a/src/parser.hpp b/src/parser.hpp
index dd7bd0928..bce818652 100644
--- a/src/parser.hpp
+++ b/src/parser.hpp
@@ -464,6 +464,7 @@ AST_KIND(_ExprBegin, "", bool) \
AST_KIND(TernaryWhenExpr, "ternary when expression", struct { Ast *x, *cond, *y; }) \
AST_KIND(OrElseExpr, "or_else expression", struct { Ast *x; Token token; Ast *y; }) \
AST_KIND(OrReturnExpr, "or_return expression", struct { Ast *expr; Token token; }) \
+ AST_KIND(OrBranchExpr, "or branch expression", struct { Ast *expr; Token token; Ast *label; }) \
AST_KIND(TypeAssertion, "type assertion", struct { \
Ast *expr; \
Token dot; \
diff --git a/src/parser_pos.cpp b/src/parser_pos.cpp
index 3d2e8f27d..f49c40f16 100644
--- a/src/parser_pos.cpp
+++ b/src/parser_pos.cpp
@@ -52,6 +52,7 @@ gb_internal Token ast_token(Ast *node) {
case Ast_TernaryWhenExpr: return ast_token(node->TernaryWhenExpr.x);
case Ast_OrElseExpr: return ast_token(node->OrElseExpr.x);
case Ast_OrReturnExpr: return ast_token(node->OrReturnExpr.expr);
+ case Ast_OrBranchExpr: return ast_token(node->OrBranchExpr.expr);
case Ast_TypeAssertion: return ast_token(node->TypeAssertion.expr);
case Ast_TypeCast: return node->TypeCast.token;
case Ast_AutoCast: return node->AutoCast.token;
@@ -195,6 +196,11 @@ Token ast_end_token(Ast *node) {
case Ast_TernaryWhenExpr: return ast_end_token(node->TernaryWhenExpr.y);
case Ast_OrElseExpr: return ast_end_token(node->OrElseExpr.y);
case Ast_OrReturnExpr: return node->OrReturnExpr.token;
+ case Ast_OrBranchExpr:
+ if (node->OrBranchExpr.label != nullptr) {
+ return ast_end_token(node->OrBranchExpr.label);
+ }
+ return node->OrBranchExpr.token;
case Ast_TypeAssertion: return ast_end_token(node->TypeAssertion.type);
case Ast_TypeCast: return ast_end_token(node->TypeCast.expr);
case Ast_AutoCast: return ast_end_token(node->AutoCast.expr);
diff --git a/src/tokenizer.cpp b/src/tokenizer.cpp
index ad7aa81de..dd9908be5 100644
--- a/src/tokenizer.cpp
+++ b/src/tokenizer.cpp
@@ -116,6 +116,8 @@ TOKEN_KIND(Token__KeywordBegin, ""), \
TOKEN_KIND(Token_context, "context"), \
TOKEN_KIND(Token_or_else, "or_else"), \
TOKEN_KIND(Token_or_return, "or_return"), \
+ TOKEN_KIND(Token_or_break, "or_break"), \
+ TOKEN_KIND(Token_or_continue, "or_continue"), \
TOKEN_KIND(Token_asm, "asm"), \
TOKEN_KIND(Token_matrix, "matrix"), \
TOKEN_KIND(Token__KeywordEnd, ""), \
@@ -1072,6 +1074,8 @@ semicolon_check:;
case Token_fallthrough:
case Token_return:
case Token_or_return:
+ case Token_or_break:
+ case Token_or_continue:
/*fallthrough*/
case Token_Integer:
case Token_Float: