1 //! Path expression resolution.
3 use chalk_ir::cast::Cast;
5 path::{Path, PathSegment},
6 resolver::{ResolveValueResult, Resolver, TypeNs, ValueNs},
7 AdtId, AssocItemId, EnumVariantId, ItemContainerId, Lookup,
9 use hir_expand::name::Name;
14 method_resolution::{self, VisibleFromModule},
15 GenericArgData, Interner, Substitution, TraitRefExt, Ty, TyBuilder, TyExt, TyKind,
19 use super::{ExprOrPatId, InferenceContext, TraitRef};
21 impl<'a> InferenceContext<'a> {
22 pub(super) fn infer_path(
28 let ty = self.resolve_value_path(resolver, path, id)?;
29 let ty = self.insert_type_vars(ty);
30 let ty = self.normalize_associated_types_in(ty);
34 fn resolve_value_path(
40 let (value, self_subst) = if let Some(type_ref) = path.type_anchor() {
41 if path.segments().is_empty() {
42 // This can't actually happen syntax-wise
45 let ty = self.make_ty(type_ref);
46 let remaining_segments_for_ty = path.segments().take(path.segments().len() - 1);
47 let ctx = crate::lower::TyLoweringContext::new(self.db, resolver);
48 let (ty, _) = ctx.lower_ty_relative_path(ty, None, remaining_segments_for_ty);
49 self.resolve_ty_assoc_item(
51 path.segments().last().expect("path had at least one segment").name,
55 let value_or_partial =
56 resolver.resolve_path_in_value_ns(self.db.upcast(), path.mod_path())?;
58 match value_or_partial {
59 ResolveValueResult::ValueNs(it) => (it, None),
60 ResolveValueResult::Partial(def, remaining_index) => {
61 self.resolve_assoc_item(def, path, remaining_index, id)?
66 let typable: ValueTyDefId = match value {
67 ValueNs::LocalBinding(pat) => {
68 let ty = self.result.type_of_pat.get(pat)?.clone();
71 ValueNs::FunctionId(it) => it.into(),
72 ValueNs::ConstId(it) => it.into(),
73 ValueNs::StaticId(it) => it.into(),
74 ValueNs::StructId(it) => {
75 self.write_variant_resolution(id, it.into());
79 ValueNs::EnumVariantId(it) => {
80 self.write_variant_resolution(id, it.into());
84 ValueNs::ImplSelf(impl_id) => {
85 let generics = crate::utils::generics(self.db.upcast(), impl_id.into());
86 let substs = generics.placeholder_subst(self.db);
87 let ty = self.db.impl_self_ty(impl_id).substitute(Interner, &substs);
88 if let Some((AdtId::StructId(struct_id), substs)) = ty.as_adt() {
89 let ty = self.db.value_ty(struct_id.into()).substitute(Interner, &substs);
92 // FIXME: diagnostic, invalid Self reference
96 ValueNs::GenericParam(it) => return Some(self.db.const_param_ty(it)),
99 let parent_substs = self_subst.unwrap_or_else(|| Substitution::empty(Interner));
100 let ctx = crate::lower::TyLoweringContext::new(self.db, &self.resolver);
101 let substs = ctx.substs_from_path(path, typable, true);
102 let mut it = substs.as_slice(Interner)[parent_substs.len(Interner)..].iter().cloned();
103 let ty = TyBuilder::value_ty(self.db, typable)
104 .use_parent_substs(&parent_substs)
106 it.next().unwrap_or_else(|| match x {
108 GenericArgData::Ty(TyKind::Error.intern(Interner)).intern(Interner)
110 ParamKind::Const(ty) => consteval::unknown_const_as_generic(ty.clone()),
117 fn resolve_assoc_item(
121 remaining_index: usize,
123 ) -> Option<(ValueNs, Option<Substitution>)> {
124 assert!(remaining_index < path.segments().len());
125 // there may be more intermediate segments between the resolved one and
126 // the end. Only the last segment needs to be resolved to a value; from
127 // the segments before that, we need to get either a type or a trait ref.
129 let resolved_segment = path.segments().get(remaining_index - 1).unwrap();
130 let remaining_segments = path.segments().skip(remaining_index);
131 let is_before_last = remaining_segments.len() == 1;
133 match (def, is_before_last) {
134 (TypeNs::TraitId(trait_), true) => {
136 remaining_segments.last().expect("there should be at least one segment here");
137 let ctx = crate::lower::TyLoweringContext::new(self.db, &self.resolver);
139 ctx.lower_trait_ref_from_resolved_path(trait_, resolved_segment, None);
140 self.resolve_trait_assoc_item(trait_ref, segment, id)
143 // Either we already have a type (e.g. `Vec::new`), or we have a
144 // trait but it's not the last segment, so the next segment
145 // should resolve to an associated type of that trait (e.g. `<T
146 // as Iterator>::Item::default`)
147 let remaining_segments_for_ty =
148 remaining_segments.take(remaining_segments.len() - 1);
149 let ctx = crate::lower::TyLoweringContext::new(self.db, &self.resolver);
150 let (ty, _) = ctx.lower_partly_resolved_path(
153 remaining_segments_for_ty,
156 if let TyKind::Error = ty.kind(Interner) {
160 let ty = self.insert_type_vars(ty);
161 let ty = self.normalize_associated_types_in(ty);
164 remaining_segments.last().expect("there should be at least one segment here");
166 self.resolve_ty_assoc_item(ty, segment.name, id)
171 fn resolve_trait_assoc_item(
174 segment: PathSegment<'_>,
176 ) -> Option<(ValueNs, Option<Substitution>)> {
177 let trait_ = trait_ref.hir_trait_id();
179 self.db.trait_data(trait_).items.iter().map(|(_name, id)| (*id)).find_map(|item| {
181 AssocItemId::FunctionId(func) => {
182 if segment.name == &self.db.function_data(func).name {
183 Some(AssocItemId::FunctionId(func))
189 AssocItemId::ConstId(konst) => {
195 .map_or(false, |n| n == segment.name)
197 Some(AssocItemId::ConstId(konst))
202 AssocItemId::TypeAliasId(_) => None,
205 let def = match item {
206 AssocItemId::FunctionId(f) => ValueNs::FunctionId(f),
207 AssocItemId::ConstId(c) => ValueNs::ConstId(c),
208 AssocItemId::TypeAliasId(_) => unreachable!(),
211 self.write_assoc_resolution(id, item);
212 Some((def, Some(trait_ref.substitution)))
215 fn resolve_ty_assoc_item(
220 ) -> Option<(ValueNs, Option<Substitution>)> {
221 if let TyKind::Error = ty.kind(Interner) {
225 if let Some(result) = self.resolve_enum_variant_on_ty(&ty, name, id) {
229 let canonical_ty = self.canonicalize(ty.clone());
230 let traits_in_scope = self.resolver.traits_in_scope(self.db.upcast());
232 method_resolution::iterate_method_candidates(
235 self.table.trait_env.clone(),
237 VisibleFromModule::Filter(self.resolver.module()),
239 method_resolution::LookupMode::Path,
241 let (def, container) = match item {
242 AssocItemId::FunctionId(f) => {
243 (ValueNs::FunctionId(f), f.lookup(self.db.upcast()).container)
245 AssocItemId::ConstId(c) => {
246 (ValueNs::ConstId(c), c.lookup(self.db.upcast()).container)
248 AssocItemId::TypeAliasId(_) => unreachable!(),
250 let substs = match container {
251 ItemContainerId::ImplId(impl_id) => {
252 let impl_substs = TyBuilder::subst_for_def(self.db, impl_id)
253 .fill_with_inference_vars(&mut self.table)
256 self.db.impl_self_ty(impl_id).substitute(Interner, &impl_substs);
257 self.unify(&impl_self_ty, &ty);
260 ItemContainerId::TraitId(trait_) => {
261 // we're picking this method
262 let trait_ref = TyBuilder::trait_ref(self.db, trait_)
264 .fill_with_inference_vars(&mut self.table)
266 self.push_obligation(trait_ref.clone().cast(Interner));
267 Some(trait_ref.substitution)
269 ItemContainerId::ModuleId(_) | ItemContainerId::ExternBlockId(_) => None,
272 self.write_assoc_resolution(id, item);
278 fn resolve_enum_variant_on_ty(
283 ) -> Option<(ValueNs, Option<Substitution>)> {
284 let ty = self.resolve_ty_shallow(ty);
285 let (enum_id, subst) = match ty.as_adt() {
286 Some((AdtId::EnumId(e), subst)) => (e, subst),
289 let enum_data = self.db.enum_data(enum_id);
290 let local_id = enum_data.variant(name)?;
291 let variant = EnumVariantId { parent: enum_id, local_id };
292 self.write_variant_resolution(id, variant.into());
293 Some((ValueNs::EnumVariantId(variant), Some(subst.clone())))