mirror of
https://github.com/odin-lang/Odin.git
synced 2025-12-30 01:44:36 +00:00
Fix trailing default argument checking
This commit is contained in:
@@ -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);
|
||||
|
||||
Reference in New Issue
Block a user