diff options
| author | Ginger Bill <bill@gingerbill.org> | 2017-06-26 18:20:24 +0100 |
|---|---|---|
| committer | Ginger Bill <bill@gingerbill.org> | 2017-06-26 18:20:24 +0100 |
| commit | c949ca2a5c8fe18fd965bd294454243c683a042c (patch) | |
| tree | 4df9a9fc6e697b43327db2a6f9e84c41ab19c7e7 /src/check_expr.cpp | |
| parent | d974b29f67ea536e1da033e41f3b03e05696f438 (diff) | |
Allow for named arguments for polymorphic procedures
Diffstat (limited to 'src/check_expr.cpp')
| -rw-r--r-- | src/check_expr.cpp | 266 |
1 files changed, 153 insertions, 113 deletions
diff --git a/src/check_expr.cpp b/src/check_expr.cpp index 8f21d257a..b0f8e88cf 100644 --- a/src/check_expr.cpp +++ b/src/check_expr.cpp @@ -4967,8 +4967,99 @@ bool check_unpack_arguments(Checker *c, isize lhs_count, Array<Operand> *operand return optional_ok; } +// NOTE(bill): Returns `NULL` on failure +Entity *find_or_generate_polymorphic_procedure(Checker *c, Entity *base_entity, Array<Operand> *operands, ProcedureInfo *proc_info_) { + if (base_entity == NULL) { + return NULL; + } + if (!is_type_proc(base_entity->type)) { + return NULL; + } + TypeProc *pt = &base_type(base_entity->type)->Proc; + if (!pt->is_generic || pt->is_generic_specialized) { + return NULL; + } + + if (pt->param_count != operands->count) { + return NULL; + } + + DeclInfo *old_decl = decl_info_of_entity(&c->info, base_entity); + GB_ASSERT(old_decl != NULL); + + gbAllocator a = heap_allocator(); + + CheckerContext prev_context = c->context; + defer (c->context = prev_context); + + Scope *scope = make_scope(base_entity->scope, a); + scope->is_proc = true; + c->context.scope = scope; + + // NOTE(bill): This is slightly memory leaking if the type already exists + // Maybe it's better to check with the previous types first? + Type *final_proc_type = make_type_proc(c->allocator, c->context.scope, NULL, 0, NULL, 0, false, pt->calling_convention); + check_procedure_type(c, final_proc_type, pt->node, operands); + + auto *found = map_get(&c->info.gen_procs, hash_pointer(base_entity->identifier)); + if (found) { + for_array(i, *found) { + Entity *other = (*found)[i]; + if (are_types_identical(other->type, final_proc_type)) { + // NOTE(bill): This scope is not needed any more, destroy it + destroy_scope(scope); + return other; + } + } + } + + + AstNode *proc_decl = clone_ast_node(a, old_decl->proc_decl); + ast_node(pd, ProcDecl, proc_decl); + // NOTE(bill): Associate the scope declared above with this procedure declaration's type + add_scope(c, pd->type, final_proc_type->Proc.scope); + final_proc_type->Proc.is_generic_specialized = true; + + u64 tags = base_entity->Procedure.tags; + AstNode *ident = clone_ast_node(a, base_entity->identifier); + Token token = ident->Ident; + DeclInfo *d = make_declaration_info(c->allocator, c->context.scope, old_decl->parent); + d->gen_proc_type = final_proc_type; + d->type_expr = pd->type; + d->proc_decl = proc_decl; + + + Entity *entity = make_entity_procedure(c->allocator, NULL, token, final_proc_type, tags); + entity->identifier = ident; + + add_entity_and_decl_info(c, ident, entity, d); + entity->scope = base_entity->scope; + + ProcedureInfo proc_info = {}; + proc_info.file = c->curr_ast_file; + proc_info.token = token; + proc_info.decl = d; + proc_info.type = final_proc_type; + proc_info.body = pd->body; + proc_info.tags = tags; + + if (found) { + array_add(found, entity); + } else { + Array<Entity *> array = {}; + array_init(&array, heap_allocator()); + array_add(&array, entity); + map_set(&c->info.gen_procs, hash_pointer(entity->identifier), array); + } + + GB_ASSERT(entity != NULL); + + + if (proc_info_) *proc_info_ = proc_info; + return entity; +} CALL_ARGUMENT_CHECKER(check_call_arguments_internal) { ast_node(ce, CallExpr, call); @@ -5056,89 +5147,17 @@ CALL_ARGUMENT_CHECKER(check_call_arguments_internal) { } } else { // NOTE(bill): Generate the procedure type for this generic instance - // TODO(bill): Clean this shit up! - ProcedureInfo proc_info = {}; if (pt->is_generic && !pt->is_generic_specialized) { - GB_ASSERT(entity != NULL); - - DeclInfo *old_decl = decl_info_of_entity(&c->info, entity); - GB_ASSERT(old_decl != NULL); - - gbAllocator a = heap_allocator(); - - CheckerContext prev_context = c->context; - defer (c->context = prev_context); - - Scope *scope = make_scope(entity->scope, a); - scope->is_proc = true; - c->context.scope = scope; - - // NOTE(bill): This is slightly memory leaking if the type already exists - // Maybe it's better to check with the previous types first? - final_proc_type = make_type_proc(c->allocator, c->context.scope, NULL, 0, NULL, 0, false, pt->calling_convention); - check_procedure_type(c, final_proc_type, pt->node, &operands); - - bool skip = false; - auto *found = map_get(&c->info.gen_procs, hash_pointer(entity->identifier)); - if (found) { - for_array(i, *found) { - Entity *other = (*found)[i]; - if (are_types_identical(other->type, final_proc_type)) { - skip = true; - gen_entity = other; - final_proc_type = other->type; - break; - } - } - } - - if (skip) { - // NOTE(bill): It is not needed any more, destroy it - destroy_scope(scope); - } else { - AstNode *proc_decl = clone_ast_node(a, old_decl->proc_decl); - ast_node(pd, ProcDecl, proc_decl); - // NOTE(bill): Associate the scope declared above with this procedure declaration's type - add_scope(c, pd->type, final_proc_type->Proc.scope); - final_proc_type->Proc.is_generic_specialized = true; - - u64 tags = entity->Procedure.tags; - AstNode *ident = clone_ast_node(a, entity->identifier); - Token token = ident->Ident; - DeclInfo *d = make_declaration_info(c->allocator, c->context.scope, old_decl->parent); - d->gen_proc_type = final_proc_type; - d->type_expr = pd->type; - d->proc_decl = proc_decl; - - gen_entity = make_entity_procedure(c->allocator, NULL, token, final_proc_type, tags); - gen_entity->identifier = ident; - - add_entity_and_decl_info(c, ident, gen_entity, d); - gen_entity->scope = entity->scope; - add_entity_use(c, ident, gen_entity); - - proc_info.file = c->curr_ast_file; - proc_info.token = token; - proc_info.decl = d; - proc_info.type = final_proc_type; - proc_info.body = pd->body; - proc_info.tags = tags; - - if (found) { - array_add(found, gen_entity); - } else { - Array<Entity *> array = {}; - array_init(&array, heap_allocator()); - array_add(&array, gen_entity); - map_set(&c->info.gen_procs, hash_pointer(entity->identifier), array); - } + gen_entity = find_or_generate_polymorphic_procedure(c, entity, &operands, &proc_info); + if (gen_entity != NULL) { + GB_ASSERT(is_type_proc(gen_entity->type)); + final_proc_type = gen_entity->type; } - - GB_ASSERT(gen_entity != NULL); } + GB_ASSERT(is_type_proc(final_proc_type)); TypeProc *pt = &final_proc_type->Proc; GB_ASSERT(pt->params != NULL); @@ -5154,7 +5173,7 @@ CALL_ARGUMENT_CHECKER(check_call_arguments_internal) { if (o.mode == Addressing_Invalid) { continue; } else if (o.mode != Addressing_Type) { - error(o.expr, "Expected a type for the argument"); + error(o.expr, "Expected a type for the argument `%.*s`", LIT(e->token.string)); err = CallArgumentError_WrongTypes; } @@ -5214,7 +5233,10 @@ CALL_ARGUMENT_CHECKER(check_call_arguments_internal) { } if (gen_entity != NULL && err == CallArgumentError_None) { - check_procedure_later(c, proc_info); + if (proc_info.decl != NULL) { + // NOTE(bill): Check the newly generated procedure body + check_procedure_later(c, proc_info); + } } } } @@ -5278,6 +5300,10 @@ CALL_ARGUMENT_CHECKER(check_named_call_arguments) { isize param_count = pt->param_count; bool *visited = gb_alloc_array(c->allocator, bool, param_count); + Array<Operand> ordered_operands = {}; + array_init_count(&ordered_operands, heap_allocator(), param_count); + defer (array_free(&ordered_operands)); + for_array(i, ce->args) { AstNode *arg = ce->args[i]; ast_node(fv, FieldValue, arg); @@ -5308,36 +5334,9 @@ CALL_ARGUMENT_CHECKER(check_named_call_arguments) { } visited[index] = true; - Operand *o = &operands[i]; - Entity *e = pt->params->Tuple.variables[index]; - - if (e->kind == Entity_TypeName) { - GB_ASSERT(pt->is_generic); - GB_ASSERT(!pt->variadic); - if (o->mode == Addressing_Invalid) { - continue; - } else if (o->mode != Addressing_Type) { - error(o->expr, "Expected a type for the argument"); - } - if (are_types_identical(e->type, o->type)) { - score += assign_score_function(1); - } else { - score += assign_score_function(5); - } - } else { - i64 s = 0; - if (!check_is_assignable_to_with_score(c, o, e->type, &s)) { - if (show_error) { - check_assignment(c, o, e->type, str_lit("procedure argument")); - } - err = CallArgumentError_WrongTypes; - } - score += s; - } + ordered_operands[index] = operands[i]; } - -#if 1 isize param_count_to_check = param_count; if (pt->variadic) { param_count_to_check--; @@ -5368,19 +5367,58 @@ CALL_ARGUMENT_CHECKER(check_named_call_arguments) { err = CallArgumentError_ParameterMissing; } } -#endif - if (pt->is_generic) { - if (show_error) { - error(call, "Generic procedures do not yet support named arguments"); + Entity *gen_entity = NULL; + if (pt->is_generic && err == CallArgumentError_None) { + // err = CallArgumentError_GenericProcedureNotSupported; + ProcedureInfo proc_info = {}; + gen_entity = find_or_generate_polymorphic_procedure(c, entity, &ordered_operands, &proc_info); + if (gen_entity != NULL) { + if (proc_info.decl != NULL) { + check_procedure_later(c, proc_info); + } + + pt = &base_type(gen_entity->type)->Proc; + } + } + + for (isize i = 0; i < param_count; i++) { + Operand *o = &ordered_operands[i]; + if (o->mode == Addressing_Invalid) { + continue; + } + Entity *e = pt->params->Tuple.variables[i]; + + if (e->kind == Entity_TypeName) { + GB_ASSERT(pt->is_generic); + GB_ASSERT(!pt->variadic); + if (o->mode != Addressing_Type) { + if (show_error) { + error(o->expr, "Expected a type for the argument `%.*s`", LIT(e->token.string)); + } + err = CallArgumentError_WrongTypes; + } + if (are_types_identical(e->type, o->type)) { + score += assign_score_function(1); + } else { + score += assign_score_function(10); + } + } else { + i64 s = 0; + if (!check_is_assignable_to_with_score(c, o, e->type, &s)) { + if (show_error) { + check_assignment(c, o, e->type, str_lit("procedure argument")); + } + err = CallArgumentError_WrongTypes; + } + score += s; } - err = CallArgumentError_GenericProcedureNotSupported; } if (data) { data->score = score; data->result_type = pt->results; - data->gen_entity = NULL; + data->gen_entity = gen_entity; } return err; @@ -5487,12 +5525,14 @@ CallArgumentData check_call_arguments(Checker *c, Operand *operand, Type *proc_t proc_type = e->type; CallArgumentData data = {}; CallArgumentError err = call_checker(c, call, proc_type, e, operands, CallArgumentMode_ShowErrors, &data); + if (data.gen_entity != NULL) add_entity_use(c, ce->proc, data.gen_entity); return data; } } else { Entity *e = entity_of_ident(&c->info, operand->expr); CallArgumentData data = {}; CallArgumentError err = call_checker(c, call, proc_type, e, operands, CallArgumentMode_ShowErrors, &data); + if (data.gen_entity != NULL) add_entity_use(c, ce->proc, data.gen_entity); return data; } |