diff options
| author | gingerBill <gingerBill@users.noreply.github.com> | 2025-10-10 11:48:18 +0100 |
|---|---|---|
| committer | gingerBill <gingerBill@users.noreply.github.com> | 2025-10-10 11:48:18 +0100 |
| commit | 061b14e1649ea5829e2612dc905cdc9af863a071 (patch) | |
| tree | 97526a2536ac5e209b8882b490e0d5b7b854ae19 /src | |
| parent | 1572ed57b669ed06d4ad7513bc51f70ba0952c16 (diff) | |
Allow `intrinsics.concatenate` to be variadic
Diffstat (limited to 'src')
| -rw-r--r-- | src/check_builtin.cpp | 87 | ||||
| -rw-r--r-- | src/checker_builtin_procs.hpp | 2 |
2 files changed, 48 insertions, 41 deletions
diff --git a/src/check_builtin.cpp b/src/check_builtin.cpp index 13a1c4e69..c337802c9 100644 --- a/src/check_builtin.cpp +++ b/src/check_builtin.cpp @@ -4887,56 +4887,32 @@ gb_internal bool check_builtin_procedure(CheckerContext *c, Operand *operand, As case BuiltinProc_concatenate: { Operand lhs = {}; - Operand rhs = {}; check_expr(c, &lhs, ce->args[0]); if (lhs.mode == Addressing_Invalid) { return false; } - check_expr(c, &rhs, ce->args[1]); - if (rhs.mode == Addressing_Invalid) { - return false; - } if (lhs.mode != Addressing_Constant) { - error(lhs.expr, "'%*.s' expects a constant array or slice", LIT(builtin_name)); - return false; - } - if (rhs.mode != Addressing_Constant) { - error(rhs.expr, "'%*.s' expects a constant array or slice", LIT(builtin_name)); - return false; - } - - if (!are_types_identical(lhs.type, rhs.type)) { - gbString a = type_to_string(lhs.type); - gbString b = type_to_string(rhs.type); - error(rhs.expr, "'%*.s' expects a two constant values of the same type, got '%s' vs '%s'", LIT(builtin_name), a, b); - gb_string_free(b); - gb_string_free(a); + error(lhs.expr, "'%.*s' expects a constant array or slice", LIT(builtin_name)); return false; } + operand->type = lhs.type; + operand->mode = Addressing_Value; if (!is_type_slice(lhs.type) && !is_type_array(lhs.type)) { gbString a = type_to_string(lhs.type); - error(lhs.expr, "'%*.s' expects a constant array or slice, got %s", LIT(builtin_name), a); + error(lhs.expr, "'%.*s' expects a constant array or slice, got %s", LIT(builtin_name), a); gb_string_free(a); return false; } - if (lhs.value.kind != ExactValue_Compound) { gbString a = exact_value_to_string(lhs.value); error(lhs.expr, "Expected a compound literal value for '%.*s', got '%s'", LIT(builtin_name), a); gb_string_free(a); return false; } - if (rhs.value.kind != ExactValue_Compound) { - gbString a = exact_value_to_string(rhs.value); - error(rhs.expr, "Expected a compound literal value for '%.*s', got '%s'", LIT(builtin_name), a); - gb_string_free(a); - return false; - } ast_node(lhs_cl, CompoundLit, lhs.value.value_compound); - ast_node(rhs_cl, CompoundLit, rhs.value.value_compound); for (Ast *elem : lhs_cl->elems) { if (elem->kind == Ast_FieldValue) { @@ -4944,26 +4920,57 @@ gb_internal bool check_builtin_procedure(CheckerContext *c, Operand *operand, As return false; } } - for (Ast *elem : rhs_cl->elems) { - if (elem->kind == Ast_FieldValue) { - error(elem, "'%.*s' does not allow the use of 'field = value' to be concatenated together", LIT(builtin_name)); - return false; - } - } Ast *type_ast = lhs_cl->type; - if (type_ast == nullptr) { - type_ast = rhs_cl->type; - } - Array<Ast *> new_elems = {}; array_init(&new_elems, heap_allocator()); array_add_elems(&new_elems, lhs_cl->elems.data, lhs_cl->elems.count); - array_add_elems(&new_elems, rhs_cl->elems.data, rhs_cl->elems.count); - Ast *new_compound_lit = ast_compound_lit(lhs.expr->file(), type_ast, new_elems, ast_token(lhs.expr), ast_end_token(rhs.expr)); + for (isize i = 1; i < ce->args.count; i++) { + Operand extra = {}; + check_expr(c, &extra, ce->args[i]); + if (extra.mode == Addressing_Invalid) { + return false; + } + if (extra.mode != Addressing_Constant) { + error(extra.expr, "'%.*s' expects a constant array or slice", LIT(builtin_name)); + return false; + } + if (!are_types_identical(lhs.type, extra.type)) { + gbString a = type_to_string(lhs.type); + gbString b = type_to_string(extra.type); + error(extra.expr, "'%.*s' expects constant values of the same type, got '%s' vs '%s'", LIT(builtin_name), a, b); + gb_string_free(b); + gb_string_free(a); + return false; + } + + if (extra.value.kind != ExactValue_Compound) { + gbString a = exact_value_to_string(extra.value); + error(extra.expr, "Expected a compound literal value for '%.*s', got '%s'", LIT(builtin_name), a); + gb_string_free(a); + return false; + } + + ast_node(extra_cl, CompoundLit, extra.value.value_compound); + + + for (Ast *elem : extra_cl->elems) { + if (elem->kind == Ast_FieldValue) { + error(elem, "'%.*s' does not allow the use of 'field = value' to be concatenated together", LIT(builtin_name)); + return false; + } + } + + if (type_ast == nullptr) { + type_ast = extra_cl->type; + } + array_add_elems(&new_elems, extra_cl->elems.data, extra_cl->elems.count); + } + + Ast *new_compound_lit = ast_compound_lit(lhs.expr->file(), type_ast, new_elems, ast_token(lhs.expr), ast_end_token(ce->args[ce->args.count-1])); operand->mode = Addressing_Constant; operand->value = exact_value_compound(new_compound_lit); diff --git a/src/checker_builtin_procs.hpp b/src/checker_builtin_procs.hpp index 8de6a4f5e..01502128a 100644 --- a/src/checker_builtin_procs.hpp +++ b/src/checker_builtin_procs.hpp @@ -429,7 +429,7 @@ gb_global BuiltinProc builtin_procs[BuiltinProc_COUNT] = { {STR_LIT("soa_struct"), 2, false, Expr_Expr, BuiltinProcPkg_intrinsics}, // Type - {STR_LIT("concatenate"), 2, false, Expr_Expr, BuiltinProcPkg_intrinsics}, + {STR_LIT("concatenate"), 2, true, Expr_Expr, BuiltinProcPkg_intrinsics}, {STR_LIT("alloca"), 2, false, Expr_Expr, BuiltinProcPkg_intrinsics}, {STR_LIT("cpu_relax"), 0, false, Expr_Stmt, BuiltinProcPkg_intrinsics}, |