mirror of
https://github.com/odin-lang/Odin.git
synced 2025-12-30 18:02:02 +00:00
Allow assignment of procedure calls with #optional_ok to single values
This commit is contained in:
@@ -651,12 +651,26 @@ i64 check_distance_between_types(CheckerContext *c, Operand *operand, Type *type
|
||||
}
|
||||
|
||||
Ast *expr = unparen_expr(operand->expr);
|
||||
if (expr != nullptr && expr->kind == Ast_AutoCast) {
|
||||
Operand x = *operand;
|
||||
x.expr = expr->AutoCast.expr;
|
||||
bool ok = check_cast_internal(c, &x, type);
|
||||
if (ok) {
|
||||
return MAXIMUM_TYPE_DISTANCE;
|
||||
if (expr != nullptr) {
|
||||
if (expr->kind == Ast_AutoCast) {
|
||||
Operand x = *operand;
|
||||
x.expr = expr->AutoCast.expr;
|
||||
bool ok = check_cast_internal(c, &x, type);
|
||||
if (ok) {
|
||||
return MAXIMUM_TYPE_DISTANCE;
|
||||
}
|
||||
} else if (expr->kind == Ast_CallExpr) {
|
||||
// NOTE(bill, 2021-04-19): Allow assignment of procedure calls with #optional_ok
|
||||
ast_node(ce, CallExpr, expr);
|
||||
Type *pt = base_type(type_of_expr(ce->proc));
|
||||
if (pt->kind == Type_Proc && pt->Proc.optional_ok) {
|
||||
Operand x = *operand;
|
||||
x.type = pt->Proc.results->Tuple.variables[0]->type;
|
||||
i64 res = check_distance_between_types(c, &x, type);
|
||||
if (res >= 0) {
|
||||
return res+1;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -694,7 +708,7 @@ bool check_is_assignable_to(CheckerContext *c, Operand *operand, Type *type) {
|
||||
return check_is_assignable_to_with_score(c, operand, type, &score);
|
||||
}
|
||||
|
||||
void add_optional_ok_for_procedure(Type *type, Operand *operand) {
|
||||
void add_optional_ok_for_procedure(Type *type, Operand *operand, Type *type_hint) {
|
||||
type = base_type(type);
|
||||
if (type->kind == Type_Proc && type->Proc.optional_ok) {
|
||||
operand->mode = Addressing_OptionalOk;
|
||||
@@ -757,7 +771,7 @@ void check_assignment(CheckerContext *c, Operand *operand, Type *type, String co
|
||||
Entity *e = procs[i];
|
||||
add_entity_use(c, operand->expr, e);
|
||||
good = true;
|
||||
add_optional_ok_for_procedure(e->type, operand);
|
||||
add_optional_ok_for_procedure(e->type, operand, type);
|
||||
break;
|
||||
}
|
||||
}
|
||||
@@ -8254,7 +8268,7 @@ ExprKind check_call_expr(CheckerContext *c, Operand *operand, Ast *call, Ast *pr
|
||||
if (type == nullptr) {
|
||||
type = pt;
|
||||
}
|
||||
add_optional_ok_for_procedure(type, operand);
|
||||
add_optional_ok_for_procedure(type, operand, type_hint);
|
||||
}
|
||||
|
||||
// add_type_and_value(c->info, operand->expr, operand->mode, operand->type, operand->value);
|
||||
|
||||
Reference in New Issue
Block a user