2019-10-12 15:39:20 +00:00
|
|
|
//! Type inference for expressions.
|
|
|
|
|
2021-07-09 17:12:56 +00:00
|
|
|
use std::{
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
collections::hash_map::Entry,
|
2021-07-09 17:12:56 +00:00
|
|
|
iter::{repeat, repeat_with},
|
|
|
|
mem,
|
|
|
|
sync::Arc,
|
|
|
|
};
|
2019-10-12 15:39:20 +00:00
|
|
|
|
2021-05-14 07:59:30 +00:00
|
|
|
use chalk_ir::{cast::Cast, fold::Shift, Mutability, TyVariableKind};
|
2019-11-13 06:56:33 +00:00
|
|
|
use hir_def::{
|
2022-01-23 03:59:35 +00:00
|
|
|
expr::{ArithOp, Array, BinaryOp, CmpOp, Expr, ExprId, Literal, Ordering, Statement, UnaryOp},
|
2021-12-29 13:35:59 +00:00
|
|
|
generics::TypeOrConstParamData,
|
2019-11-13 06:56:33 +00:00
|
|
|
path::{GenericArg, GenericArgs},
|
2019-11-21 12:39:09 +00:00
|
|
|
resolver::resolver_for_expr,
|
2021-12-07 16:31:26 +00:00
|
|
|
FieldId, FunctionId, ItemContainerId, Lookup,
|
2019-11-13 06:56:33 +00:00
|
|
|
};
|
2020-06-20 09:53:55 +00:00
|
|
|
use hir_expand::name::{name, Name};
|
2021-03-21 16:40:14 +00:00
|
|
|
use stdx::always;
|
2020-08-12 16:26:51 +00:00
|
|
|
use syntax::ast::RangeOp;
|
2019-10-30 14:19:30 +00:00
|
|
|
|
2019-10-12 15:39:20 +00:00
|
|
|
use crate::{
|
2021-07-09 17:12:56 +00:00
|
|
|
autoderef::{self, Autoderef},
|
|
|
|
consteval,
|
2021-07-08 12:16:23 +00:00
|
|
|
infer::coerce::CoerceMany,
|
2021-03-01 18:30:34 +00:00
|
|
|
lower::lower_to_chalk_mutability,
|
2021-04-09 12:15:26 +00:00
|
|
|
mapping::from_chalk,
|
2021-08-22 15:21:47 +00:00
|
|
|
method_resolution,
|
2021-02-28 09:58:34 +00:00
|
|
|
primitive::{self, UintTy},
|
2021-04-06 08:50:55 +00:00
|
|
|
static_lifetime, to_chalk_trait_id,
|
2021-04-06 15:59:18 +00:00
|
|
|
utils::{generics, Generics},
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
AdtId, Binders, CallableDefId, FnPointer, FnSig, FnSubst, Interner, Rawness, Scalar,
|
|
|
|
Substitution, TraitRef, Ty, TyBuilder, TyExt, TyKind,
|
2019-10-12 15:39:20 +00:00
|
|
|
};
|
|
|
|
|
2020-05-08 15:59:58 +00:00
|
|
|
use super::{
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
coerce::auto_deref_adjust_steps, find_breakable, BindingMode, BreakableContext, Diverges,
|
|
|
|
Expectation, InferenceContext, InferenceDiagnostic, TypeMismatch,
|
2020-05-08 15:59:58 +00:00
|
|
|
};
|
2019-11-21 12:39:09 +00:00
|
|
|
|
2020-03-13 15:05:46 +00:00
|
|
|
impl<'a> InferenceContext<'a> {
|
2019-10-12 15:39:20 +00:00
|
|
|
pub(super) fn infer_expr(&mut self, tgt_expr: ExprId, expected: &Expectation) -> Ty {
|
|
|
|
let ty = self.infer_expr_inner(tgt_expr, expected);
|
2021-05-16 15:56:38 +00:00
|
|
|
if self.resolve_ty_shallow(&ty).is_never() {
|
2020-05-08 15:36:11 +00:00
|
|
|
// Any expression that produces a value of type `!` must have diverged
|
|
|
|
self.diverges = Diverges::Always;
|
|
|
|
}
|
2021-05-21 15:41:20 +00:00
|
|
|
if let Some(expected_ty) = expected.only_has_type(&mut self.table) {
|
|
|
|
let could_unify = self.unify(&ty, &expected_ty);
|
|
|
|
if !could_unify {
|
|
|
|
self.result.type_mismatches.insert(
|
|
|
|
tgt_expr.into(),
|
2021-05-26 15:34:50 +00:00
|
|
|
TypeMismatch { expected: expected_ty, actual: ty.clone() },
|
2021-05-21 15:41:20 +00:00
|
|
|
);
|
|
|
|
}
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
2021-05-16 15:56:38 +00:00
|
|
|
ty
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Infer type of expression with possibly implicit coerce to the expected type.
|
|
|
|
/// Return the type after possible coercion.
|
2019-12-20 17:27:51 +00:00
|
|
|
pub(super) fn infer_expr_coerce(&mut self, expr: ExprId, expected: &Expectation) -> Ty {
|
2021-06-13 03:54:16 +00:00
|
|
|
let ty = self.infer_expr_inner(expr, expected);
|
2021-09-03 14:00:50 +00:00
|
|
|
if let Some(target) = expected.only_has_type(&mut self.table) {
|
2021-07-08 12:16:23 +00:00
|
|
|
match self.coerce(Some(expr), &ty, &target) {
|
|
|
|
Ok(res) => res.value,
|
2021-07-06 16:05:40 +00:00
|
|
|
Err(_) => {
|
|
|
|
self.result
|
|
|
|
.type_mismatches
|
|
|
|
.insert(expr.into(), TypeMismatch { expected: target, actual: ty.clone() });
|
|
|
|
// Return actual type when type mismatch.
|
|
|
|
// This is needed for diagnostic when return type mismatch.
|
|
|
|
ty
|
|
|
|
}
|
2021-05-21 15:41:20 +00:00
|
|
|
}
|
2019-10-12 15:39:20 +00:00
|
|
|
} else {
|
2021-05-21 15:41:20 +00:00
|
|
|
ty
|
2021-09-03 14:00:50 +00:00
|
|
|
}
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn infer_expr_inner(&mut self, tgt_expr: ExprId, expected: &Expectation) -> Ty {
|
2021-05-17 17:07:10 +00:00
|
|
|
self.db.unwind_if_cancelled();
|
2021-03-29 20:15:12 +00:00
|
|
|
|
2019-10-12 15:39:20 +00:00
|
|
|
let body = Arc::clone(&self.body); // avoid borrow checker problem
|
|
|
|
let ty = match &body[tgt_expr] {
|
2021-03-13 13:44:51 +00:00
|
|
|
Expr::Missing => self.err_ty(),
|
2021-07-08 12:16:23 +00:00
|
|
|
&Expr::If { condition, then_branch, else_branch } => {
|
2021-03-13 13:44:51 +00:00
|
|
|
self.infer_expr(
|
2021-07-08 12:16:23 +00:00
|
|
|
condition,
|
2021-12-19 16:58:39 +00:00
|
|
|
&Expectation::has_type(TyKind::Scalar(Scalar::Bool).intern(Interner)),
|
2021-03-13 13:44:51 +00:00
|
|
|
);
|
2019-10-12 15:39:20 +00:00
|
|
|
|
2020-05-08 15:36:11 +00:00
|
|
|
let condition_diverges = mem::replace(&mut self.diverges, Diverges::Maybe);
|
|
|
|
let mut both_arms_diverge = Diverges::Always;
|
|
|
|
|
2021-07-08 12:16:23 +00:00
|
|
|
let result_ty = self.table.new_type_var();
|
|
|
|
let then_ty = self.infer_expr_inner(then_branch, expected);
|
2020-05-08 15:36:11 +00:00
|
|
|
both_arms_diverge &= mem::replace(&mut self.diverges, Diverges::Maybe);
|
2021-07-08 12:16:23 +00:00
|
|
|
let mut coerce = CoerceMany::new(result_ty);
|
|
|
|
coerce.coerce(self, Some(then_branch), &then_ty);
|
2019-10-12 15:39:20 +00:00
|
|
|
let else_ty = match else_branch {
|
2021-07-08 12:16:23 +00:00
|
|
|
Some(else_branch) => self.infer_expr_inner(else_branch, expected),
|
2021-04-03 18:22:59 +00:00
|
|
|
None => TyBuilder::unit(),
|
2019-10-12 15:39:20 +00:00
|
|
|
};
|
2020-05-08 15:36:11 +00:00
|
|
|
both_arms_diverge &= self.diverges;
|
2021-05-21 15:48:15 +00:00
|
|
|
// FIXME: create a synthetic `else {}` so we have something to refer to here instead of None?
|
2021-07-08 12:16:23 +00:00
|
|
|
coerce.coerce(self, else_branch, &else_ty);
|
2020-05-08 15:36:11 +00:00
|
|
|
|
|
|
|
self.diverges = condition_diverges | both_arms_diverge;
|
2019-10-12 15:39:20 +00:00
|
|
|
|
2021-07-08 12:16:23 +00:00
|
|
|
coerce.complete()
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
2022-01-23 03:59:35 +00:00
|
|
|
&Expr::Let { pat, expr } => {
|
|
|
|
let input_ty = self.infer_expr(expr, &Expectation::none());
|
|
|
|
self.infer_pat(pat, &input_ty, BindingMode::default());
|
|
|
|
TyKind::Scalar(Scalar::Bool).intern(Interner)
|
|
|
|
}
|
2021-02-10 13:41:54 +00:00
|
|
|
Expr::Block { statements, tail, label, id: _ } => {
|
2021-02-10 14:12:45 +00:00
|
|
|
let old_resolver = mem::replace(
|
|
|
|
&mut self.resolver,
|
|
|
|
resolver_for_expr(self.db.upcast(), self.owner, tgt_expr),
|
|
|
|
);
|
|
|
|
let ty = match label {
|
2021-02-10 13:41:54 +00:00
|
|
|
Some(_) => {
|
|
|
|
let break_ty = self.table.new_type_var();
|
|
|
|
self.breakables.push(BreakableContext {
|
|
|
|
may_break: false,
|
2021-07-09 12:56:49 +00:00
|
|
|
coerce: CoerceMany::new(break_ty.clone()),
|
2021-02-10 13:41:54 +00:00
|
|
|
label: label.map(|label| self.body[label].name.clone()),
|
|
|
|
});
|
2021-07-06 16:05:40 +00:00
|
|
|
let ty = self.infer_block(
|
|
|
|
tgt_expr,
|
|
|
|
statements,
|
|
|
|
*tail,
|
|
|
|
&Expectation::has_type(break_ty),
|
|
|
|
);
|
2021-02-10 13:41:54 +00:00
|
|
|
let ctxt = self.breakables.pop().expect("breakable stack broken");
|
|
|
|
if ctxt.may_break {
|
2021-07-09 12:56:49 +00:00
|
|
|
ctxt.coerce.complete()
|
2021-02-10 13:41:54 +00:00
|
|
|
} else {
|
|
|
|
ty
|
|
|
|
}
|
2020-12-11 23:03:36 +00:00
|
|
|
}
|
2021-07-06 16:05:40 +00:00
|
|
|
None => self.infer_block(tgt_expr, statements, *tail, expected),
|
2021-02-10 14:12:45 +00:00
|
|
|
};
|
|
|
|
self.resolver = old_resolver;
|
|
|
|
ty
|
2021-02-10 13:41:54 +00:00
|
|
|
}
|
2020-12-23 11:24:24 +00:00
|
|
|
Expr::Unsafe { body } | Expr::Const { body } => self.infer_expr(*body, expected),
|
2020-05-01 23:12:37 +00:00
|
|
|
Expr::TryBlock { body } => {
|
|
|
|
let _inner = self.infer_expr(*body, expected);
|
|
|
|
// FIXME should be std::result::Result<{inner}, _>
|
2021-03-13 13:44:51 +00:00
|
|
|
self.err_ty()
|
2020-05-01 23:12:37 +00:00
|
|
|
}
|
2020-09-10 12:01:23 +00:00
|
|
|
Expr::Async { body } => {
|
|
|
|
// Use the first type parameter as the output type of future.
|
2021-07-08 12:16:23 +00:00
|
|
|
// existential type AsyncBlockImplTrait<InnerType>: Future<Output = InnerType>
|
2020-09-10 12:01:23 +00:00
|
|
|
let inner_ty = self.infer_expr(*body, &Expectation::none());
|
2021-03-13 19:05:47 +00:00
|
|
|
let impl_trait_id = crate::ImplTraitId::AsyncBlockTypeImplTrait(self.owner, *body);
|
|
|
|
let opaque_ty_id = self.db.intern_impl_trait_id(impl_trait_id).into();
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::OpaqueType(opaque_ty_id, Substitution::from1(Interner, inner_ty))
|
|
|
|
.intern(Interner)
|
2020-09-10 12:01:23 +00:00
|
|
|
}
|
2020-05-31 08:59:40 +00:00
|
|
|
Expr::Loop { body, label } => {
|
2020-05-19 19:03:59 +00:00
|
|
|
self.breakables.push(BreakableContext {
|
|
|
|
may_break: false,
|
2021-07-09 12:56:49 +00:00
|
|
|
coerce: CoerceMany::new(self.table.new_type_var()),
|
2020-12-23 15:34:30 +00:00
|
|
|
label: label.map(|label| self.body[label].name.clone()),
|
2020-05-19 19:03:59 +00:00
|
|
|
});
|
2021-04-03 18:22:59 +00:00
|
|
|
self.infer_expr(*body, &Expectation::has_type(TyBuilder::unit()));
|
2020-05-08 15:59:58 +00:00
|
|
|
|
|
|
|
let ctxt = self.breakables.pop().expect("breakable stack broken");
|
2020-05-19 19:03:59 +00:00
|
|
|
|
2020-05-08 15:59:58 +00:00
|
|
|
if ctxt.may_break {
|
2021-07-09 12:56:49 +00:00
|
|
|
self.diverges = Diverges::Maybe;
|
|
|
|
ctxt.coerce.complete()
|
2020-05-08 15:59:58 +00:00
|
|
|
} else {
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Never.intern(Interner)
|
2020-05-08 15:59:58 +00:00
|
|
|
}
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
2020-05-31 08:59:40 +00:00
|
|
|
Expr::While { condition, body, label } => {
|
|
|
|
self.breakables.push(BreakableContext {
|
|
|
|
may_break: false,
|
2021-07-09 12:56:49 +00:00
|
|
|
coerce: CoerceMany::new(self.err_ty()),
|
2020-12-23 15:34:30 +00:00
|
|
|
label: label.map(|label| self.body[label].name.clone()),
|
2020-05-31 08:59:40 +00:00
|
|
|
});
|
2021-03-13 13:44:51 +00:00
|
|
|
self.infer_expr(
|
|
|
|
*condition,
|
2021-12-19 16:58:39 +00:00
|
|
|
&Expectation::has_type(TyKind::Scalar(Scalar::Bool).intern(Interner)),
|
2021-03-13 13:44:51 +00:00
|
|
|
);
|
2021-04-03 18:22:59 +00:00
|
|
|
self.infer_expr(*body, &Expectation::has_type(TyBuilder::unit()));
|
2020-05-08 15:59:58 +00:00
|
|
|
let _ctxt = self.breakables.pop().expect("breakable stack broken");
|
|
|
|
// the body may not run, so it diverging doesn't mean we diverge
|
|
|
|
self.diverges = Diverges::Maybe;
|
2021-04-03 18:22:59 +00:00
|
|
|
TyBuilder::unit()
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
2020-05-31 08:59:40 +00:00
|
|
|
Expr::For { iterable, body, pat, label } => {
|
2019-10-12 15:39:20 +00:00
|
|
|
let iterable_ty = self.infer_expr(*iterable, &Expectation::none());
|
|
|
|
|
2020-05-31 08:59:40 +00:00
|
|
|
self.breakables.push(BreakableContext {
|
|
|
|
may_break: false,
|
2021-07-09 12:56:49 +00:00
|
|
|
coerce: CoerceMany::new(self.err_ty()),
|
2020-12-23 15:34:30 +00:00
|
|
|
label: label.map(|label| self.body[label].name.clone()),
|
2020-05-31 08:59:40 +00:00
|
|
|
});
|
2019-12-13 11:44:07 +00:00
|
|
|
let pat_ty =
|
|
|
|
self.resolve_associated_type(iterable_ty, self.resolve_into_iter_item());
|
2019-10-12 15:39:20 +00:00
|
|
|
|
|
|
|
self.infer_pat(*pat, &pat_ty, BindingMode::default());
|
2020-05-08 15:59:58 +00:00
|
|
|
|
2021-04-03 18:22:59 +00:00
|
|
|
self.infer_expr(*body, &Expectation::has_type(TyBuilder::unit()));
|
2020-05-08 15:59:58 +00:00
|
|
|
let _ctxt = self.breakables.pop().expect("breakable stack broken");
|
|
|
|
// the body may not run, so it diverging doesn't mean we diverge
|
|
|
|
self.diverges = Diverges::Maybe;
|
2021-04-03 18:22:59 +00:00
|
|
|
TyBuilder::unit()
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
2019-12-20 15:41:32 +00:00
|
|
|
Expr::Lambda { body, args, ret_type, arg_types } => {
|
2019-10-12 15:39:20 +00:00
|
|
|
assert_eq!(args.len(), arg_types.len());
|
|
|
|
|
|
|
|
let mut sig_tys = Vec::new();
|
|
|
|
|
2020-05-29 15:35:57 +00:00
|
|
|
// collect explicitly written argument types
|
|
|
|
for arg_type in arg_types.iter() {
|
2021-10-03 12:53:01 +00:00
|
|
|
let arg_ty = match arg_type {
|
|
|
|
Some(type_ref) => self.make_ty(type_ref),
|
|
|
|
None => self.table.new_type_var(),
|
2019-10-12 15:39:20 +00:00
|
|
|
};
|
|
|
|
sig_tys.push(arg_ty);
|
|
|
|
}
|
|
|
|
|
|
|
|
// add return type
|
2019-12-20 15:41:32 +00:00
|
|
|
let ret_ty = match ret_type {
|
|
|
|
Some(type_ref) => self.make_ty(type_ref),
|
|
|
|
None => self.table.new_type_var(),
|
|
|
|
};
|
2019-10-12 15:39:20 +00:00
|
|
|
sig_tys.push(ret_ty.clone());
|
2021-03-13 13:44:51 +00:00
|
|
|
let sig_ty = TyKind::Function(FnPointer {
|
2021-04-05 20:23:16 +00:00
|
|
|
num_binders: 0,
|
2021-03-14 15:30:02 +00:00
|
|
|
sig: FnSig { abi: (), safety: chalk_ir::Safety::Safe, variadic: false },
|
2021-04-05 20:32:58 +00:00
|
|
|
substitution: FnSubst(
|
2021-12-19 16:58:39 +00:00
|
|
|
Substitution::from_iter(Interner, sig_tys.clone()).shifted_in(Interner),
|
2021-04-05 20:32:58 +00:00
|
|
|
),
|
2021-03-13 13:44:51 +00:00
|
|
|
})
|
2021-12-19 16:58:39 +00:00
|
|
|
.intern(Interner);
|
2021-03-13 18:27:09 +00:00
|
|
|
let closure_id = self.db.intern_closure((self.owner, tgt_expr)).into();
|
2021-03-13 13:44:51 +00:00
|
|
|
let closure_ty =
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Closure(closure_id, Substitution::from1(Interner, sig_ty.clone()))
|
|
|
|
.intern(Interner);
|
2019-10-12 15:39:20 +00:00
|
|
|
|
|
|
|
// Eagerly try to relate the closure type with the expected
|
|
|
|
// type, otherwise we often won't have enough information to
|
|
|
|
// infer the body.
|
2021-07-06 16:05:40 +00:00
|
|
|
self.deduce_closure_type_from_expectations(
|
|
|
|
tgt_expr,
|
|
|
|
&closure_ty,
|
|
|
|
&sig_ty,
|
|
|
|
expected,
|
|
|
|
);
|
2019-10-12 15:39:20 +00:00
|
|
|
|
2020-05-29 15:35:57 +00:00
|
|
|
// Now go through the argument patterns
|
|
|
|
for (arg_pat, arg_ty) in args.iter().zip(sig_tys) {
|
2021-05-16 15:56:38 +00:00
|
|
|
self.infer_pat(*arg_pat, &arg_ty, BindingMode::default());
|
2020-05-29 15:35:57 +00:00
|
|
|
}
|
|
|
|
|
2020-05-08 15:36:11 +00:00
|
|
|
let prev_diverges = mem::replace(&mut self.diverges, Diverges::Maybe);
|
|
|
|
let prev_ret_ty = mem::replace(&mut self.return_ty, ret_ty.clone());
|
2019-12-20 15:41:32 +00:00
|
|
|
|
|
|
|
self.infer_expr_coerce(*body, &Expectation::has_type(ret_ty));
|
|
|
|
|
2020-05-08 15:36:11 +00:00
|
|
|
self.diverges = prev_diverges;
|
2019-12-20 15:41:32 +00:00
|
|
|
self.return_ty = prev_ret_ty;
|
|
|
|
|
2019-10-12 15:39:20 +00:00
|
|
|
closure_ty
|
|
|
|
}
|
|
|
|
Expr::Call { callee, args } => {
|
|
|
|
let callee_ty = self.infer_expr(*callee, &Expectation::none());
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
let mut derefs = Autoderef::new(&mut self.table, callee_ty.clone());
|
|
|
|
let mut res = None;
|
|
|
|
// manual loop to be able to access `derefs.table`
|
|
|
|
while let Some((callee_deref_ty, _)) = derefs.next() {
|
|
|
|
res = derefs.table.callable_sig(&callee_deref_ty, args.len());
|
|
|
|
if res.is_some() {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2021-07-09 17:12:56 +00:00
|
|
|
let (param_tys, ret_ty): (Vec<Ty>, Ty) = match res {
|
|
|
|
Some(res) => {
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
let adjustments = auto_deref_adjust_steps(&derefs);
|
|
|
|
self.write_expr_adj(*callee, adjustments);
|
2021-07-09 17:12:56 +00:00
|
|
|
res
|
|
|
|
}
|
|
|
|
None => (Vec::new(), self.err_ty()),
|
|
|
|
};
|
2019-10-12 15:39:20 +00:00
|
|
|
self.register_obligations_for_call(&callee_ty);
|
2021-07-15 18:02:58 +00:00
|
|
|
|
|
|
|
let expected_inputs = self.expected_inputs_for_expected_output(
|
|
|
|
expected,
|
|
|
|
ret_ty.clone(),
|
|
|
|
param_tys.clone(),
|
|
|
|
);
|
|
|
|
|
|
|
|
self.check_call_arguments(args, &expected_inputs, ¶m_tys);
|
2020-02-18 13:32:19 +00:00
|
|
|
self.normalize_associated_types_in(ret_ty)
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
Expr::MethodCall { receiver, args, method_name, generic_args } => self
|
2021-07-15 18:02:58 +00:00
|
|
|
.infer_method_call(
|
|
|
|
tgt_expr,
|
|
|
|
*receiver,
|
|
|
|
args,
|
|
|
|
method_name,
|
|
|
|
generic_args.as_deref(),
|
|
|
|
expected,
|
|
|
|
),
|
2019-10-12 15:39:20 +00:00
|
|
|
Expr::Match { expr, arms } => {
|
|
|
|
let input_ty = self.infer_expr(*expr, &Expectation::none());
|
|
|
|
|
2021-06-03 21:12:35 +00:00
|
|
|
let expected = expected.adjust_for_branches(&mut self.table);
|
|
|
|
|
2021-07-08 12:16:23 +00:00
|
|
|
let result_ty = if arms.is_empty() {
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Never.intern(Interner)
|
2021-03-13 13:44:51 +00:00
|
|
|
} else {
|
2021-06-03 21:12:35 +00:00
|
|
|
match &expected {
|
|
|
|
Expectation::HasType(ty) => ty.clone(),
|
|
|
|
_ => self.table.new_type_var(),
|
|
|
|
}
|
2021-03-13 13:44:51 +00:00
|
|
|
};
|
2021-07-08 12:16:23 +00:00
|
|
|
let mut coerce = CoerceMany::new(result_ty);
|
2019-10-12 15:39:20 +00:00
|
|
|
|
2020-05-08 15:36:11 +00:00
|
|
|
let matchee_diverges = self.diverges;
|
|
|
|
let mut all_arms_diverge = Diverges::Always;
|
|
|
|
|
2021-11-20 15:00:45 +00:00
|
|
|
for arm in arms.iter() {
|
2020-05-08 15:36:11 +00:00
|
|
|
self.diverges = Diverges::Maybe;
|
2020-02-09 18:57:01 +00:00
|
|
|
let _pat_ty = self.infer_pat(arm.pat, &input_ty, BindingMode::default());
|
2022-01-23 03:59:35 +00:00
|
|
|
if let Some(guard_expr) = arm.guard {
|
|
|
|
self.infer_expr(
|
|
|
|
guard_expr,
|
|
|
|
&Expectation::has_type(TyKind::Scalar(Scalar::Bool).intern(Interner)),
|
|
|
|
);
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
let arm_ty = self.infer_expr_inner(arm.expr, &expected);
|
2020-05-08 15:36:11 +00:00
|
|
|
all_arms_diverge &= self.diverges;
|
2021-07-08 12:16:23 +00:00
|
|
|
coerce.coerce(self, Some(arm.expr), &arm_ty);
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
|
2020-05-08 15:36:11 +00:00
|
|
|
self.diverges = matchee_diverges | all_arms_diverge;
|
|
|
|
|
2021-07-08 12:16:23 +00:00
|
|
|
coerce.complete()
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
Expr::Path(p) => {
|
|
|
|
// FIXME this could be more efficient...
|
2020-03-13 15:05:46 +00:00
|
|
|
let resolver = resolver_for_expr(self.db.upcast(), self.owner, tgt_expr);
|
2021-06-18 11:40:51 +00:00
|
|
|
self.infer_path(&resolver, p, tgt_expr.into()).unwrap_or_else(|| self.err_ty())
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
2021-12-19 16:58:39 +00:00
|
|
|
Expr::Continue { .. } => TyKind::Never.intern(Interner),
|
2020-05-31 08:59:40 +00:00
|
|
|
Expr::Break { expr, label } => {
|
2021-07-09 12:56:49 +00:00
|
|
|
let mut coerce = match find_breakable(&mut self.breakables, label.as_ref()) {
|
|
|
|
Some(ctxt) => {
|
|
|
|
// avoiding the borrowck
|
|
|
|
mem::replace(
|
|
|
|
&mut ctxt.coerce,
|
|
|
|
CoerceMany::new(self.result.standard_types.unknown.clone()),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
None => CoerceMany::new(self.result.standard_types.unknown.clone()),
|
|
|
|
};
|
2020-05-19 20:52:15 +00:00
|
|
|
|
2021-07-09 12:56:49 +00:00
|
|
|
let val_ty = if let Some(expr) = *expr {
|
2021-07-08 12:16:23 +00:00
|
|
|
self.infer_expr(expr, &Expectation::none())
|
2021-05-21 15:48:15 +00:00
|
|
|
} else {
|
|
|
|
TyBuilder::unit()
|
|
|
|
};
|
|
|
|
|
|
|
|
// FIXME: create a synthetic `()` during lowering so we have something to refer to here?
|
2021-07-09 12:56:49 +00:00
|
|
|
coerce.coerce(self, *expr, &val_ty);
|
2020-05-18 21:39:10 +00:00
|
|
|
|
2020-05-31 08:59:40 +00:00
|
|
|
if let Some(ctxt) = find_breakable(&mut self.breakables, label.as_ref()) {
|
2021-07-09 12:56:49 +00:00
|
|
|
ctxt.coerce = coerce;
|
2020-05-19 20:52:15 +00:00
|
|
|
ctxt.may_break = true;
|
2020-05-08 17:48:03 +00:00
|
|
|
} else {
|
|
|
|
self.push_diagnostic(InferenceDiagnostic::BreakOutsideOfLoop {
|
|
|
|
expr: tgt_expr,
|
|
|
|
});
|
2021-07-09 12:56:49 +00:00
|
|
|
};
|
|
|
|
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Never.intern(Interner)
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
Expr::Return { expr } => {
|
|
|
|
if let Some(expr) = expr {
|
2019-12-05 22:02:31 +00:00
|
|
|
self.infer_expr_coerce(*expr, &Expectation::has_type(self.return_ty.clone()));
|
2019-12-20 15:41:32 +00:00
|
|
|
} else {
|
2021-04-03 18:22:59 +00:00
|
|
|
let unit = TyBuilder::unit();
|
2021-07-08 12:16:23 +00:00
|
|
|
let _ = self.coerce(Some(tgt_expr), &unit, &self.return_ty.clone());
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Never.intern(Interner)
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
2021-01-13 15:01:50 +00:00
|
|
|
Expr::Yield { expr } => {
|
|
|
|
// FIXME: track yield type for coercion
|
|
|
|
if let Some(expr) = expr {
|
|
|
|
self.infer_expr(*expr, &Expectation::none());
|
|
|
|
}
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Never.intern(Interner)
|
2021-01-13 15:01:50 +00:00
|
|
|
}
|
2019-10-12 15:39:20 +00:00
|
|
|
Expr::RecordLit { path, fields, spread } => {
|
2021-07-22 11:40:18 +00:00
|
|
|
let (ty, def_id) = self.resolve_variant(path.as_deref(), false);
|
2019-10-12 15:39:20 +00:00
|
|
|
if let Some(variant) = def_id {
|
|
|
|
self.write_variant_resolution(tgt_expr.into(), variant);
|
|
|
|
}
|
|
|
|
|
2021-05-21 15:41:20 +00:00
|
|
|
if let Some(t) = expected.only_has_type(&mut self.table) {
|
|
|
|
self.unify(&ty, &t);
|
|
|
|
}
|
2019-10-12 15:39:20 +00:00
|
|
|
|
2021-04-07 15:26:01 +00:00
|
|
|
let substs = ty
|
|
|
|
.as_adt()
|
|
|
|
.map(|(_, s)| s.clone())
|
2021-12-19 16:58:39 +00:00
|
|
|
.unwrap_or_else(|| Substitution::empty(Interner));
|
2020-02-18 12:53:02 +00:00
|
|
|
let field_types = def_id.map(|it| self.db.field_types(it)).unwrap_or_default();
|
2021-04-06 15:59:18 +00:00
|
|
|
let variant_data = def_id.map(|it| it.variant_data(self.db.upcast()));
|
2021-03-15 12:38:50 +00:00
|
|
|
for field in fields.iter() {
|
2019-11-27 13:25:01 +00:00
|
|
|
let field_def =
|
|
|
|
variant_data.as_ref().and_then(|it| match it.field(&field.name) {
|
2020-04-25 12:23:34 +00:00
|
|
|
Some(local_id) => Some(FieldId { parent: def_id.unwrap(), local_id }),
|
2019-11-27 13:25:01 +00:00
|
|
|
None => {
|
|
|
|
self.push_diagnostic(InferenceDiagnostic::NoSuchField {
|
2021-03-15 12:38:50 +00:00
|
|
|
expr: field.expr,
|
2019-11-27 13:25:01 +00:00
|
|
|
});
|
|
|
|
None
|
|
|
|
}
|
|
|
|
});
|
2021-03-13 13:44:51 +00:00
|
|
|
let field_ty = field_def.map_or(self.err_ty(), |it| {
|
2021-12-19 16:58:39 +00:00
|
|
|
field_types[it.local_id].clone().substitute(Interner, &substs)
|
2021-03-13 13:44:51 +00:00
|
|
|
});
|
2019-10-12 15:39:20 +00:00
|
|
|
self.infer_expr_coerce(field.expr, &Expectation::has_type(field_ty));
|
|
|
|
}
|
|
|
|
if let Some(expr) = spread {
|
|
|
|
self.infer_expr(*expr, &Expectation::has_type(ty.clone()));
|
|
|
|
}
|
|
|
|
ty
|
|
|
|
}
|
|
|
|
Expr::Field { expr, name } => {
|
2019-12-05 22:02:31 +00:00
|
|
|
let receiver_ty = self.infer_expr_inner(*expr, &Expectation::none());
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
|
|
|
|
let mut autoderef = Autoderef::new(&mut self.table, receiver_ty);
|
2021-07-09 17:12:56 +00:00
|
|
|
let ty = autoderef.by_ref().find_map(|(derefed_ty, _)| {
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
let (field_id, parameters) = match derefed_ty.kind(Interner) {
|
|
|
|
TyKind::Tuple(_, substs) => {
|
|
|
|
return name.as_tuple_index().and_then(|idx| {
|
|
|
|
substs
|
|
|
|
.as_slice(Interner)
|
|
|
|
.get(idx)
|
|
|
|
.map(|a| a.assert_ty_ref(Interner))
|
|
|
|
.cloned()
|
|
|
|
});
|
|
|
|
}
|
2021-03-13 13:44:51 +00:00
|
|
|
TyKind::Adt(AdtId(hir_def::AdtId::StructId(s)), parameters) => {
|
2021-03-20 18:07:23 +00:00
|
|
|
let local_id = self.db.struct_data(*s).variant_data.field(name)?;
|
|
|
|
let field = FieldId { parent: (*s).into(), local_id };
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
(field, parameters.clone())
|
2021-03-13 13:44:51 +00:00
|
|
|
}
|
|
|
|
TyKind::Adt(AdtId(hir_def::AdtId::UnionId(u)), parameters) => {
|
2021-03-20 18:07:23 +00:00
|
|
|
let local_id = self.db.union_data(*u).variant_data.field(name)?;
|
|
|
|
let field = FieldId { parent: (*u).into(), local_id };
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
(field, parameters.clone())
|
2021-03-13 13:44:51 +00:00
|
|
|
}
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
_ => return None,
|
|
|
|
};
|
|
|
|
let module = self.resolver.module();
|
|
|
|
let is_visible = module
|
|
|
|
.map(|mod_id| {
|
|
|
|
self.db.field_visibilities(field_id.parent)[field_id.local_id]
|
|
|
|
.is_visible_from(self.db.upcast(), mod_id)
|
|
|
|
})
|
|
|
|
.unwrap_or(true);
|
|
|
|
if !is_visible {
|
|
|
|
// Write down the first field resolution even if it is not visible
|
|
|
|
// This aids IDE features for private fields like goto def and in
|
|
|
|
// case of autoderef finding an applicable field, this will be
|
|
|
|
// overwritten in a following cycle
|
|
|
|
if let Entry::Vacant(entry) = self.result.field_resolutions.entry(tgt_expr)
|
|
|
|
{
|
|
|
|
entry.insert(field_id);
|
|
|
|
}
|
|
|
|
return None;
|
2021-02-28 18:13:37 +00:00
|
|
|
}
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
// can't have `write_field_resolution` here because `self.table` is borrowed :(
|
|
|
|
self.result.field_resolutions.insert(tgt_expr, field_id);
|
|
|
|
let ty = self.db.field_types(field_id.parent)[field_id.local_id]
|
|
|
|
.clone()
|
|
|
|
.substitute(Interner, ¶meters);
|
|
|
|
Some(ty)
|
2021-07-09 17:12:56 +00:00
|
|
|
});
|
|
|
|
let ty = match ty {
|
|
|
|
Some(ty) => {
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
let adjustments = auto_deref_adjust_steps(&autoderef);
|
|
|
|
self.write_expr_adj(*expr, adjustments);
|
|
|
|
let ty = self.insert_type_vars(ty);
|
|
|
|
let ty = self.normalize_associated_types_in(ty);
|
2021-07-09 17:12:56 +00:00
|
|
|
ty
|
|
|
|
}
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
_ => self.err_ty(),
|
2021-07-09 17:12:56 +00:00
|
|
|
};
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
ty
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
Expr::Await { expr } => {
|
2019-12-05 22:02:31 +00:00
|
|
|
let inner_ty = self.infer_expr_inner(*expr, &Expectation::none());
|
2020-02-18 13:32:19 +00:00
|
|
|
self.resolve_associated_type(inner_ty, self.resolve_future_future_output())
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
Expr::Try { expr } => {
|
2019-12-05 22:02:31 +00:00
|
|
|
let inner_ty = self.infer_expr_inner(*expr, &Expectation::none());
|
2020-02-18 13:32:19 +00:00
|
|
|
self.resolve_associated_type(inner_ty, self.resolve_ops_try_ok())
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
Expr::Cast { expr, type_ref } => {
|
2021-05-21 15:41:20 +00:00
|
|
|
// FIXME: propagate the "castable to" expectation (and find a test case that shows this is necessary)
|
2019-12-05 22:02:31 +00:00
|
|
|
let _inner_ty = self.infer_expr_inner(*expr, &Expectation::none());
|
2019-10-12 15:39:20 +00:00
|
|
|
let cast_ty = self.make_ty(type_ref);
|
|
|
|
// FIXME check the cast...
|
|
|
|
cast_ty
|
|
|
|
}
|
2020-05-28 19:42:22 +00:00
|
|
|
Expr::Ref { expr, rawness, mutability } => {
|
2021-03-01 18:30:34 +00:00
|
|
|
let mutability = lower_to_chalk_mutability(*mutability);
|
2021-05-21 15:41:20 +00:00
|
|
|
let expectation = if let Some((exp_inner, exp_rawness, exp_mutability)) = expected
|
|
|
|
.only_has_type(&mut self.table)
|
|
|
|
.as_ref()
|
|
|
|
.and_then(|t| t.as_reference_or_ptr())
|
2020-05-28 19:42:22 +00:00
|
|
|
{
|
2021-05-21 15:41:20 +00:00
|
|
|
if exp_mutability == Mutability::Mut && mutability == Mutability::Not {
|
2021-05-16 15:56:38 +00:00
|
|
|
// FIXME: record type error - expected mut reference but found shared ref,
|
2020-05-28 19:42:22 +00:00
|
|
|
// which cannot be coerced
|
|
|
|
}
|
2021-05-21 15:41:20 +00:00
|
|
|
if exp_rawness == Rawness::Ref && *rawness == Rawness::RawPtr {
|
2021-05-16 15:56:38 +00:00
|
|
|
// FIXME: record type error - expected reference but found ptr,
|
2020-05-28 19:42:22 +00:00
|
|
|
// which cannot be coerced
|
|
|
|
}
|
2021-07-15 18:02:58 +00:00
|
|
|
Expectation::rvalue_hint(&mut self.table, Ty::clone(exp_inner))
|
2020-05-28 19:42:22 +00:00
|
|
|
} else {
|
|
|
|
Expectation::none()
|
|
|
|
};
|
2019-12-05 22:02:31 +00:00
|
|
|
let inner_ty = self.infer_expr_inner(*expr, &expectation);
|
2021-02-28 18:13:37 +00:00
|
|
|
match rawness {
|
2021-03-14 16:40:55 +00:00
|
|
|
Rawness::RawPtr => TyKind::Raw(mutability, inner_ty),
|
2021-04-06 08:50:55 +00:00
|
|
|
Rawness::Ref => TyKind::Ref(mutability, static_lifetime(), inner_ty),
|
2021-02-28 18:13:37 +00:00
|
|
|
}
|
2021-12-19 16:58:39 +00:00
|
|
|
.intern(Interner)
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
Expr::Box { expr } => {
|
2019-12-05 22:02:31 +00:00
|
|
|
let inner_ty = self.infer_expr_inner(*expr, &Expectation::none());
|
2019-10-12 15:39:20 +00:00
|
|
|
if let Some(box_) = self.resolve_boxed_box() {
|
2021-04-03 19:29:49 +00:00
|
|
|
TyBuilder::adt(self.db, box_)
|
|
|
|
.push(inner_ty)
|
|
|
|
.fill_with_defaults(self.db, || self.table.new_type_var())
|
|
|
|
.build()
|
2019-10-12 15:39:20 +00:00
|
|
|
} else {
|
2021-03-13 13:44:51 +00:00
|
|
|
self.err_ty()
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
Expr::UnaryOp { expr, op } => {
|
2019-12-05 22:02:31 +00:00
|
|
|
let inner_ty = self.infer_expr_inner(*expr, &Expectation::none());
|
2021-05-16 15:56:38 +00:00
|
|
|
let inner_ty = self.resolve_ty_shallow(&inner_ty);
|
2019-10-12 15:39:20 +00:00
|
|
|
match op {
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
UnaryOp::Deref => {
|
|
|
|
autoderef::deref(&mut self.table, inner_ty).unwrap_or_else(|| self.err_ty())
|
|
|
|
}
|
2019-10-12 15:39:20 +00:00
|
|
|
UnaryOp::Neg => {
|
2021-12-19 16:58:39 +00:00
|
|
|
match inner_ty.kind(Interner) {
|
2019-12-13 11:44:42 +00:00
|
|
|
// Fast path for builtins
|
2021-06-17 15:37:14 +00:00
|
|
|
TyKind::Scalar(Scalar::Int(_) | Scalar::Uint(_) | Scalar::Float(_))
|
|
|
|
| TyKind::InferenceVar(
|
|
|
|
_,
|
|
|
|
TyVariableKind::Integer | TyVariableKind::Float,
|
|
|
|
) => inner_ty,
|
2019-12-13 11:44:42 +00:00
|
|
|
// Otherwise we resolve via the std::ops::Neg trait
|
|
|
|
_ => self
|
|
|
|
.resolve_associated_type(inner_ty, self.resolve_ops_neg_output()),
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
UnaryOp::Not => {
|
2021-12-19 16:58:39 +00:00
|
|
|
match inner_ty.kind(Interner) {
|
2019-12-13 11:44:42 +00:00
|
|
|
// Fast path for builtins
|
2021-06-17 15:37:14 +00:00
|
|
|
TyKind::Scalar(Scalar::Bool | Scalar::Int(_) | Scalar::Uint(_))
|
2021-03-13 13:44:51 +00:00
|
|
|
| TyKind::InferenceVar(_, TyVariableKind::Integer) => inner_ty,
|
2019-12-13 11:44:42 +00:00
|
|
|
// Otherwise we resolve via the std::ops::Not trait
|
|
|
|
_ => self
|
|
|
|
.resolve_associated_type(inner_ty, self.resolve_ops_not_output()),
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Expr::BinaryOp { lhs, rhs, op } => match op {
|
2021-08-22 15:21:47 +00:00
|
|
|
Some(BinaryOp::Assignment { op: None }) => {
|
|
|
|
let lhs_ty = self.infer_expr(*lhs, &Expectation::none());
|
|
|
|
self.infer_expr_coerce(*rhs, &Expectation::has_type(lhs_ty));
|
|
|
|
self.result.standard_types.unit.clone()
|
|
|
|
}
|
|
|
|
Some(BinaryOp::LogicOp(_)) => {
|
|
|
|
let bool_ty = self.result.standard_types.bool_.clone();
|
|
|
|
self.infer_expr_coerce(*lhs, &Expectation::HasType(bool_ty.clone()));
|
|
|
|
let lhs_diverges = self.diverges;
|
|
|
|
self.infer_expr_coerce(*rhs, &Expectation::HasType(bool_ty.clone()));
|
|
|
|
// Depending on the LHS' value, the RHS can never execute.
|
|
|
|
self.diverges = lhs_diverges;
|
|
|
|
bool_ty
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
2021-08-22 15:21:47 +00:00
|
|
|
Some(op) => self.infer_overloadable_binop(*lhs, *op, *rhs, tgt_expr),
|
2021-03-13 13:44:51 +00:00
|
|
|
_ => self.err_ty(),
|
2019-10-12 15:39:20 +00:00
|
|
|
},
|
2019-11-29 06:49:12 +00:00
|
|
|
Expr::Range { lhs, rhs, range_type } => {
|
2019-12-05 22:02:31 +00:00
|
|
|
let lhs_ty = lhs.map(|e| self.infer_expr_inner(e, &Expectation::none()));
|
2019-11-29 06:49:12 +00:00
|
|
|
let rhs_expect = lhs_ty
|
|
|
|
.as_ref()
|
|
|
|
.map_or_else(Expectation::none, |ty| Expectation::has_type(ty.clone()));
|
|
|
|
let rhs_ty = rhs.map(|e| self.infer_expr(e, &rhs_expect));
|
|
|
|
match (range_type, lhs_ty, rhs_ty) {
|
|
|
|
(RangeOp::Exclusive, None, None) => match self.resolve_range_full() {
|
2021-04-03 19:29:49 +00:00
|
|
|
Some(adt) => TyBuilder::adt(self.db, adt).build(),
|
2021-03-13 13:44:51 +00:00
|
|
|
None => self.err_ty(),
|
2019-11-29 06:49:12 +00:00
|
|
|
},
|
|
|
|
(RangeOp::Exclusive, None, Some(ty)) => match self.resolve_range_to() {
|
2021-04-03 19:29:49 +00:00
|
|
|
Some(adt) => TyBuilder::adt(self.db, adt).push(ty).build(),
|
2021-03-13 13:44:51 +00:00
|
|
|
None => self.err_ty(),
|
2019-11-29 06:49:12 +00:00
|
|
|
},
|
|
|
|
(RangeOp::Inclusive, None, Some(ty)) => {
|
|
|
|
match self.resolve_range_to_inclusive() {
|
2021-04-03 19:29:49 +00:00
|
|
|
Some(adt) => TyBuilder::adt(self.db, adt).push(ty).build(),
|
2021-03-13 13:44:51 +00:00
|
|
|
None => self.err_ty(),
|
2019-11-29 06:49:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
(RangeOp::Exclusive, Some(_), Some(ty)) => match self.resolve_range() {
|
2021-04-03 19:29:49 +00:00
|
|
|
Some(adt) => TyBuilder::adt(self.db, adt).push(ty).build(),
|
2021-03-13 13:44:51 +00:00
|
|
|
None => self.err_ty(),
|
2019-11-29 06:49:12 +00:00
|
|
|
},
|
|
|
|
(RangeOp::Inclusive, Some(_), Some(ty)) => {
|
|
|
|
match self.resolve_range_inclusive() {
|
2021-04-03 19:29:49 +00:00
|
|
|
Some(adt) => TyBuilder::adt(self.db, adt).push(ty).build(),
|
2021-03-13 13:44:51 +00:00
|
|
|
None => self.err_ty(),
|
2019-11-29 06:49:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
(RangeOp::Exclusive, Some(ty), None) => match self.resolve_range_from() {
|
2021-04-03 19:29:49 +00:00
|
|
|
Some(adt) => TyBuilder::adt(self.db, adt).push(ty).build(),
|
2021-03-13 13:44:51 +00:00
|
|
|
None => self.err_ty(),
|
2019-11-29 06:49:12 +00:00
|
|
|
},
|
2021-03-13 13:44:51 +00:00
|
|
|
(RangeOp::Inclusive, _, None) => self.err_ty(),
|
2019-11-28 19:10:16 +00:00
|
|
|
}
|
|
|
|
}
|
2019-10-12 15:39:20 +00:00
|
|
|
Expr::Index { base, index } => {
|
2019-12-19 04:45:07 +00:00
|
|
|
let base_ty = self.infer_expr_inner(*base, &Expectation::none());
|
|
|
|
let index_ty = self.infer_expr(*index, &Expectation::none());
|
|
|
|
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
if let Some(index_trait) = self.resolve_ops_index() {
|
|
|
|
let canonicalized = self.canonicalize(base_ty.clone());
|
|
|
|
let receiver_adjustments = method_resolution::resolve_indexing_op(
|
2020-02-29 21:48:23 +00:00
|
|
|
self.db,
|
|
|
|
self.trait_env.clone(),
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
canonicalized.value,
|
2020-02-29 21:48:23 +00:00
|
|
|
index_trait,
|
|
|
|
);
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
let (self_ty, adj) = receiver_adjustments
|
|
|
|
.map_or((self.err_ty(), Vec::new()), |adj| {
|
|
|
|
adj.apply(&mut self.table, base_ty)
|
|
|
|
});
|
|
|
|
self.write_expr_adj(*base, adj);
|
2020-02-29 21:48:23 +00:00
|
|
|
self.resolve_associated_type_with_params(
|
|
|
|
self_ty,
|
|
|
|
self.resolve_ops_index_output(),
|
|
|
|
&[index_ty],
|
|
|
|
)
|
|
|
|
} else {
|
2021-03-13 13:44:51 +00:00
|
|
|
self.err_ty()
|
2020-02-29 21:48:23 +00:00
|
|
|
}
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
Expr::Tuple { exprs } => {
|
2021-05-21 15:41:20 +00:00
|
|
|
let mut tys = match expected
|
|
|
|
.only_has_type(&mut self.table)
|
|
|
|
.as_ref()
|
2021-12-19 16:58:39 +00:00
|
|
|
.map(|t| t.kind(Interner))
|
2021-05-21 15:41:20 +00:00
|
|
|
{
|
|
|
|
Some(TyKind::Tuple(_, substs)) => substs
|
2021-12-19 16:58:39 +00:00
|
|
|
.iter(Interner)
|
|
|
|
.map(|a| a.assert_ty_ref(Interner).clone())
|
2019-12-01 19:30:28 +00:00
|
|
|
.chain(repeat_with(|| self.table.new_type_var()))
|
2019-10-12 15:39:20 +00:00
|
|
|
.take(exprs.len())
|
|
|
|
.collect::<Vec<_>>(),
|
2019-12-01 19:30:28 +00:00
|
|
|
_ => (0..exprs.len()).map(|_| self.table.new_type_var()).collect(),
|
2019-10-12 15:39:20 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
for (expr, ty) in exprs.iter().zip(tys.iter_mut()) {
|
|
|
|
self.infer_expr_coerce(*expr, &Expectation::has_type(ty.clone()));
|
|
|
|
}
|
|
|
|
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Tuple(tys.len(), Substitution::from_iter(Interner, tys)).intern(Interner)
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
Expr::Array(array) => {
|
2021-07-08 12:16:23 +00:00
|
|
|
let elem_ty =
|
2021-12-19 16:58:39 +00:00
|
|
|
match expected.to_option(&mut self.table).as_ref().map(|t| t.kind(Interner)) {
|
2021-06-17 15:37:14 +00:00
|
|
|
Some(TyKind::Array(st, _) | TyKind::Slice(st)) => st.clone(),
|
2021-05-21 15:41:20 +00:00
|
|
|
_ => self.table.new_type_var(),
|
|
|
|
};
|
2021-07-08 12:16:23 +00:00
|
|
|
let mut coerce = CoerceMany::new(elem_ty.clone());
|
2019-10-12 15:39:20 +00:00
|
|
|
|
2021-07-08 08:22:29 +00:00
|
|
|
let expected = Expectation::has_type(elem_ty.clone());
|
2021-05-11 12:06:33 +00:00
|
|
|
let len = match array {
|
2019-10-12 15:39:20 +00:00
|
|
|
Array::ElementList(items) => {
|
2021-07-08 12:16:23 +00:00
|
|
|
for &expr in items.iter() {
|
|
|
|
let cur_elem_ty = self.infer_expr_inner(expr, &expected);
|
|
|
|
coerce.coerce(self, Some(expr), &cur_elem_ty);
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
2021-05-12 12:59:35 +00:00
|
|
|
Some(items.len() as u64)
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
2021-07-08 12:16:23 +00:00
|
|
|
&Array::Repeat { initializer, repeat } => {
|
|
|
|
self.infer_expr_coerce(initializer, &Expectation::has_type(elem_ty));
|
2019-10-12 15:39:20 +00:00
|
|
|
self.infer_expr(
|
2021-07-08 12:16:23 +00:00
|
|
|
repeat,
|
2021-03-13 13:44:51 +00:00
|
|
|
&Expectation::has_type(
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Scalar(Scalar::Uint(UintTy::Usize)).intern(Interner),
|
2021-03-13 13:44:51 +00:00
|
|
|
),
|
2019-10-12 15:39:20 +00:00
|
|
|
);
|
2021-05-12 12:59:35 +00:00
|
|
|
|
2021-12-23 17:49:18 +00:00
|
|
|
consteval::eval_usize(
|
|
|
|
repeat,
|
|
|
|
consteval::ConstEvalCtx {
|
|
|
|
exprs: &body.exprs,
|
|
|
|
pats: &body.pats,
|
|
|
|
local_data: Default::default(),
|
|
|
|
infer: &mut |x| self.infer_expr(x, &expected),
|
|
|
|
},
|
|
|
|
)
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
2021-05-11 12:06:33 +00:00
|
|
|
};
|
2019-10-12 15:39:20 +00:00
|
|
|
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Array(coerce.complete(), consteval::usize_const(len)).intern(Interner)
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
Expr::Literal(lit) => match lit {
|
2021-12-19 16:58:39 +00:00
|
|
|
Literal::Bool(..) => TyKind::Scalar(Scalar::Bool).intern(Interner),
|
2021-04-06 08:50:55 +00:00
|
|
|
Literal::String(..) => {
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Ref(Mutability::Not, static_lifetime(), TyKind::Str.intern(Interner))
|
|
|
|
.intern(Interner)
|
2021-04-06 08:50:55 +00:00
|
|
|
}
|
2021-05-12 12:44:01 +00:00
|
|
|
Literal::ByteString(bs) => {
|
2021-12-19 16:58:39 +00:00
|
|
|
let byte_type = TyKind::Scalar(Scalar::Uint(UintTy::U8)).intern(Interner);
|
2021-05-11 12:06:33 +00:00
|
|
|
|
2021-05-14 07:59:30 +00:00
|
|
|
let len = consteval::usize_const(Some(bs.len() as u64));
|
2021-05-12 12:44:01 +00:00
|
|
|
|
2021-12-19 16:58:39 +00:00
|
|
|
let array_type = TyKind::Array(byte_type, len).intern(Interner);
|
|
|
|
TyKind::Ref(Mutability::Not, static_lifetime(), array_type).intern(Interner)
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
2021-12-19 16:58:39 +00:00
|
|
|
Literal::Char(..) => TyKind::Scalar(Scalar::Char).intern(Interner),
|
2020-06-06 15:52:00 +00:00
|
|
|
Literal::Int(_v, ty) => match ty {
|
2021-02-28 18:13:37 +00:00
|
|
|
Some(int_ty) => {
|
2021-03-13 13:44:51 +00:00
|
|
|
TyKind::Scalar(Scalar::Int(primitive::int_ty_from_builtin(*int_ty)))
|
2021-12-19 16:58:39 +00:00
|
|
|
.intern(Interner)
|
2021-02-28 18:13:37 +00:00
|
|
|
}
|
2021-02-28 00:20:04 +00:00
|
|
|
None => self.table.new_integer_var(),
|
|
|
|
},
|
|
|
|
Literal::Uint(_v, ty) => match ty {
|
2021-02-28 18:13:37 +00:00
|
|
|
Some(int_ty) => {
|
2021-03-13 13:44:51 +00:00
|
|
|
TyKind::Scalar(Scalar::Uint(primitive::uint_ty_from_builtin(*int_ty)))
|
2021-12-19 16:58:39 +00:00
|
|
|
.intern(Interner)
|
2021-02-28 18:13:37 +00:00
|
|
|
}
|
2020-06-06 15:52:00 +00:00
|
|
|
None => self.table.new_integer_var(),
|
|
|
|
},
|
|
|
|
Literal::Float(_v, ty) => match ty {
|
2021-02-28 18:13:37 +00:00
|
|
|
Some(float_ty) => {
|
2021-03-13 13:44:51 +00:00
|
|
|
TyKind::Scalar(Scalar::Float(primitive::float_ty_from_builtin(*float_ty)))
|
2021-12-19 16:58:39 +00:00
|
|
|
.intern(Interner)
|
2021-02-28 18:13:37 +00:00
|
|
|
}
|
2020-06-06 15:52:00 +00:00
|
|
|
None => self.table.new_float_var(),
|
|
|
|
},
|
2019-10-12 15:39:20 +00:00
|
|
|
},
|
2021-05-25 09:15:02 +00:00
|
|
|
Expr::MacroStmts { tail } => self.infer_expr_inner(*tail, expected),
|
2019-10-12 15:39:20 +00:00
|
|
|
};
|
2021-03-13 13:44:51 +00:00
|
|
|
// use a new type variable if we got unknown here
|
2019-10-12 15:39:20 +00:00
|
|
|
let ty = self.insert_type_vars_shallow(ty);
|
|
|
|
self.write_expr_ty(tgt_expr, ty.clone());
|
|
|
|
ty
|
|
|
|
}
|
|
|
|
|
2021-08-22 15:21:47 +00:00
|
|
|
fn infer_overloadable_binop(
|
|
|
|
&mut self,
|
|
|
|
lhs: ExprId,
|
|
|
|
op: BinaryOp,
|
|
|
|
rhs: ExprId,
|
|
|
|
tgt_expr: ExprId,
|
|
|
|
) -> Ty {
|
|
|
|
let lhs_expectation = Expectation::none();
|
|
|
|
let lhs_ty = self.infer_expr(lhs, &lhs_expectation);
|
|
|
|
let rhs_ty = self.table.new_type_var();
|
|
|
|
|
|
|
|
let func = self.resolve_binop_method(op);
|
|
|
|
let func = match func {
|
|
|
|
Some(func) => func,
|
|
|
|
None => {
|
|
|
|
let rhs_ty = self.builtin_binary_op_rhs_expectation(op, lhs_ty.clone());
|
|
|
|
let rhs_ty = self.infer_expr_coerce(rhs, &Expectation::from_option(rhs_ty));
|
|
|
|
return self
|
|
|
|
.builtin_binary_op_return_ty(op, lhs_ty, rhs_ty)
|
|
|
|
.unwrap_or_else(|| self.err_ty());
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
let subst = TyBuilder::subst_for_def(self.db, func)
|
|
|
|
.push(lhs_ty.clone())
|
|
|
|
.push(rhs_ty.clone())
|
|
|
|
.build();
|
|
|
|
self.write_method_resolution(tgt_expr, func, subst.clone());
|
|
|
|
|
2021-12-19 16:58:39 +00:00
|
|
|
let method_ty = self.db.value_ty(func.into()).substitute(Interner, &subst);
|
2021-08-22 15:21:47 +00:00
|
|
|
self.register_obligations_for_call(&method_ty);
|
|
|
|
|
|
|
|
self.infer_expr_coerce(rhs, &Expectation::has_type(rhs_ty.clone()));
|
|
|
|
|
|
|
|
let ret_ty = match method_ty.callable_sig(self.db) {
|
|
|
|
Some(sig) => sig.ret().clone(),
|
|
|
|
None => self.err_ty(),
|
|
|
|
};
|
|
|
|
|
|
|
|
let ret_ty = self.normalize_associated_types_in(ret_ty);
|
|
|
|
|
|
|
|
// FIXME: record autoref adjustments
|
|
|
|
|
|
|
|
// use knowledge of built-in binary ops, which can sometimes help inference
|
|
|
|
if let Some(builtin_rhs) = self.builtin_binary_op_rhs_expectation(op, lhs_ty.clone()) {
|
|
|
|
self.unify(&builtin_rhs, &rhs_ty);
|
|
|
|
}
|
2021-09-03 14:00:50 +00:00
|
|
|
if let Some(builtin_ret) = self.builtin_binary_op_return_ty(op, lhs_ty, rhs_ty) {
|
2021-08-22 15:21:47 +00:00
|
|
|
self.unify(&builtin_ret, &ret_ty);
|
|
|
|
}
|
|
|
|
|
|
|
|
ret_ty
|
|
|
|
}
|
|
|
|
|
2019-10-12 15:39:20 +00:00
|
|
|
fn infer_block(
|
|
|
|
&mut self,
|
2021-07-06 16:05:40 +00:00
|
|
|
expr: ExprId,
|
2019-10-12 15:39:20 +00:00
|
|
|
statements: &[Statement],
|
|
|
|
tail: Option<ExprId>,
|
|
|
|
expected: &Expectation,
|
|
|
|
) -> Ty {
|
|
|
|
for stmt in statements {
|
|
|
|
match stmt {
|
2021-10-07 15:05:50 +00:00
|
|
|
Statement::Let { pat, type_ref, initializer, else_branch } => {
|
2021-06-18 11:40:51 +00:00
|
|
|
let decl_ty = type_ref
|
|
|
|
.as_ref()
|
|
|
|
.map(|tr| self.make_ty(tr))
|
|
|
|
.unwrap_or_else(|| self.err_ty());
|
2019-10-12 15:39:20 +00:00
|
|
|
|
|
|
|
// Always use the declared type when specified
|
|
|
|
let mut ty = decl_ty.clone();
|
|
|
|
|
|
|
|
if let Some(expr) = initializer {
|
|
|
|
let actual_ty =
|
|
|
|
self.infer_expr_coerce(*expr, &Expectation::has_type(decl_ty.clone()));
|
2021-03-13 13:44:51 +00:00
|
|
|
if decl_ty.is_unknown() {
|
2019-10-12 15:39:20 +00:00
|
|
|
ty = actual_ty;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-07 15:05:50 +00:00
|
|
|
if let Some(expr) = else_branch {
|
|
|
|
self.infer_expr_coerce(
|
|
|
|
*expr,
|
2021-12-19 16:58:39 +00:00
|
|
|
&Expectation::has_type(Ty::new(Interner, TyKind::Never)),
|
2021-10-07 15:05:50 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-10-12 15:39:20 +00:00
|
|
|
self.infer_pat(*pat, &ty, BindingMode::default());
|
|
|
|
}
|
2021-04-07 11:45:17 +00:00
|
|
|
Statement::Expr { expr, .. } => {
|
2020-05-08 15:36:11 +00:00
|
|
|
self.infer_expr(*expr, &Expectation::none());
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-03 14:00:50 +00:00
|
|
|
if let Some(expr) = tail {
|
2019-10-12 15:39:20 +00:00
|
|
|
self.infer_expr_coerce(expr, expected)
|
|
|
|
} else {
|
2020-05-08 15:36:11 +00:00
|
|
|
// Citing rustc: if there is no explicit tail expression,
|
|
|
|
// that is typically equivalent to a tail expression
|
|
|
|
// of `()` -- except if the block diverges. In that
|
|
|
|
// case, there is no value supplied from the tail
|
|
|
|
// expression (assuming there are no other breaks,
|
|
|
|
// this implies that the type of the block will be
|
|
|
|
// `!`).
|
|
|
|
if self.diverges.is_always() {
|
|
|
|
// we don't even make an attempt at coercion
|
2021-03-01 11:35:11 +00:00
|
|
|
self.table.new_maybe_never_var()
|
2020-05-08 15:36:11 +00:00
|
|
|
} else {
|
2021-05-21 15:41:20 +00:00
|
|
|
if let Some(t) = expected.only_has_type(&mut self.table) {
|
2021-07-08 12:16:23 +00:00
|
|
|
let _ = self.coerce(Some(expr), &TyBuilder::unit(), &t);
|
2021-05-21 15:41:20 +00:00
|
|
|
}
|
2021-04-03 18:22:59 +00:00
|
|
|
TyBuilder::unit()
|
2020-05-08 15:36:11 +00:00
|
|
|
}
|
2021-09-03 14:00:50 +00:00
|
|
|
}
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn infer_method_call(
|
|
|
|
&mut self,
|
|
|
|
tgt_expr: ExprId,
|
|
|
|
receiver: ExprId,
|
|
|
|
args: &[ExprId],
|
|
|
|
method_name: &Name,
|
|
|
|
generic_args: Option<&GenericArgs>,
|
2021-07-15 18:02:58 +00:00
|
|
|
expected: &Expectation,
|
2019-10-12 15:39:20 +00:00
|
|
|
) -> Ty {
|
|
|
|
let receiver_ty = self.infer_expr(receiver, &Expectation::none());
|
2021-04-11 09:20:45 +00:00
|
|
|
let canonicalized_receiver = self.canonicalize(receiver_ty.clone());
|
2020-01-14 13:20:33 +00:00
|
|
|
|
2020-03-13 15:05:46 +00:00
|
|
|
let traits_in_scope = self.resolver.traits_in_scope(self.db.upcast());
|
2020-01-14 13:20:33 +00:00
|
|
|
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
let resolved = method_resolution::lookup_method(
|
|
|
|
&canonicalized_receiver.value,
|
|
|
|
self.db,
|
|
|
|
self.trait_env.clone(),
|
|
|
|
&traits_in_scope,
|
|
|
|
self.resolver.module().into(),
|
|
|
|
method_name,
|
|
|
|
);
|
2021-05-31 18:04:18 +00:00
|
|
|
let (receiver_ty, method_ty, substs) = match resolved {
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
Some((adjust, func)) => {
|
|
|
|
let (ty, adjustments) = adjust.apply(&mut self.table, receiver_ty);
|
2021-05-23 14:59:23 +00:00
|
|
|
let generics = generics(self.db.upcast(), func.into());
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
let substs = self.substs_for_method_call(generics, generic_args);
|
|
|
|
self.write_expr_adj(receiver, adjustments);
|
2021-05-23 14:59:23 +00:00
|
|
|
self.write_method_resolution(tgt_expr, func, substs.clone());
|
|
|
|
(ty, self.db.value_ty(func.into()), substs)
|
|
|
|
}
|
|
|
|
None => (
|
|
|
|
receiver_ty,
|
2021-12-19 16:58:39 +00:00
|
|
|
Binders::empty(Interner, self.err_ty()),
|
|
|
|
Substitution::empty(Interner),
|
2021-05-23 14:59:23 +00:00
|
|
|
),
|
2019-10-12 15:39:20 +00:00
|
|
|
};
|
2021-12-19 16:58:39 +00:00
|
|
|
let method_ty = method_ty.substitute(Interner, &substs);
|
2019-10-12 15:39:20 +00:00
|
|
|
self.register_obligations_for_call(&method_ty);
|
2021-07-15 18:02:58 +00:00
|
|
|
let (formal_receiver_ty, param_tys, ret_ty) = match method_ty.callable_sig(self.db) {
|
2019-10-12 15:39:20 +00:00
|
|
|
Some(sig) => {
|
|
|
|
if !sig.params().is_empty() {
|
|
|
|
(sig.params()[0].clone(), sig.params()[1..].to_vec(), sig.ret().clone())
|
|
|
|
} else {
|
2021-03-13 13:44:51 +00:00
|
|
|
(self.err_ty(), Vec::new(), sig.ret().clone())
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
}
|
2021-03-13 13:44:51 +00:00
|
|
|
None => (self.err_ty(), Vec::new(), self.err_ty()),
|
2019-10-12 15:39:20 +00:00
|
|
|
};
|
2021-07-15 18:02:58 +00:00
|
|
|
self.unify(&formal_receiver_ty, &receiver_ty);
|
|
|
|
|
|
|
|
let expected_inputs =
|
|
|
|
self.expected_inputs_for_expected_output(expected, ret_ty.clone(), param_tys.clone());
|
2019-10-12 15:39:20 +00:00
|
|
|
|
2021-07-15 18:02:58 +00:00
|
|
|
self.check_call_arguments(args, &expected_inputs, ¶m_tys);
|
2020-02-18 13:32:19 +00:00
|
|
|
self.normalize_associated_types_in(ret_ty)
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
|
2021-07-15 18:02:58 +00:00
|
|
|
fn expected_inputs_for_expected_output(
|
|
|
|
&mut self,
|
|
|
|
expected_output: &Expectation,
|
|
|
|
output: Ty,
|
|
|
|
inputs: Vec<Ty>,
|
|
|
|
) -> Vec<Ty> {
|
|
|
|
if let Some(expected_ty) = expected_output.to_option(&mut self.table) {
|
2021-09-03 14:00:50 +00:00
|
|
|
self.table.fudge_inference(|table| {
|
2021-08-21 17:47:06 +00:00
|
|
|
if table.try_unify(&expected_ty, &output).is_ok() {
|
2021-10-20 17:20:17 +00:00
|
|
|
table.resolve_with_fallback(inputs, &|var, kind, _, _| match kind {
|
2021-12-19 16:58:39 +00:00
|
|
|
chalk_ir::VariableKind::Ty(tk) => var.to_ty(Interner, tk).cast(Interner),
|
2021-08-21 17:47:06 +00:00
|
|
|
chalk_ir::VariableKind::Lifetime => {
|
2021-12-19 16:58:39 +00:00
|
|
|
var.to_lifetime(Interner).cast(Interner)
|
2021-08-21 17:47:06 +00:00
|
|
|
}
|
|
|
|
chalk_ir::VariableKind::Const(ty) => {
|
2021-12-19 16:58:39 +00:00
|
|
|
var.to_const(Interner, ty).cast(Interner)
|
2021-08-21 17:47:06 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
Vec::new()
|
|
|
|
}
|
2021-09-03 14:00:50 +00:00
|
|
|
})
|
2021-07-15 18:02:58 +00:00
|
|
|
} else {
|
|
|
|
Vec::new()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn check_call_arguments(&mut self, args: &[ExprId], expected_inputs: &[Ty], param_tys: &[Ty]) {
|
2019-10-12 15:39:20 +00:00
|
|
|
// Quoting https://github.com/rust-lang/rust/blob/6ef275e6c3cb1384ec78128eceeb4963ff788dca/src/librustc_typeck/check/mod.rs#L3325 --
|
|
|
|
// We do this in a pretty awful way: first we type-check any arguments
|
|
|
|
// that are not closures, then we type-check the closures. This is so
|
|
|
|
// that we have more information about the types of arguments when we
|
|
|
|
// type-check the functions. This isn't really the right way to do this.
|
|
|
|
for &check_closures in &[false, true] {
|
2021-03-13 13:44:51 +00:00
|
|
|
let param_iter = param_tys.iter().cloned().chain(repeat(self.err_ty()));
|
2021-07-15 18:02:58 +00:00
|
|
|
let expected_iter = expected_inputs
|
|
|
|
.iter()
|
|
|
|
.cloned()
|
|
|
|
.chain(param_iter.clone().skip(expected_inputs.len()));
|
|
|
|
for ((&arg, param_ty), expected_ty) in args.iter().zip(param_iter).zip(expected_iter) {
|
2020-06-28 01:02:03 +00:00
|
|
|
let is_closure = matches!(&self.body[arg], Expr::Lambda { .. });
|
2019-10-12 15:39:20 +00:00
|
|
|
if is_closure != check_closures {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2021-07-15 18:02:58 +00:00
|
|
|
// the difference between param_ty and expected here is that
|
|
|
|
// expected is the parameter when the expected *return* type is
|
|
|
|
// taken into account. So in `let _: &[i32] = identity(&[1, 2])`
|
|
|
|
// the expected type is already `&[i32]`, whereas param_ty is
|
|
|
|
// still an unbound type variable. We don't always want to force
|
|
|
|
// the parameter to coerce to the expected type (for example in
|
|
|
|
// `coerce_unsize_expected_type_4`).
|
2019-10-12 15:39:20 +00:00
|
|
|
let param_ty = self.normalize_associated_types_in(param_ty);
|
2021-07-15 18:02:58 +00:00
|
|
|
let expected = Expectation::rvalue_hint(&mut self.table, expected_ty);
|
|
|
|
// infer with the expected type we have...
|
|
|
|
let ty = self.infer_expr_inner(arg, &expected);
|
|
|
|
|
|
|
|
// then coerce to either the expected type or just the formal parameter type
|
|
|
|
let coercion_target = if let Some(ty) = expected.only_has_type(&mut self.table) {
|
|
|
|
// if we are coercing to the expectation, unify with the
|
|
|
|
// formal parameter type to connect everything
|
|
|
|
self.unify(&ty, ¶m_ty);
|
|
|
|
ty
|
|
|
|
} else {
|
|
|
|
param_ty
|
|
|
|
};
|
|
|
|
if !coercion_target.is_unknown() {
|
|
|
|
if self.coerce(Some(arg), &ty, &coercion_target).is_err() {
|
|
|
|
self.result.type_mismatches.insert(
|
|
|
|
arg.into(),
|
|
|
|
TypeMismatch { expected: coercion_target, actual: ty.clone() },
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn substs_for_method_call(
|
|
|
|
&mut self,
|
2021-05-23 14:59:23 +00:00
|
|
|
def_generics: Generics,
|
2019-10-12 15:39:20 +00:00
|
|
|
generic_args: Option<&GenericArgs>,
|
2021-03-15 20:02:34 +00:00
|
|
|
) -> Substitution {
|
2020-02-07 15:24:09 +00:00
|
|
|
let (parent_params, self_params, type_params, impl_trait_params) =
|
2021-05-23 14:59:23 +00:00
|
|
|
def_generics.provenance_split();
|
2020-02-07 15:24:09 +00:00
|
|
|
assert_eq!(self_params, 0); // method shouldn't have another Self param
|
|
|
|
let total_len = parent_params + type_params + impl_trait_params;
|
2019-12-07 12:05:05 +00:00
|
|
|
let mut substs = Vec::with_capacity(total_len);
|
Refactor autoderef and method resolution
- don't return the receiver type from method resolution; instead just
return the autorefs/autoderefs that happened and repeat them. This
ensures all the effects like trait obligations and whatever we learned
about type variables from derefing them are actually applied. Also, it
allows us to get rid of `decanonicalize_ty`, which was just wrong in
principle.
- Autoderef itself now directly works with an inference table. Sadly
this has the effect of making it harder to use as an iterator, often
requiring manual `while let` loops. (rustc works around this by using
inner mutability in the inference context, so that things like unifying
types don't require a unique reference.)
- We now record the adjustments (autoref/deref) for method receivers
and index expressions, which we didn't before.
- Removed the redundant crate parameter from method resolution, since
the trait_env contains the crate as well.
- in the HIR API, the methods now take a scope to determine the trait env.
`Type` carries a trait env, but I think that's probably a bad decision
because it's easy to create it with the wrong env, e.g. by using
`Adt::ty`. This mostly didn't matter so far because
`iterate_method_candidates` took a crate parameter and ignored
`self.krate`, but the trait env would still have been wrong in those
cases, which I think would give some wrong results in some edge cases.
Fixes #10058.
2022-02-16 16:44:03 +00:00
|
|
|
// Parent arguments are unknown
|
2021-12-29 13:35:59 +00:00
|
|
|
for (_id, param) in def_generics.iter_parent() {
|
|
|
|
match param {
|
|
|
|
TypeOrConstParamData::TypeParamData(_) => {
|
|
|
|
substs.push(self.table.new_type_var());
|
|
|
|
}
|
|
|
|
TypeOrConstParamData::ConstParamData(_) => {
|
|
|
|
// FIXME: here we should do something
|
|
|
|
}
|
|
|
|
}
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
// handle provided type arguments
|
|
|
|
if let Some(generic_args) = generic_args {
|
|
|
|
// if args are provided, it should be all of them, but we can't rely on that
|
2020-12-13 10:34:44 +00:00
|
|
|
for arg in generic_args
|
|
|
|
.args
|
|
|
|
.iter()
|
|
|
|
.filter(|arg| matches!(arg, GenericArg::Type(_)))
|
|
|
|
.take(type_params)
|
|
|
|
{
|
2019-10-12 15:39:20 +00:00
|
|
|
match arg {
|
|
|
|
GenericArg::Type(type_ref) => {
|
|
|
|
let ty = self.make_ty(type_ref);
|
|
|
|
substs.push(ty);
|
|
|
|
}
|
2020-12-11 12:49:32 +00:00
|
|
|
GenericArg::Lifetime(_) => {}
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
let supplied_params = substs.len();
|
2019-12-07 12:05:05 +00:00
|
|
|
for _ in supplied_params..total_len {
|
2021-05-23 14:59:23 +00:00
|
|
|
substs.push(self.table.new_type_var());
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
2019-12-07 12:05:05 +00:00
|
|
|
assert_eq!(substs.len(), total_len);
|
2021-12-19 16:58:39 +00:00
|
|
|
Substitution::from_iter(Interner, substs)
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn register_obligations_for_call(&mut self, callable_ty: &Ty) {
|
2021-06-13 03:54:16 +00:00
|
|
|
let callable_ty = self.resolve_ty_shallow(callable_ty);
|
2021-12-19 16:58:39 +00:00
|
|
|
if let TyKind::FnDef(fn_def, parameters) = callable_ty.kind(Interner) {
|
2021-03-13 16:55:50 +00:00
|
|
|
let def: CallableDefId = from_chalk(self.db, *fn_def);
|
|
|
|
let generic_predicates = self.db.generic_predicates(def.into());
|
2021-02-28 18:13:37 +00:00
|
|
|
for predicate in generic_predicates.iter() {
|
2021-04-05 16:49:26 +00:00
|
|
|
let (predicate, binders) = predicate
|
|
|
|
.clone()
|
2021-12-19 16:58:39 +00:00
|
|
|
.substitute(Interner, parameters)
|
2021-04-05 16:49:26 +00:00
|
|
|
.into_value_and_skipped_binders();
|
2021-12-19 16:58:39 +00:00
|
|
|
always!(binders.len(Interner) == 0); // quantified where clauses not yet handled
|
|
|
|
self.push_obligation(predicate.cast(Interner));
|
2021-02-28 18:13:37 +00:00
|
|
|
}
|
|
|
|
// add obligation for trait implementation, if this is a trait method
|
|
|
|
match def {
|
|
|
|
CallableDefId::FunctionId(f) => {
|
2021-12-07 16:31:26 +00:00
|
|
|
if let ItemContainerId::TraitId(trait_) = f.lookup(self.db.upcast()).container {
|
2021-03-20 10:23:59 +00:00
|
|
|
// construct a TraitRef
|
2021-04-05 19:56:40 +00:00
|
|
|
let substs = crate::subst_prefix(
|
|
|
|
&*parameters,
|
|
|
|
generics(self.db.upcast(), trait_.into()).len(),
|
|
|
|
);
|
2021-04-01 19:45:44 +00:00
|
|
|
self.push_obligation(
|
2021-03-20 10:23:59 +00:00
|
|
|
TraitRef { trait_id: to_chalk_trait_id(trait_), substitution: substs }
|
2021-12-19 16:58:39 +00:00
|
|
|
.cast(Interner),
|
2021-03-20 10:23:59 +00:00
|
|
|
);
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
}
|
2021-02-28 18:13:37 +00:00
|
|
|
CallableDefId::StructId(_) | CallableDefId::EnumVariantId(_) => {}
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-08-22 15:21:47 +00:00
|
|
|
|
|
|
|
fn builtin_binary_op_return_ty(&mut self, op: BinaryOp, lhs_ty: Ty, rhs_ty: Ty) -> Option<Ty> {
|
|
|
|
let lhs_ty = self.resolve_ty_shallow(&lhs_ty);
|
|
|
|
let rhs_ty = self.resolve_ty_shallow(&rhs_ty);
|
|
|
|
match op {
|
|
|
|
BinaryOp::LogicOp(_) | BinaryOp::CmpOp(_) => {
|
2021-12-19 16:58:39 +00:00
|
|
|
Some(TyKind::Scalar(Scalar::Bool).intern(Interner))
|
2021-08-22 15:21:47 +00:00
|
|
|
}
|
|
|
|
BinaryOp::Assignment { .. } => Some(TyBuilder::unit()),
|
|
|
|
BinaryOp::ArithOp(ArithOp::Shl | ArithOp::Shr) => {
|
|
|
|
// all integer combinations are valid here
|
|
|
|
if matches!(
|
2021-12-19 16:58:39 +00:00
|
|
|
lhs_ty.kind(Interner),
|
2021-08-22 15:21:47 +00:00
|
|
|
TyKind::Scalar(Scalar::Int(_) | Scalar::Uint(_))
|
|
|
|
| TyKind::InferenceVar(_, TyVariableKind::Integer)
|
|
|
|
) && matches!(
|
2021-12-19 16:58:39 +00:00
|
|
|
rhs_ty.kind(Interner),
|
2021-08-22 15:21:47 +00:00
|
|
|
TyKind::Scalar(Scalar::Int(_) | Scalar::Uint(_))
|
|
|
|
| TyKind::InferenceVar(_, TyVariableKind::Integer)
|
|
|
|
) {
|
|
|
|
Some(lhs_ty)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
2021-12-19 16:58:39 +00:00
|
|
|
BinaryOp::ArithOp(_) => match (lhs_ty.kind(Interner), rhs_ty.kind(Interner)) {
|
2021-08-22 15:21:47 +00:00
|
|
|
// (int, int) | (uint, uint) | (float, float)
|
|
|
|
(TyKind::Scalar(Scalar::Int(_)), TyKind::Scalar(Scalar::Int(_)))
|
|
|
|
| (TyKind::Scalar(Scalar::Uint(_)), TyKind::Scalar(Scalar::Uint(_)))
|
|
|
|
| (TyKind::Scalar(Scalar::Float(_)), TyKind::Scalar(Scalar::Float(_))) => {
|
|
|
|
Some(rhs_ty)
|
|
|
|
}
|
|
|
|
// ({int}, int) | ({int}, uint)
|
|
|
|
(
|
|
|
|
TyKind::InferenceVar(_, TyVariableKind::Integer),
|
|
|
|
TyKind::Scalar(Scalar::Int(_) | Scalar::Uint(_)),
|
|
|
|
) => Some(rhs_ty),
|
|
|
|
// (int, {int}) | (uint, {int})
|
|
|
|
(
|
|
|
|
TyKind::Scalar(Scalar::Int(_) | Scalar::Uint(_)),
|
|
|
|
TyKind::InferenceVar(_, TyVariableKind::Integer),
|
|
|
|
) => Some(lhs_ty),
|
|
|
|
// ({float} | float)
|
|
|
|
(
|
|
|
|
TyKind::InferenceVar(_, TyVariableKind::Float),
|
|
|
|
TyKind::Scalar(Scalar::Float(_)),
|
|
|
|
) => Some(rhs_ty),
|
|
|
|
// (float, {float})
|
|
|
|
(
|
|
|
|
TyKind::Scalar(Scalar::Float(_)),
|
|
|
|
TyKind::InferenceVar(_, TyVariableKind::Float),
|
|
|
|
) => Some(lhs_ty),
|
|
|
|
// ({int}, {int}) | ({float}, {float})
|
|
|
|
(
|
|
|
|
TyKind::InferenceVar(_, TyVariableKind::Integer),
|
|
|
|
TyKind::InferenceVar(_, TyVariableKind::Integer),
|
|
|
|
)
|
|
|
|
| (
|
|
|
|
TyKind::InferenceVar(_, TyVariableKind::Float),
|
|
|
|
TyKind::InferenceVar(_, TyVariableKind::Float),
|
|
|
|
) => Some(rhs_ty),
|
|
|
|
_ => None,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn builtin_binary_op_rhs_expectation(&mut self, op: BinaryOp, lhs_ty: Ty) -> Option<Ty> {
|
|
|
|
Some(match op {
|
2021-12-19 16:58:39 +00:00
|
|
|
BinaryOp::LogicOp(..) => TyKind::Scalar(Scalar::Bool).intern(Interner),
|
2021-08-22 15:21:47 +00:00
|
|
|
BinaryOp::Assignment { op: None } => lhs_ty,
|
|
|
|
BinaryOp::CmpOp(CmpOp::Eq { .. }) => match self
|
|
|
|
.resolve_ty_shallow(&lhs_ty)
|
2021-12-19 16:58:39 +00:00
|
|
|
.kind(Interner)
|
2021-08-22 15:21:47 +00:00
|
|
|
{
|
|
|
|
TyKind::Scalar(_) | TyKind::Str => lhs_ty,
|
|
|
|
TyKind::InferenceVar(_, TyVariableKind::Integer | TyVariableKind::Float) => lhs_ty,
|
|
|
|
_ => return None,
|
|
|
|
},
|
|
|
|
BinaryOp::ArithOp(ArithOp::Shl | ArithOp::Shr) => return None,
|
|
|
|
BinaryOp::CmpOp(CmpOp::Ord { .. })
|
|
|
|
| BinaryOp::Assignment { op: Some(_) }
|
2021-12-19 16:58:39 +00:00
|
|
|
| BinaryOp::ArithOp(_) => match self.resolve_ty_shallow(&lhs_ty).kind(Interner) {
|
2021-08-22 15:21:47 +00:00
|
|
|
TyKind::Scalar(Scalar::Int(_) | Scalar::Uint(_) | Scalar::Float(_)) => lhs_ty,
|
|
|
|
TyKind::InferenceVar(_, TyVariableKind::Integer | TyVariableKind::Float) => lhs_ty,
|
|
|
|
_ => return None,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
fn resolve_binop_method(&self, op: BinaryOp) -> Option<FunctionId> {
|
|
|
|
let (name, lang_item) = match op {
|
|
|
|
BinaryOp::LogicOp(_) => return None,
|
|
|
|
BinaryOp::ArithOp(aop) => match aop {
|
2021-12-09 17:13:15 +00:00
|
|
|
ArithOp::Add => (name!(add), name!(add)),
|
|
|
|
ArithOp::Mul => (name!(mul), name!(mul)),
|
|
|
|
ArithOp::Sub => (name!(sub), name!(sub)),
|
|
|
|
ArithOp::Div => (name!(div), name!(div)),
|
|
|
|
ArithOp::Rem => (name!(rem), name!(rem)),
|
|
|
|
ArithOp::Shl => (name!(shl), name!(shl)),
|
|
|
|
ArithOp::Shr => (name!(shr), name!(shr)),
|
|
|
|
ArithOp::BitXor => (name!(bitxor), name!(bitxor)),
|
|
|
|
ArithOp::BitOr => (name!(bitor), name!(bitor)),
|
|
|
|
ArithOp::BitAnd => (name!(bitand), name!(bitand)),
|
2021-08-22 15:21:47 +00:00
|
|
|
},
|
|
|
|
BinaryOp::Assignment { op: Some(aop) } => match aop {
|
2021-12-09 17:13:15 +00:00
|
|
|
ArithOp::Add => (name!(add_assign), name!(add_assign)),
|
|
|
|
ArithOp::Mul => (name!(mul_assign), name!(mul_assign)),
|
|
|
|
ArithOp::Sub => (name!(sub_assign), name!(sub_assign)),
|
|
|
|
ArithOp::Div => (name!(div_assign), name!(div_assign)),
|
|
|
|
ArithOp::Rem => (name!(rem_assign), name!(rem_assign)),
|
|
|
|
ArithOp::Shl => (name!(shl_assign), name!(shl_assign)),
|
|
|
|
ArithOp::Shr => (name!(shr_assign), name!(shr_assign)),
|
|
|
|
ArithOp::BitXor => (name!(bitxor_assign), name!(bitxor_assign)),
|
|
|
|
ArithOp::BitOr => (name!(bitor_assign), name!(bitor_assign)),
|
|
|
|
ArithOp::BitAnd => (name!(bitand_assign), name!(bitand_assign)),
|
2021-08-22 15:21:47 +00:00
|
|
|
},
|
|
|
|
BinaryOp::CmpOp(cop) => match cop {
|
2021-12-09 17:13:15 +00:00
|
|
|
CmpOp::Eq { negated: false } => (name!(eq), name!(eq)),
|
|
|
|
CmpOp::Eq { negated: true } => (name!(ne), name!(eq)),
|
2021-08-22 15:21:47 +00:00
|
|
|
CmpOp::Ord { ordering: Ordering::Less, strict: false } => {
|
2021-12-09 17:13:15 +00:00
|
|
|
(name!(le), name!(partial_ord))
|
|
|
|
}
|
|
|
|
CmpOp::Ord { ordering: Ordering::Less, strict: true } => {
|
|
|
|
(name!(lt), name!(partial_ord))
|
2021-08-22 15:21:47 +00:00
|
|
|
}
|
|
|
|
CmpOp::Ord { ordering: Ordering::Greater, strict: false } => {
|
2021-12-09 17:13:15 +00:00
|
|
|
(name!(ge), name!(partial_ord))
|
2021-08-22 15:21:47 +00:00
|
|
|
}
|
|
|
|
CmpOp::Ord { ordering: Ordering::Greater, strict: true } => {
|
2021-12-09 17:13:15 +00:00
|
|
|
(name!(gt), name!(partial_ord))
|
2021-08-22 15:21:47 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
BinaryOp::Assignment { op: None } => return None,
|
|
|
|
};
|
|
|
|
|
|
|
|
let trait_ = self.resolve_lang_item(lang_item)?.as_trait()?;
|
|
|
|
|
|
|
|
self.db.trait_data(trait_).method_by_name(&name)
|
|
|
|
}
|
2019-10-12 15:39:20 +00:00
|
|
|
}
|