diff options
| author | Ginger Bill <bill@gingerbill.org> | 2017-06-11 20:52:54 +0100 |
|---|---|---|
| committer | Ginger Bill <bill@gingerbill.org> | 2017-06-11 20:52:54 +0100 |
| commit | c2c935ba818167a7056da5ac61687ecd2d97cc59 (patch) | |
| tree | e0a345e33aebe9811cfdeb3a65b63db13777ed72 /src/check_expr.cpp | |
| parent | 2d73c8868b62dd3555409ee4522205d670ddeb42 (diff) | |
Fix trailing default argument checking
Diffstat (limited to 'src/check_expr.cpp')
| -rw-r--r-- | src/check_expr.cpp | 46 |
1 files changed, 34 insertions, 12 deletions
diff --git a/src/check_expr.cpp b/src/check_expr.cpp index 7e171f095..8286483bc 100644 --- a/src/check_expr.cpp +++ b/src/check_expr.cpp @@ -2,6 +2,7 @@ void check_expr (Checker *c, Operand *operand, AstNode * void check_multi_expr (Checker *c, Operand *operand, AstNode *expression); void check_expr_or_type (Checker *c, Operand *operand, AstNode *expression); ExprKind check_expr_base (Checker *c, Operand *operand, AstNode *expression, Type *type_hint); +void check_expr_with_type_hint (Checker *c, Operand *o, AstNode *e, Type *t); Type * check_type (Checker *c, AstNode *expression, Type *named_type = NULL); void check_type_decl (Checker *c, Entity *e, AstNode *type_expr, Type *def); Entity * check_selector (Checker *c, Operand *operand, AstNode *node, Type *type_hint); @@ -1051,20 +1052,23 @@ Type *check_get_params(Checker *c, Scope *scope, AstNode *_params, bool *is_vari Entity **variables = gb_alloc_array(c->allocator, Entity *, variable_count); isize variable_index = 0; for_array(i, params) { - if (params[i]->kind != AstNode_Field) { + AstNode *param = params[i]; + if (param->kind != AstNode_Field) { continue; } - ast_node(p, Field, params[i]); + ast_node(p, Field, param); AstNode *type_expr = p->type; Type *type = NULL; AstNode *default_value = p->default_value; ExactValue value = {}; + bool default_is_nil = false; if (type_expr == NULL) { Operand o = {}; check_expr(c, &o, default_value); - - if (o.mode != Addressing_Constant) { + if (is_operand_nil(o)) { + default_is_nil = true; + } else if (o.mode != Addressing_Constant) { error_node(default_value, "Default parameter must be a constant"); } else { value = o.value; @@ -1077,7 +1081,7 @@ Type *check_get_params(Checker *c, Scope *scope, AstNode *_params, bool *is_vari if (i+1 == params.count) { is_variadic = true; } else { - error_node(params[i], "Invalid AST: Invalid variadic parameter"); + error_node(param, "Invalid AST: Invalid variadic parameter"); } } @@ -1085,9 +1089,11 @@ Type *check_get_params(Checker *c, Scope *scope, AstNode *_params, bool *is_vari if (default_value != NULL) { Operand o = {}; - check_expr(c, &o, default_value); + check_expr_with_type_hint(c, &o, default_value, type); - if (o.mode != Addressing_Constant) { + if (is_operand_nil(o)) { + default_is_nil = true; + } else if (o.mode != Addressing_Constant) { error_node(default_value, "Default parameter must be a constant"); } else { value = o.value; @@ -1101,6 +1107,10 @@ Type *check_get_params(Checker *c, Scope *scope, AstNode *_params, bool *is_vari error_node(params[i], "Invalid parameter type"); type = t_invalid; } + if (is_type_untyped(type)) { + error_node(params[i], "Cannot determine parameter type from a nil"); + type = t_invalid; + } if (p->flags&FieldFlag_no_alias) { if (!is_type_pointer(type)) { @@ -1121,6 +1131,7 @@ Type *check_get_params(Checker *c, Scope *scope, AstNode *_params, bool *is_vari param->Variable.is_immutable = true; } param->Variable.default_value = value; + param->Variable.default_is_nil = default_is_nil; add_entity(c, scope, name, param); variables[variable_index++] = param; @@ -4838,10 +4849,15 @@ CALL_ARGUMENT_CHECKER(check_call_arguments_internal) { for (isize i = param_count-1; i >= 0; i--) { Entity *e = param_tuple->variables[i]; GB_ASSERT(e->kind == Entity_Variable); - if (e->Variable.default_value.kind == ExactValue_Invalid) { - break; + if (e->Variable.default_value.kind != ExactValue_Invalid) { + param_count_excluding_defaults--; + continue; } - param_count_excluding_defaults--; + if (e->Variable.default_is_nil) { + param_count_excluding_defaults--; + continue; + } + break; } } @@ -4888,7 +4904,8 @@ CALL_ARGUMENT_CHECKER(check_call_arguments_internal) { GB_ASSERT(proc_type->Proc.params != NULL); Entity **sig_params = param_tuple->variables; isize operand_index = 0; - for (; operand_index < param_count_excluding_defaults; operand_index++) { + isize max_operand_count = gb_min(param_count, operands.count); + for (; operand_index < max_operand_count; operand_index++) { Type *t = sig_params[operand_index]->type; Operand o = operands[operand_index]; if (variadic) { @@ -5036,6 +5053,11 @@ CALL_ARGUMENT_CHECKER(check_named_call_arguments) { continue; } + if (e->Variable.default_is_nil) { + score += assign_score_function(1); + continue; + } + if (show_error) { gbString str = type_to_string(e->type); error_node(call, "Parameter `%.*s` of type `%s` is missing in procedure call", @@ -5067,7 +5089,7 @@ Type *check_call_arguments(Checker *c, Operand *operand, Type *proc_type, AstNod for_array(i, ce->args) { AstNode *arg = ce->args[i]; ast_node(fv, FieldValue, arg); - check_expr(c, &operands[i], fv->value); + check_expr_or_type(c, &operands[i], fv->value); } bool vari_expand = (ce->ellipsis.pos.line != 0); |