|
@@ -5117,6 +5117,202 @@ gb_internal bool check_builtin_procedure(CheckerContext *c, Operand *operand, As
|
|
}
|
|
}
|
|
break;
|
|
break;
|
|
|
|
|
|
|
|
+ case BuiltinProc_type_union_tag_type:
|
|
|
|
+ {
|
|
|
|
+ if (operand->mode != Addressing_Type) {
|
|
|
|
+ error(operand->expr, "Expected a type for '%.*s'", LIT(builtin_name));
|
|
|
|
+ operand->mode = Addressing_Invalid;
|
|
|
|
+ operand->type = t_invalid;
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ Type *u = operand->type;
|
|
|
|
+
|
|
|
|
+ if (!is_type_union(u)) {
|
|
|
|
+ error(operand->expr, "Expected a union type for '%.*s'", LIT(builtin_name));
|
|
|
|
+ operand->mode = Addressing_Invalid;
|
|
|
|
+ operand->type = t_invalid;
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ u = base_type(u);
|
|
|
|
+ GB_ASSERT(u->kind == Type_Union);
|
|
|
|
+
|
|
|
|
+ operand->mode = Addressing_Type;
|
|
|
|
+ operand->type = union_tag_type(u);
|
|
|
|
+ }
|
|
|
|
+ break;
|
|
|
|
+
|
|
|
|
+ case BuiltinProc_type_union_tag_offset:
|
|
|
|
+ {
|
|
|
|
+ if (operand->mode != Addressing_Type) {
|
|
|
|
+ error(operand->expr, "Expected a type for '%.*s'", LIT(builtin_name));
|
|
|
|
+ operand->mode = Addressing_Invalid;
|
|
|
|
+ operand->type = t_invalid;
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ Type *u = operand->type;
|
|
|
|
+
|
|
|
|
+ if (!is_type_union(u)) {
|
|
|
|
+ error(operand->expr, "Expected a union type for '%.*s'", LIT(builtin_name));
|
|
|
|
+ operand->mode = Addressing_Invalid;
|
|
|
|
+ operand->type = t_invalid;
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ u = base_type(u);
|
|
|
|
+ GB_ASSERT(u->kind == Type_Union);
|
|
|
|
+
|
|
|
|
+ // NOTE(jakubtomsu): forces calculation of variant_block_size
|
|
|
|
+ type_size_of(u);
|
|
|
|
+ i64 tag_offset = u->Union.variant_block_size;
|
|
|
|
+ GB_ASSERT(tag_offset > 0);
|
|
|
|
+
|
|
|
|
+ operand->mode = Addressing_Constant;
|
|
|
|
+ operand->type = t_untyped_integer;
|
|
|
|
+ operand->value = exact_value_i64(tag_offset);
|
|
|
|
+ }
|
|
|
|
+ break;
|
|
|
|
+
|
|
|
|
+ case BuiltinProc_type_union_base_tag_value:
|
|
|
|
+ {
|
|
|
|
+ if (operand->mode != Addressing_Type) {
|
|
|
|
+ error(operand->expr, "Expected a type for '%.*s'", LIT(builtin_name));
|
|
|
|
+ operand->mode = Addressing_Invalid;
|
|
|
|
+ operand->type = t_invalid;
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ Type *u = operand->type;
|
|
|
|
+
|
|
|
|
+ if (!is_type_union(u)) {
|
|
|
|
+ error(operand->expr, "Expected a union type for '%.*s'", LIT(builtin_name));
|
|
|
|
+ operand->mode = Addressing_Invalid;
|
|
|
|
+ operand->type = t_invalid;
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ u = base_type(u);
|
|
|
|
+ GB_ASSERT(u->kind == Type_Union);
|
|
|
|
+
|
|
|
|
+ operand->mode = Addressing_Constant;
|
|
|
|
+ operand->type = t_untyped_integer;
|
|
|
|
+ operand->value = exact_value_i64(u->Union.kind == UnionType_no_nil ? 0 : 1);
|
|
|
|
+ } break;
|
|
|
|
+
|
|
|
|
+ case BuiltinProc_type_union_variant_count:
|
|
|
|
+ {
|
|
|
|
+ if (operand->mode != Addressing_Type) {
|
|
|
|
+ error(operand->expr, "Expected a type for '%.*s'", LIT(builtin_name));
|
|
|
|
+ operand->mode = Addressing_Invalid;
|
|
|
|
+ operand->type = t_invalid;
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ Type *u = operand->type;
|
|
|
|
+
|
|
|
|
+ if (!is_type_union(u)) {
|
|
|
|
+ error(operand->expr, "Expected a union type for '%.*s'", LIT(builtin_name));
|
|
|
|
+ operand->mode = Addressing_Invalid;
|
|
|
|
+ operand->type = t_invalid;
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ u = base_type(u);
|
|
|
|
+ GB_ASSERT(u->kind == Type_Union);
|
|
|
|
+
|
|
|
|
+ operand->mode = Addressing_Constant;
|
|
|
|
+ operand->type = t_untyped_integer;
|
|
|
|
+ operand->value = exact_value_i64(u->Union.variants.count);
|
|
|
|
+ } break;
|
|
|
|
+
|
|
|
|
+ case BuiltinProc_type_variant_type_of:
|
|
|
|
+ {
|
|
|
|
+ if (operand->mode != Addressing_Type) {
|
|
|
|
+ error(operand->expr, "Expected a type for '%.*s'", LIT(builtin_name));
|
|
|
|
+ operand->mode = Addressing_Invalid;
|
|
|
|
+ operand->type = t_invalid;
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ Type *u = operand->type;
|
|
|
|
+
|
|
|
|
+ if (!is_type_union(u)) {
|
|
|
|
+ error(operand->expr, "Expected a union type for '%.*s'", LIT(builtin_name));
|
|
|
|
+ operand->mode = Addressing_Invalid;
|
|
|
|
+ operand->type = t_invalid;
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ u = base_type(u);
|
|
|
|
+ GB_ASSERT(u->kind == Type_Union);
|
|
|
|
+ Operand x = {};
|
|
|
|
+ check_expr_or_type(c, &x, ce->args[1]);
|
|
|
|
+ if (!is_type_integer(x.type) || x.mode != Addressing_Constant) {
|
|
|
|
+ error(call, "Expected a constant integer for '%.*s", LIT(builtin_name));
|
|
|
|
+ operand->mode = Addressing_Type;
|
|
|
|
+ operand->type = t_invalid;
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ i64 index = big_int_to_i64(&x.value.value_integer);
|
|
|
|
+ if (index < 0 || index >= u->Union.variants.count) {
|
|
|
|
+ error(call, "Variant tag out of bounds index for '%.*s", LIT(builtin_name));
|
|
|
|
+ operand->mode = Addressing_Type;
|
|
|
|
+ operand->type = t_invalid;
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ operand->mode = Addressing_Type;
|
|
|
|
+ operand->type = u->Union.variants[index];
|
|
|
|
+ }
|
|
|
|
+ break;
|
|
|
|
+
|
|
|
|
+ case BuiltinProc_type_variant_index_of:
|
|
|
|
+ {
|
|
|
|
+ if (operand->mode != Addressing_Type) {
|
|
|
|
+ error(operand->expr, "Expected a type for '%.*s'", LIT(builtin_name));
|
|
|
|
+ operand->mode = Addressing_Invalid;
|
|
|
|
+ operand->type = t_invalid;
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ Type *u = operand->type;
|
|
|
|
+
|
|
|
|
+ if (!is_type_union(u)) {
|
|
|
|
+ error(operand->expr, "Expected a union type for '%.*s'", LIT(builtin_name));
|
|
|
|
+ operand->mode = Addressing_Invalid;
|
|
|
|
+ operand->type = t_invalid;
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ Type *v = check_type(c, ce->args[1]);
|
|
|
|
+ u = base_type(u);
|
|
|
|
+ GB_ASSERT(u->kind == Type_Union);
|
|
|
|
+
|
|
|
|
+ i64 index = -1;
|
|
|
|
+ for_array(i, u->Union.variants) {
|
|
|
|
+ Type *vt = u->Union.variants[i];
|
|
|
|
+ if (union_variant_index_types_equal(v, vt)) {
|
|
|
|
+ index = i64(i);
|
|
|
|
+ break;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if (index < 0) {
|
|
|
|
+ error(operand->expr, "Expected a variant type for '%.*s'", LIT(builtin_name));
|
|
|
|
+ operand->mode = Addressing_Invalid;
|
|
|
|
+ operand->type = t_invalid;
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ operand->mode = Addressing_Constant;
|
|
|
|
+ operand->type = t_untyped_integer;
|
|
|
|
+ operand->value = exact_value_i64(index);
|
|
|
|
+ }
|
|
|
|
+ break;
|
|
|
|
+
|
|
case BuiltinProc_type_struct_field_count:
|
|
case BuiltinProc_type_struct_field_count:
|
|
operand->value = exact_value_i64(0);
|
|
operand->value = exact_value_i64(0);
|
|
if (operand->mode != Addressing_Type) {
|
|
if (operand->mode != Addressing_Type) {
|