rust-analyzer/crates/ra_hir_ty/src/autoderef.rs

109 lines
3.8 KiB
Rust
Raw Normal View History

//! In certain situations, rust automatically inserts derefs as necessary: for
2019-01-06 18:51:42 +00:00
//! example, field accesses `foo.bar` still work when `foo` is actually a
//! reference to a type with the field `bar`. This is an approximation of the
//! logic in rustc (which lives in librustc_typeck/check/autoderef.rs).
use std::iter::successors;
2019-01-06 18:51:42 +00:00
use hir_def::lang_item::LangItemTarget;
2019-10-30 15:56:20 +00:00
use hir_expand::name;
use log::{info, warn};
2019-11-25 09:45:45 +00:00
use ra_db::CrateId;
2019-01-06 18:51:42 +00:00
2019-11-26 14:21:29 +00:00
use crate::db::HirDatabase;
2019-11-23 09:58:01 +00:00
2019-11-25 09:45:45 +00:00
use super::{
traits::{InEnvironment, Solution},
Canonical, Substs, Ty, TypeWalk,
};
2019-06-16 10:04:08 +00:00
const AUTODEREF_RECURSION_LIMIT: usize = 10;
2019-11-27 14:46:02 +00:00
pub fn autoderef<'a>(
db: &'a impl HirDatabase,
krate: Option<CrateId>,
ty: InEnvironment<Canonical<Ty>>,
) -> impl Iterator<Item = Canonical<Ty>> + 'a {
let InEnvironment { value: ty, environment } = ty;
successors(Some(ty), move |ty| {
deref(db, krate?, InEnvironment { value: ty, environment: environment.clone() })
})
.take(AUTODEREF_RECURSION_LIMIT)
}
pub(crate) fn deref(
db: &impl HirDatabase,
krate: CrateId,
ty: InEnvironment<&Canonical<Ty>>,
) -> Option<Canonical<Ty>> {
if let Some(derefed) = ty.value.value.builtin_deref() {
Some(Canonical { value: derefed, num_vars: ty.value.num_vars })
} else {
deref_by_trait(db, krate, ty)
2019-01-06 18:51:42 +00:00
}
}
fn deref_by_trait(
db: &impl HirDatabase,
2019-11-25 09:45:45 +00:00
krate: CrateId,
ty: InEnvironment<&Canonical<Ty>>,
) -> Option<Canonical<Ty>> {
2019-11-21 12:24:51 +00:00
let deref_trait = match db.lang_item(krate.into(), "deref".into())? {
2019-11-26 14:21:29 +00:00
LangItemTarget::TraitId(it) => it,
_ => return None,
};
2019-11-26 14:21:29 +00:00
let target = db.trait_data(deref_trait).associated_type_by_name(&name::TARGET_TYPE)?;
2019-11-26 14:21:29 +00:00
let generic_params = db.generic_params(target.into());
if generic_params.count_params_including_parent() != 1 {
// the Target type + Deref trait should only have one generic parameter,
// namely Deref's Self type
return None;
}
// FIXME make the Canonical handling nicer
let parameters = Substs::build_for_generics(&generic_params)
2019-11-25 09:45:45 +00:00
.push(ty.value.value.clone().shift_bound_vars(1))
.build();
let projection = super::traits::ProjectionPredicate {
ty: Ty::Bound(0),
2019-11-26 14:21:29 +00:00
projection_ty: super::ProjectionTy { associated_ty: target, parameters },
};
let obligation = super::Obligation::Projection(projection);
2019-11-25 09:45:45 +00:00
let in_env = InEnvironment { value: obligation, environment: ty.environment };
2019-11-25 09:45:45 +00:00
let canonical = super::Canonical { num_vars: 1 + ty.value.num_vars, value: in_env };
2019-11-21 12:24:51 +00:00
let solution = db.trait_solve(krate.into(), canonical)?;
2019-01-06 18:51:42 +00:00
match &solution {
Solution::Unique(vars) => {
// FIXME: vars may contain solutions for any inference variables
// that happened to be inside ty. To correctly handle these, we
// would have to pass the solution up to the inference context, but
// that requires a larger refactoring (especially if the deref
// happens during method resolution). So for the moment, we just
// check that we're not in the situation we're we would actually
// need to handle the values of the additional variables, i.e.
// they're just being 'passed through'. In the 'standard' case where
// we have `impl<T> Deref for Foo<T> { Target = T }`, that should be
// the case.
for i in 1..vars.0.num_vars {
if vars.0.value[i] != Ty::Bound((i - 1) as u32) {
2019-11-25 09:45:45 +00:00
warn!("complex solution for derefing {:?}: {:?}, ignoring", ty.value, solution);
return None;
}
}
Some(Canonical { value: vars.0.value[0].clone(), num_vars: vars.0.num_vars })
}
Solution::Ambig(_) => {
2019-11-25 09:45:45 +00:00
info!("Ambiguous solution for derefing {:?}: {:?}", ty.value, solution);
None
}
2019-01-06 18:51:42 +00:00
}
}