diff options
| author | gingerBill <bill@gingerbill.org> | 2021-10-03 12:24:54 +0100 |
|---|---|---|
| committer | gingerBill <bill@gingerbill.org> | 2021-10-03 12:24:54 +0100 |
| commit | 51b5a973e27099a906ba3c8d9bb4a4440d57d618 (patch) | |
| tree | 3b28191ae5ebdc5d3cbcaf4ae88ce5a51f3c5586 /src/check_builtin.cpp | |
| parent | 2bdae52fed2ac0845ebb24348f3834dbc63f6d34 (diff) | |
Allow constant string value for the field argument of `offset_of`
Diffstat (limited to 'src/check_builtin.cpp')
| -rw-r--r-- | src/check_builtin.cpp | 30 |
1 files changed, 20 insertions, 10 deletions
diff --git a/src/check_builtin.cpp b/src/check_builtin.cpp index d8081cc14..2df7c2f28 100644 --- a/src/check_builtin.cpp +++ b/src/check_builtin.cpp @@ -669,39 +669,49 @@ bool check_builtin_procedure(CheckerContext *c, Operand *operand, Ast *call, i32 return false; } GB_ASSERT(type != nullptr); + + String field_name = {}; + + if (field_arg == nullptr) { + error(call, "Expected an identifier or constant string for field argument"); + return false; + } - if (field_arg == nullptr || - field_arg->kind != Ast_Ident) { - error(field_arg, "Expected an identifier for field argument"); + if (field_arg->kind == Ast_Ident) { + field_name = field_arg->Ident.token.string; + } else if (field_arg->tav.mode == Addressing_Constant && field_arg->tav.value.kind == ExactValue_String) { + field_name = field_arg->tav.value.value_string; + } + if (field_name.len == 0) { + error(field_arg, "Expected an identifier or constant (non-empty) string for field argument"); return false; } + + if (is_type_array(type)) { gbString t = type_to_string(type); error(field_arg, "Invalid a struct type for 'offset_of', got '%s'", t); gb_string_free(t); return false; } - - - ast_node(arg, Ident, field_arg); - String field_name = arg->token.string; + Selection sel = lookup_field(type, field_name, false); if (sel.entity == nullptr) { gbString type_str = type_to_string(type); error(ce->args[0], - "'%s' has no field named '%.*s'", type_str, LIT(arg->token.string)); + "'%s' has no field named '%.*s'", type_str, LIT(field_name)); gb_string_free(type_str); Type *bt = base_type(type); if (bt->kind == Type_Struct) { - check_did_you_mean_type(arg->token.string, bt->Struct.fields); + check_did_you_mean_type(field_name, bt->Struct.fields); } return false; } if (sel.indirect) { gbString type_str = type_to_string(type); error(ce->args[0], - "Field '%.*s' is embedded via a pointer in '%s'", LIT(arg->token.string), type_str); + "Field '%.*s' is embedded via a pointer in '%s'", LIT(field_name), type_str); gb_string_free(type_str); return false; } |