]> git.lizzy.rs Git - rust.git/blob - crates/hir_ty/src/infer/path.rs
Merge #8432
[rust.git] / crates / hir_ty / src / infer / path.rs
1 //! Path expression resolution.
2
3 use std::iter;
4
5 use chalk_ir::cast::Cast;
6 use hir_def::{
7     path::{Path, PathSegment},
8     resolver::{ResolveValueResult, Resolver, TypeNs, ValueNs},
9     AdtId, AssocContainerId, AssocItemId, EnumVariantId, Lookup,
10 };
11 use hir_expand::name::Name;
12
13 use crate::{
14     method_resolution, Interner, Substitution, TraitRefExt, Ty, TyBuilder, TyExt, TyKind,
15     ValueTyDefId,
16 };
17
18 use super::{ExprOrPatId, InferenceContext, TraitRef};
19
20 impl<'a> InferenceContext<'a> {
21     pub(super) fn infer_path(
22         &mut self,
23         resolver: &Resolver,
24         path: &Path,
25         id: ExprOrPatId,
26     ) -> Option<Ty> {
27         let ty = self.resolve_value_path(resolver, path, id)?;
28         let ty = self.insert_type_vars(ty);
29         let ty = self.normalize_associated_types_in(ty);
30         Some(ty)
31     }
32
33     fn resolve_value_path(
34         &mut self,
35         resolver: &Resolver,
36         path: &Path,
37         id: ExprOrPatId,
38     ) -> Option<Ty> {
39         let (value, self_subst) = if let Some(type_ref) = path.type_anchor() {
40             if path.segments().is_empty() {
41                 // This can't actually happen syntax-wise
42                 return None;
43             }
44             let ty = self.make_ty(type_ref);
45             let remaining_segments_for_ty = path.segments().take(path.segments().len() - 1);
46             let ctx = crate::lower::TyLoweringContext::new(self.db, &resolver);
47             let (ty, _) = ctx.lower_ty_relative_path(ty, None, remaining_segments_for_ty);
48             self.resolve_ty_assoc_item(
49                 ty,
50                 &path.segments().last().expect("path had at least one segment").name,
51                 id,
52             )?
53         } else {
54             let value_or_partial =
55                 resolver.resolve_path_in_value_ns(self.db.upcast(), path.mod_path())?;
56
57             match value_or_partial {
58                 ResolveValueResult::ValueNs(it) => (it, None),
59                 ResolveValueResult::Partial(def, remaining_index) => {
60                     self.resolve_assoc_item(def, path, remaining_index, id)?
61                 }
62             }
63         };
64
65         let typable: ValueTyDefId = match value {
66             ValueNs::LocalBinding(pat) => {
67                 let ty = self.result.type_of_pat.get(pat)?.clone();
68                 let ty = self.resolve_ty_as_possible(ty);
69                 return Some(ty);
70             }
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());
76
77                 it.into()
78             }
79             ValueNs::EnumVariantId(it) => {
80                 self.write_variant_resolution(id, it.into());
81
82                 it.into()
83             }
84             ValueNs::ImplSelf(impl_id) => {
85                 let generics = crate::utils::generics(self.db.upcast(), impl_id.into());
86                 let substs = generics.type_params_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);
90                     return Some(ty);
91                 } else {
92                     // FIXME: diagnostic, invalid Self reference
93                     return None;
94                 }
95             }
96             ValueNs::GenericParam(it) => return Some(self.db.const_param_ty(it)),
97         };
98
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 ty = TyBuilder::value_ty(self.db, typable)
103             .use_parent_substs(&parent_substs)
104             .fill(substs.as_slice(&Interner)[parent_substs.len(&Interner)..].iter().cloned())
105             .build();
106         Some(ty)
107     }
108
109     fn resolve_assoc_item(
110         &mut self,
111         def: TypeNs,
112         path: &Path,
113         remaining_index: usize,
114         id: ExprOrPatId,
115     ) -> Option<(ValueNs, Option<Substitution>)> {
116         assert!(remaining_index < path.segments().len());
117         // there may be more intermediate segments between the resolved one and
118         // the end. Only the last segment needs to be resolved to a value; from
119         // the segments before that, we need to get either a type or a trait ref.
120
121         let resolved_segment = path.segments().get(remaining_index - 1).unwrap();
122         let remaining_segments = path.segments().skip(remaining_index);
123         let is_before_last = remaining_segments.len() == 1;
124
125         match (def, is_before_last) {
126             (TypeNs::TraitId(trait_), true) => {
127                 let segment =
128                     remaining_segments.last().expect("there should be at least one segment here");
129                 let ctx = crate::lower::TyLoweringContext::new(self.db, &self.resolver);
130                 let trait_ref =
131                     ctx.lower_trait_ref_from_resolved_path(trait_, resolved_segment, None);
132                 self.resolve_trait_assoc_item(trait_ref, segment, id)
133             }
134             (def, _) => {
135                 // Either we already have a type (e.g. `Vec::new`), or we have a
136                 // trait but it's not the last segment, so the next segment
137                 // should resolve to an associated type of that trait (e.g. `<T
138                 // as Iterator>::Item::default`)
139                 let remaining_segments_for_ty =
140                     remaining_segments.take(remaining_segments.len() - 1);
141                 let ctx = crate::lower::TyLoweringContext::new(self.db, &self.resolver);
142                 let (ty, _) = ctx.lower_partly_resolved_path(
143                     def,
144                     resolved_segment,
145                     remaining_segments_for_ty,
146                     true,
147                 );
148                 if let TyKind::Error = ty.kind(&Interner) {
149                     return None;
150                 }
151
152                 let ty = self.insert_type_vars(ty);
153                 let ty = self.normalize_associated_types_in(ty);
154
155                 let segment =
156                     remaining_segments.last().expect("there should be at least one segment here");
157
158                 self.resolve_ty_assoc_item(ty, &segment.name, id)
159             }
160         }
161     }
162
163     fn resolve_trait_assoc_item(
164         &mut self,
165         trait_ref: TraitRef,
166         segment: PathSegment<'_>,
167         id: ExprOrPatId,
168     ) -> Option<(ValueNs, Option<Substitution>)> {
169         let trait_ = trait_ref.hir_trait_id();
170         let item =
171             self.db.trait_data(trait_).items.iter().map(|(_name, id)| (*id)).find_map(|item| {
172                 match item {
173                     AssocItemId::FunctionId(func) => {
174                         if segment.name == &self.db.function_data(func).name {
175                             Some(AssocItemId::FunctionId(func))
176                         } else {
177                             None
178                         }
179                     }
180
181                     AssocItemId::ConstId(konst) => {
182                         if self
183                             .db
184                             .const_data(konst)
185                             .name
186                             .as_ref()
187                             .map_or(false, |n| n == segment.name)
188                         {
189                             Some(AssocItemId::ConstId(konst))
190                         } else {
191                             None
192                         }
193                     }
194                     AssocItemId::TypeAliasId(_) => None,
195                 }
196             })?;
197         let def = match item {
198             AssocItemId::FunctionId(f) => ValueNs::FunctionId(f),
199             AssocItemId::ConstId(c) => ValueNs::ConstId(c),
200             AssocItemId::TypeAliasId(_) => unreachable!(),
201         };
202
203         self.write_assoc_resolution(id, item);
204         Some((def, Some(trait_ref.substitution)))
205     }
206
207     fn resolve_ty_assoc_item(
208         &mut self,
209         ty: Ty,
210         name: &Name,
211         id: ExprOrPatId,
212     ) -> Option<(ValueNs, Option<Substitution>)> {
213         if let TyKind::Error = ty.kind(&Interner) {
214             return None;
215         }
216
217         if let Some(result) = self.resolve_enum_variant_on_ty(&ty, name, id) {
218             return Some(result);
219         }
220
221         let canonical_ty = self.canonicalizer().canonicalize_ty(ty.clone());
222         let krate = self.resolver.krate()?;
223         let traits_in_scope = self.resolver.traits_in_scope(self.db.upcast());
224
225         method_resolution::iterate_method_candidates(
226             &canonical_ty.value,
227             self.db,
228             self.trait_env.clone(),
229             krate,
230             &traits_in_scope,
231             None,
232             Some(name),
233             method_resolution::LookupMode::Path,
234             move |_ty, item| {
235                 let (def, container) = match item {
236                     AssocItemId::FunctionId(f) => {
237                         (ValueNs::FunctionId(f), f.lookup(self.db.upcast()).container)
238                     }
239                     AssocItemId::ConstId(c) => {
240                         (ValueNs::ConstId(c), c.lookup(self.db.upcast()).container)
241                     }
242                     AssocItemId::TypeAliasId(_) => unreachable!(),
243                 };
244                 let substs = match container {
245                     AssocContainerId::ImplId(impl_id) => {
246                         let impl_substs = TyBuilder::subst_for_def(self.db, impl_id)
247                             .fill(iter::repeat_with(|| self.table.new_type_var()))
248                             .build();
249                         let impl_self_ty =
250                             self.db.impl_self_ty(impl_id).substitute(&Interner, &impl_substs);
251                         self.unify(&impl_self_ty, &ty);
252                         Some(impl_substs)
253                     }
254                     AssocContainerId::TraitId(trait_) => {
255                         // we're picking this method
256                         let trait_ref = TyBuilder::trait_ref(self.db, trait_)
257                             .push(ty.clone())
258                             .fill(std::iter::repeat_with(|| self.table.new_type_var()))
259                             .build();
260                         self.push_obligation(trait_ref.clone().cast(&Interner));
261                         Some(trait_ref.substitution)
262                     }
263                     AssocContainerId::ModuleId(_) => None,
264                 };
265
266                 self.write_assoc_resolution(id, item);
267                 Some((def, substs))
268             },
269         )
270     }
271
272     fn resolve_enum_variant_on_ty(
273         &mut self,
274         ty: &Ty,
275         name: &Name,
276         id: ExprOrPatId,
277     ) -> Option<(ValueNs, Option<Substitution>)> {
278         let (enum_id, subst) = match ty.as_adt() {
279             Some((AdtId::EnumId(e), subst)) => (e, subst),
280             _ => return None,
281         };
282         let enum_data = self.db.enum_data(enum_id);
283         let local_id = enum_data.variant(name)?;
284         let variant = EnumVariantId { parent: enum_id, local_id };
285         self.write_variant_resolution(id, variant.into());
286         Some((ValueNs::EnumVariantId(variant), Some(subst.clone())))
287     }
288 }