From cfa50df33e1ebfa89f7fbdece7454699f858de92 Mon Sep 17 00:00:00 2001 From: Florian Diebold Date: Mon, 2 Dec 2019 18:12:49 +0100 Subject: [PATCH] Refactor a bit --- crates/ra_hir_ty/src/infer/path.rs | 38 +-------- crates/ra_hir_ty/src/method_resolution.rs | 98 +++++++++++++---------- crates/ra_hir_ty/src/tests.rs | 15 ++++ 3 files changed, 74 insertions(+), 77 deletions(-) diff --git a/crates/ra_hir_ty/src/infer/path.rs b/crates/ra_hir_ty/src/infer/path.rs index d0d7646a49d..b0024c6e1c9 100644 --- a/crates/ra_hir_ty/src/infer/path.rs +++ b/crates/ra_hir_ty/src/infer/path.rs @@ -206,7 +206,9 @@ impl<'a, D: HirDatabase> InferenceContext<'a, D> { AssocItemId::TypeAliasId(_) => unreachable!(), }; let substs = match container { - ContainerId::ImplId(_) => self.find_self_types(&def, ty.clone()), + ContainerId::ImplId(impl_id) => { + method_resolution::inherent_impl_substs(self.db, impl_id, &ty) + } ContainerId::TraitId(trait_) => { // we're picking this method let trait_substs = Substs::build_for_def(self.db, trait_) @@ -231,38 +233,4 @@ impl<'a, D: HirDatabase> InferenceContext<'a, D> { }, ) } - - fn find_self_types(&self, def: &ValueNs, actual_def_ty: Ty) -> Option { - if let ValueNs::FunctionId(func) = *def { - // We only do the infer if parent has generic params - let gen = self.db.generic_params(func.into()); - if gen.count_parent_params() == 0 { - return None; - } - - let impl_id = match func.lookup(self.db).container { - ContainerId::ImplId(it) => it, - _ => return None, - }; - let self_ty = self.db.impl_self_ty(impl_id).clone(); - let self_ty_substs = self_ty.substs()?; - let actual_substs = actual_def_ty.substs()?; - - let mut new_substs = vec![Ty::Unknown; gen.count_parent_params()]; - - // The following code *link up* the function actual parma type - // and impl_block type param index - self_ty_substs.iter().zip(actual_substs.iter()).for_each(|(param, pty)| { - if let Ty::Param { idx, .. } = param { - if let Some(s) = new_substs.get_mut(*idx as usize) { - *s = pty.clone(); - } - } - }); - - Some(Substs(new_substs.into())) - } else { - None - } - } } diff --git a/crates/ra_hir_ty/src/method_resolution.rs b/crates/ra_hir_ty/src/method_resolution.rs index 97281cf15c6..21efb196af1 100644 --- a/crates/ra_hir_ty/src/method_resolution.rs +++ b/crates/ra_hir_ty/src/method_resolution.rs @@ -191,13 +191,13 @@ pub fn iterate_method_candidates( let ty = InEnvironment { value: ty.clone(), environment }; let krate = resolver.krate()?; - // We have to be careful about the order of operations here. - // Consider the case where we're resolving `x.clone()` where `x: - // &Vec<_>`. This resolves to the clone method with self type - // `Vec<_>`, *not* `&_`. I.e. we need to consider methods where the - // receiver type exactly matches before cases where we have to do - // autoref. But in the autoderef steps, the `&_` self type comes up - // *before* the `Vec<_>` self type. + // We have to be careful about the order we're looking at candidates + // in here. Consider the case where we're resolving `x.clone()` + // where `x: &Vec<_>`. This resolves to the clone method with self + // type `Vec<_>`, *not* `&_`. I.e. we need to consider methods where + // the receiver type exactly matches before cases where we have to + // do autoref. But in the autoderef steps, the `&_` self type comes + // up *before* the `Vec<_>` self type. // // On the other hand, we don't want to just pick any by-value method // before any by-autoref method; it's just that we need to consider @@ -206,7 +206,7 @@ pub fn iterate_method_candidates( let deref_chain: Vec<_> = autoderef::autoderef(db, Some(krate), ty.clone()).collect(); for i in 0..deref_chain.len() { - if let Some(result) = iterate_method_candidates_autoref( + if let Some(result) = iterate_method_candidates_with_autoref( &deref_chain[i..], db, resolver, @@ -220,12 +220,12 @@ pub fn iterate_method_candidates( } LookupMode::Path => { // No autoderef for path lookups - iterate_method_candidates_inner(&ty, db, resolver, name, None, &mut callback) + iterate_method_candidates_for_self_ty(&ty, db, resolver, name, &mut callback) } } } -fn iterate_method_candidates_autoref( +fn iterate_method_candidates_with_autoref( deref_chain: &[Canonical], db: &impl HirDatabase, resolver: &Resolver, @@ -275,18 +275,25 @@ fn iterate_method_candidates_autoref( fn iterate_method_candidates_by_receiver( receiver_ty: &Canonical, - deref_chain: &[Canonical], + rest_of_deref_chain: &[Canonical], db: &impl HirDatabase, resolver: &Resolver, name: Option<&Name>, mut callback: impl FnMut(&Ty, AssocItemId) -> Option, ) -> Option { - // TODO: do we need to do the whole loop for inherents before traits? // We're looking for methods with *receiver* type receiver_ty. These could // be found in any of the derefs of receiver_ty, so we have to go through // that. - for self_ty in std::iter::once(receiver_ty).chain(deref_chain) { - if let Some(result) = iterate_method_candidates_inner( + let krate = resolver.krate()?; + for self_ty in std::iter::once(receiver_ty).chain(rest_of_deref_chain) { + if let Some(result) = + iterate_inherent_methods(self_ty, db, name, Some(receiver_ty), krate, &mut callback) + { + return Some(result); + } + } + for self_ty in std::iter::once(receiver_ty).chain(rest_of_deref_chain) { + if let Some(result) = iterate_trait_method_candidates( self_ty, db, resolver, @@ -300,22 +307,19 @@ fn iterate_method_candidates_by_receiver( None } -fn iterate_method_candidates_inner( +fn iterate_method_candidates_for_self_ty( self_ty: &Canonical, db: &impl HirDatabase, resolver: &Resolver, name: Option<&Name>, - receiver_ty: Option<&Canonical>, mut callback: impl FnMut(&Ty, AssocItemId) -> Option, ) -> Option { let krate = resolver.krate()?; - if let Some(result) = - iterate_inherent_methods(self_ty, db, name, receiver_ty, krate, &mut callback) - { + if let Some(result) = iterate_inherent_methods(self_ty, db, name, None, krate, &mut callback) { return Some(result); } if let Some(result) = - iterate_trait_method_candidates(self_ty, db, resolver, name, receiver_ty, &mut callback) + iterate_trait_method_candidates(self_ty, db, resolver, name, None, &mut callback) { return Some(result); } @@ -412,29 +416,11 @@ fn is_valid_candidate( if !data.has_self_param { return false; } - let substs = match m.lookup(db).container { - hir_def::ContainerId::TraitId(_) => Substs::build_for_def(db, item) - .push(self_ty.value.clone()) - .fill_with_unknown() - .build(), - hir_def::ContainerId::ImplId(impl_id) => { - let vars = - Substs::build_for_def(db, impl_id).fill_with_bound_vars(0).build(); - let self_ty_with_vars = db.impl_self_ty(impl_id).subst(&vars); - let self_ty_with_vars = - Canonical { num_vars: vars.len(), value: &self_ty_with_vars }; - if let Some(substs) = super::infer::unify(self_ty_with_vars, &self_ty.value) - { - substs - } else { - return false; - } - } - hir_def::ContainerId::ModuleId(_) => unreachable!(), + let transformed_receiver_ty = match transform_receiver_ty(db, m, self_ty) { + Some(ty) => ty, + None => return false, }; - let sig = db.callable_item_signature(m.into()); - let receiver = sig.params()[0].clone().subst(&substs); - if receiver != receiver_ty.value { + if transformed_receiver_ty != receiver_ty.value { return false; } } @@ -448,6 +434,34 @@ fn is_valid_candidate( } } +pub(crate) fn inherent_impl_substs( + db: &impl HirDatabase, + impl_id: ImplId, + self_ty: &Ty, +) -> Option { + let vars = Substs::build_for_def(db, impl_id).fill_with_bound_vars(0).build(); + let self_ty_with_vars = db.impl_self_ty(impl_id).subst(&vars); + let self_ty_with_vars = Canonical { num_vars: vars.len(), value: &self_ty_with_vars }; + super::infer::unify(self_ty_with_vars, self_ty) +} + +fn transform_receiver_ty( + db: &impl HirDatabase, + function_id: FunctionId, + self_ty: &Canonical, +) -> Option { + let substs = match function_id.lookup(db).container { + hir_def::ContainerId::TraitId(_) => Substs::build_for_def(db, function_id) + .push(self_ty.value.clone()) + .fill_with_unknown() + .build(), + hir_def::ContainerId::ImplId(impl_id) => inherent_impl_substs(db, impl_id, &self_ty.value)?, + hir_def::ContainerId::ModuleId(_) => unreachable!(), + }; + let sig = db.callable_item_signature(function_id.into()); + Some(sig.params()[0].clone().subst(&substs)) +} + pub fn implements_trait( ty: &Canonical, db: &impl HirDatabase, diff --git a/crates/ra_hir_ty/src/tests.rs b/crates/ra_hir_ty/src/tests.rs index be8768c622b..d28e835c724 100644 --- a/crates/ra_hir_ty/src/tests.rs +++ b/crates/ra_hir_ty/src/tests.rs @@ -3493,6 +3493,21 @@ fn test() { S.foo()<|>; } assert_eq!(t, "i8"); } +#[test] +fn method_resolution_impl_ref_before_trait() { + let t = type_at( + r#" +//- /main.rs +trait Trait { fn foo(self) -> u128; } +struct S; +impl S { fn foo(&self) -> i8 { 0 } } +impl Trait for &S { fn foo(self) -> u128 { 0 } } +fn test() { S.foo()<|>; } +"#, + ); + assert_eq!(t, "i8"); +} + #[test] fn method_resolution_trait_autoderef() { let t = type_at(