diff options
| author | gingerBill <bill@gingerbill.org> | 2018-12-14 18:36:06 +0000 |
|---|---|---|
| committer | gingerBill <bill@gingerbill.org> | 2018-12-14 18:36:06 +0000 |
| commit | b7eebe5d004380dec34a185bca6e5e343589de69 (patch) | |
| tree | 6e9b74c045c4c2d3880b25945291703a5f70eeac /src/check_type.cpp | |
| parent | 57d4333ed3a36778a034f144a6998587d61ed831 (diff) | |
Fix polymorphic record types with constant value parameters
Diffstat (limited to 'src/check_type.cpp')
| -rw-r--r-- | src/check_type.cpp | 60 |
1 files changed, 56 insertions, 4 deletions
diff --git a/src/check_type.cpp b/src/check_type.cpp index 33797ba27..63835dd76 100644 --- a/src/check_type.cpp +++ b/src/check_type.cpp @@ -627,7 +627,7 @@ void check_union_type(CheckerContext *ctx, Type *union_type, Ast *node, Array<Op e = alloc_entity_type_name(scope, token, operand.type); e->TypeName.is_type_alias = true; } else { - GB_ASSERT(operand.mode == Addressing_Constant); + // GB_ASSERT(operand.mode == Addressing_Constant); e = alloc_entity_constant(scope, token, operand.type, operand.value); } } else { @@ -1117,9 +1117,8 @@ bool check_type_specialization_to(CheckerContext *ctx, Type *specialization, Typ if (t->kind != s->kind) { return false; } - // gb_printf_err("#1 %s %s\n", type_to_string(type), type_to_string(specialization)); - if (t->kind == Type_Struct) { + if (t->kind == Type_Struct) { if (t->Struct.polymorphic_parent == specialization) { return true; } @@ -1136,7 +1135,58 @@ bool check_type_specialization_to(CheckerContext *ctx, Type *specialization, Typ Entity *t_e = t_tuple->variables[i]; Type *st = s_e->type; Type *tt = t_e->type; - bool ok = is_polymorphic_type_assignable(ctx, st, tt, true, modify_type); + + // NOTE(bill, 2018-12-14): This is needed to override polymorphic named constants in types + if (st->kind == Type_Generic && t_e->kind == Entity_Constant) { + Entity *e = scope_lookup(st->Generic.scope, st->Generic.name); + GB_ASSERT(e != nullptr); + if (modify_type) { + e->kind = Entity_Constant; + e->Constant.value = t_e->Constant.value; + e->type = t_e->type; + } + } else { + bool ok = is_polymorphic_type_assignable(ctx, st, tt, true, modify_type); + } + } + + if (modify_type) { + // NOTE(bill): This is needed in order to change the actual type but still have the types defined within it + gb_memmove(specialization, type, gb_size_of(Type)); + } + + return true; + } + } else if (t->kind == Type_Union) { + if (t->Union.polymorphic_parent == specialization) { + return true; + } + + if (t->Union.polymorphic_parent == s->Union.polymorphic_parent && + s->Union.polymorphic_params != nullptr && + t->Union.polymorphic_params != nullptr) { + + TypeTuple *s_tuple = &s->Union.polymorphic_params->Tuple; + TypeTuple *t_tuple = &t->Union.polymorphic_params->Tuple; + GB_ASSERT(t_tuple->variables.count == s_tuple->variables.count); + for_array(i, s_tuple->variables) { + Entity *s_e = s_tuple->variables[i]; + Entity *t_e = t_tuple->variables[i]; + Type *st = s_e->type; + Type *tt = t_e->type; + + // NOTE(bill, 2018-12-14): This is needed to override polymorphic named constants in types + if (st->kind == Type_Generic && t_e->kind == Entity_Constant) { + Entity *e = scope_lookup(st->Generic.scope, st->Generic.name); + GB_ASSERT(e != nullptr); + if (modify_type) { + e->kind = Entity_Constant; + e->Constant.value = t_e->Constant.value; + e->type = t_e->type; + } + } else { + bool ok = is_polymorphic_type_assignable(ctx, st, tt, true, modify_type); + } } if (modify_type) { @@ -1168,6 +1218,7 @@ Type *determine_type_from_polymorphic(CheckerContext *ctx, Type *poly_type, Oper } return t_invalid; } + if (is_polymorphic_type_assignable(ctx, poly_type, operand.type, false, modify_type)) { return poly_type; } @@ -2290,6 +2341,7 @@ bool check_type_internal(CheckerContext *ctx, Ast *e, Type **type, Type *named_t entity_scope = ps; } Entity *e = alloc_entity_type_name(entity_scope, token, t); + t->Generic.entity = e; e->TypeName.is_type_alias = true; e->state = EntityState_Resolved; add_entity(ctx->checker, ps, ident, e); |