aboutsummaryrefslogtreecommitdiff
path: root/src/check_builtin.cpp
diff options
context:
space:
mode:
authorgingerBill <bill@gingerbill.org>2021-10-03 12:24:54 +0100
committergingerBill <bill@gingerbill.org>2021-10-03 12:24:54 +0100
commit51b5a973e27099a906ba3c8d9bb4a4440d57d618 (patch)
tree3b28191ae5ebdc5d3cbcaf4ae88ce5a51f3c5586 /src/check_builtin.cpp
parent2bdae52fed2ac0845ebb24348f3834dbc63f6d34 (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.cpp30
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;
}