1 use super::{probe, MethodCallee};
3 use crate::astconv::{AstConv, CreateSubstsForGenericArgsCtxt, IsMethodCall};
4 use crate::check::{callee, FnCtxt};
5 use crate::hir::def_id::DefId;
6 use crate::hir::GenericArg;
8 use rustc_infer::infer::{self, InferOk};
9 use rustc_middle::traits::{ObligationCauseCode, UnifyReceiverContext};
10 use rustc_middle::ty::adjustment::{Adjust, Adjustment, PointerCast};
11 use rustc_middle::ty::adjustment::{AllowTwoPhase, AutoBorrow, AutoBorrowMutability};
12 use rustc_middle::ty::fold::TypeFoldable;
13 use rustc_middle::ty::subst::{self, Subst, SubstsRef};
14 use rustc_middle::ty::{self, GenericParamDefKind, Ty};
16 use rustc_trait_selection::traits;
20 struct ConfirmContext<'a, 'tcx> {
21 fcx: &'a FnCtxt<'a, 'tcx>,
23 self_expr: &'tcx hir::Expr<'tcx>,
24 call_expr: &'tcx hir::Expr<'tcx>,
27 impl<'a, 'tcx> Deref for ConfirmContext<'a, 'tcx> {
28 type Target = FnCtxt<'a, 'tcx>;
29 fn deref(&self) -> &Self::Target {
35 pub struct ConfirmResult<'tcx> {
36 pub callee: MethodCallee<'tcx>,
37 pub illegal_sized_bound: Option<Span>,
40 impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
41 pub fn confirm_method(
44 self_expr: &'tcx hir::Expr<'tcx>,
45 call_expr: &'tcx hir::Expr<'tcx>,
46 unadjusted_self_ty: Ty<'tcx>,
47 pick: probe::Pick<'tcx>,
48 segment: &hir::PathSegment<'_>,
49 ) -> ConfirmResult<'tcx> {
51 "confirm(unadjusted_self_ty={:?}, pick={:?}, generic_args={:?})",
52 unadjusted_self_ty, pick, segment.args,
55 let mut confirm_cx = ConfirmContext::new(self, span, self_expr, call_expr);
56 confirm_cx.confirm(unadjusted_self_ty, pick, segment)
60 impl<'a, 'tcx> ConfirmContext<'a, 'tcx> {
62 fcx: &'a FnCtxt<'a, 'tcx>,
64 self_expr: &'tcx hir::Expr<'tcx>,
65 call_expr: &'tcx hir::Expr<'tcx>,
66 ) -> ConfirmContext<'a, 'tcx> {
67 ConfirmContext { fcx, span, self_expr, call_expr }
72 unadjusted_self_ty: Ty<'tcx>,
73 pick: probe::Pick<'tcx>,
74 segment: &hir::PathSegment<'_>,
75 ) -> ConfirmResult<'tcx> {
76 // Adjust the self expression the user provided and obtain the adjusted type.
77 let self_ty = self.adjust_self_ty(unadjusted_self_ty, &pick);
79 // Create substitutions for the method's type parameters.
80 let rcvr_substs = self.fresh_receiver_substs(self_ty, &pick);
81 let all_substs = self.instantiate_method_substs(&pick, segment, rcvr_substs);
83 debug!("all_substs={:?}", all_substs);
85 // Create the final signature for the method, replacing late-bound regions.
86 let (method_sig, method_predicates) = self.instantiate_method_sig(&pick, all_substs);
88 // Unify the (adjusted) self type with what the method expects.
90 // SUBTLE: if we want good error messages, because of "guessing" while matching
91 // traits, no trait system method can be called before this point because they
92 // could alter our Self-type, except for normalizing the receiver from the
93 // signature (which is also done during probing).
94 let method_sig_rcvr = self.normalize_associated_types_in(self.span, method_sig.inputs()[0]);
96 "confirm: self_ty={:?} method_sig_rcvr={:?} method_sig={:?} method_predicates={:?}",
97 self_ty, method_sig_rcvr, method_sig, method_predicates
99 self.unify_receivers(self_ty, method_sig_rcvr, &pick, all_substs);
101 let (method_sig, method_predicates) =
102 self.normalize_associated_types_in(self.span, (method_sig, method_predicates));
104 // Make sure nobody calls `drop()` explicitly.
105 self.enforce_illegal_method_limitations(&pick);
107 // If there is a `Self: Sized` bound and `Self` is a trait object, it is possible that
108 // something which derefs to `Self` actually implements the trait and the caller
109 // wanted to make a static dispatch on it but forgot to import the trait.
110 // See test `src/test/ui/issue-35976.rs`.
112 // In that case, we'll error anyway, but we'll also re-run the search with all traits
113 // in scope, and if we find another method which can be used, we'll output an
114 // appropriate hint suggesting to import the trait.
115 let illegal_sized_bound = self.predicates_require_illegal_sized_bound(&method_predicates);
117 // Add any trait/regions obligations specified on the method's type parameters.
118 // We won't add these if we encountered an illegal sized bound, so that we can use
119 // a custom error in that case.
120 if illegal_sized_bound.is_none() {
121 let method_ty = self.tcx.mk_fn_ptr(ty::Binder::bind(method_sig));
122 self.add_obligations(method_ty, all_substs, method_predicates);
125 // Create the final `MethodCallee`.
126 let callee = MethodCallee { def_id: pick.item.def_id, substs: all_substs, sig: method_sig };
127 ConfirmResult { callee, illegal_sized_bound }
130 ///////////////////////////////////////////////////////////////////////////
135 unadjusted_self_ty: Ty<'tcx>,
136 pick: &probe::Pick<'tcx>,
138 // Commit the autoderefs by calling `autoderef` again, but this
139 // time writing the results into the various typeck results.
141 self.autoderef_overloaded_span(self.span, unadjusted_self_ty, self.call_expr.span);
142 let (_, n) = match autoderef.nth(pick.autoderefs) {
145 return self.tcx.ty_error_with_message(
146 rustc_span::DUMMY_SP,
147 &format!("failed autoderef {}", pick.autoderefs),
151 assert_eq!(n, pick.autoderefs);
153 let mut adjustments = self.adjust_steps(&autoderef);
156 self.structurally_resolved_type(autoderef.span(), autoderef.final_ty(false));
158 if let Some(mutbl) = pick.autoref {
159 let region = self.next_region_var(infer::Autoref(self.span, pick.item));
160 target = self.tcx.mk_ref(region, ty::TypeAndMut { mutbl, ty: target });
161 let mutbl = match mutbl {
162 hir::Mutability::Not => AutoBorrowMutability::Not,
163 hir::Mutability::Mut => AutoBorrowMutability::Mut {
164 // Method call receivers are the primary use case
165 // for two-phase borrows.
166 allow_two_phase_borrow: AllowTwoPhase::Yes,
170 .push(Adjustment { kind: Adjust::Borrow(AutoBorrow::Ref(region, mutbl)), target });
172 if let Some(unsize_target) = pick.unsize {
175 .mk_ref(region, ty::TypeAndMut { mutbl: mutbl.into(), ty: unsize_target });
176 adjustments.push(Adjustment { kind: Adjust::Pointer(PointerCast::Unsize), target });
179 // No unsizing should be performed without autoref (at
180 // least during method dispach). This is because we
181 // currently only unsize `[T;N]` to `[T]`, and naturally
182 // that must occur being a reference.
183 assert!(pick.unsize.is_none());
186 self.register_predicates(autoderef.into_obligations());
188 // Write out the final adjustments.
189 self.apply_adjustments(self.self_expr, adjustments);
194 /// Returns a set of substitutions for the method *receiver* where all type and region
195 /// parameters are instantiated with fresh variables. This substitution does not include any
196 /// parameters declared on the method itself.
198 /// Note that this substitution may include late-bound regions from the impl level. If so,
199 /// these are instantiated later in the `instantiate_method_sig` routine.
200 fn fresh_receiver_substs(
203 pick: &probe::Pick<'tcx>,
204 ) -> SubstsRef<'tcx> {
206 probe::InherentImplPick => {
207 let impl_def_id = pick.item.container.id();
209 self.tcx.impl_trait_ref(impl_def_id).is_none(),
210 "impl {:?} is not an inherent impl",
213 self.fresh_substs_for_item(self.span, impl_def_id)
216 probe::ObjectPick => {
217 let trait_def_id = pick.item.container.id();
218 self.extract_existential_trait_ref(self_ty, |this, object_ty, principal| {
219 // The object data has no entry for the Self
220 // Type. For the purposes of this method call, we
221 // substitute the object type itself. This
222 // wouldn't be a sound substitution in all cases,
223 // since each instance of the object type is a
224 // different existential and hence could match
225 // distinct types (e.g., if `Self` appeared as an
226 // argument type), but those cases have already
227 // been ruled out when we deemed the trait to be
229 let original_poly_trait_ref = principal.with_self_ty(this.tcx, object_ty);
230 let upcast_poly_trait_ref = this.upcast(original_poly_trait_ref, trait_def_id);
231 let upcast_trait_ref =
232 this.replace_bound_vars_with_fresh_vars(upcast_poly_trait_ref);
234 "original_poly_trait_ref={:?} upcast_trait_ref={:?} target_trait={:?}",
235 original_poly_trait_ref, upcast_trait_ref, trait_def_id
237 upcast_trait_ref.substs
241 probe::TraitPick => {
242 let trait_def_id = pick.item.container.id();
244 // Make a trait reference `$0 : Trait<$1...$n>`
245 // consisting entirely of type variables. Later on in
246 // the process we will unify the transformed-self-type
247 // of the method with the actual type in order to
248 // unify some of these variables.
249 self.fresh_substs_for_item(self.span, trait_def_id)
252 probe::WhereClausePick(poly_trait_ref) => {
253 // Where clauses can have bound regions in them. We need to instantiate
254 // those to convert from a poly-trait-ref to a trait-ref.
255 self.replace_bound_vars_with_fresh_vars(poly_trait_ref).substs
260 fn extract_existential_trait_ref<R, F>(&mut self, self_ty: Ty<'tcx>, mut closure: F) -> R
262 F: FnMut(&mut ConfirmContext<'a, 'tcx>, Ty<'tcx>, ty::PolyExistentialTraitRef<'tcx>) -> R,
264 // If we specified that this is an object method, then the
265 // self-type ought to be something that can be dereferenced to
266 // yield an object-type (e.g., `&Object` or `Box<Object>`
269 // FIXME: this feels, like, super dubious
271 .autoderef(self.span, self_ty)
272 .include_raw_pointers()
273 .find_map(|(ty, _)| match ty.kind() {
274 ty::Dynamic(ref data, ..) => Some(closure(
277 data.principal().unwrap_or_else(|| {
278 span_bug!(self.span, "calling trait method on empty object?")
286 "self-type `{}` for ObjectPick never dereferenced to an object",
292 fn instantiate_method_substs(
294 pick: &probe::Pick<'tcx>,
295 seg: &hir::PathSegment<'_>,
296 parent_substs: SubstsRef<'tcx>,
297 ) -> SubstsRef<'tcx> {
298 // Determine the values for the generic parameters of the method.
299 // If they were not explicitly supplied, just construct fresh
301 let generics = self.tcx.generics_of(pick.item.def_id);
303 let arg_count_correct = AstConv::check_generic_arg_count_for_call(
312 // Create subst for early-bound lifetime parameters, combining
313 // parameters from the type and those from the method.
314 assert_eq!(generics.parent_count, parent_substs.len());
316 struct MethodSubstsCtxt<'a, 'tcx> {
317 cfcx: &'a ConfirmContext<'a, 'tcx>,
318 pick: &'a probe::Pick<'tcx>,
319 seg: &'a hir::PathSegment<'a>,
321 impl<'a, 'tcx> CreateSubstsForGenericArgsCtxt<'a, 'tcx> for MethodSubstsCtxt<'a, 'tcx> {
325 ) -> (Option<&'a hir::GenericArgs<'a>>, bool) {
326 if def_id == self.pick.item.def_id {
327 if let Some(ref data) = self.seg.args {
328 return (Some(data), false);
336 param: &ty::GenericParamDef,
337 arg: &GenericArg<'_>,
338 ) -> subst::GenericArg<'tcx> {
339 match (¶m.kind, arg) {
340 (GenericParamDefKind::Lifetime, GenericArg::Lifetime(lt)) => {
341 AstConv::ast_region_to_region(self.cfcx.fcx, lt, Some(param)).into()
343 (GenericParamDefKind::Type { .. }, GenericArg::Type(ty)) => {
344 self.cfcx.to_ty(ty).into()
346 (GenericParamDefKind::Const, GenericArg::Const(ct)) => {
347 self.cfcx.const_arg_to_const(&ct.value, param.def_id).into()
355 _substs: Option<&[subst::GenericArg<'tcx>]>,
356 param: &ty::GenericParamDef,
358 ) -> subst::GenericArg<'tcx> {
359 self.cfcx.var_for_def(self.cfcx.span, param)
362 AstConv::create_substs_for_generic_args(
369 &mut MethodSubstsCtxt { cfcx: self, pick, seg },
376 method_self_ty: Ty<'tcx>,
377 pick: &probe::Pick<'tcx>,
378 substs: SubstsRef<'tcx>,
381 "unify_receivers: self_ty={:?} method_self_ty={:?} span={:?} pick={:?}",
382 self_ty, method_self_ty, self.span, pick
384 let cause = self.cause(
386 ObligationCauseCode::UnifyReceiver(Box::new(UnifyReceiverContext {
387 assoc_item: pick.item,
388 param_env: self.param_env,
392 match self.at(&cause, self.param_env).sup(method_self_ty, self_ty) {
393 Ok(InferOk { obligations, value: () }) => {
394 self.register_predicates(obligations);
399 "{} was a subtype of {} but now is not?",
407 // NOTE: this returns the *unnormalized* predicates and method sig. Because of
408 // inference guessing, the predicates and method signature can't be normalized
409 // until we unify the `Self` type.
410 fn instantiate_method_sig(
412 pick: &probe::Pick<'tcx>,
413 all_substs: SubstsRef<'tcx>,
414 ) -> (ty::FnSig<'tcx>, ty::InstantiatedPredicates<'tcx>) {
415 debug!("instantiate_method_sig(pick={:?}, all_substs={:?})", pick, all_substs);
417 // Instantiate the bounds on the method with the
418 // type/early-bound-regions substitutions performed. There can
419 // be no late-bound regions appearing here.
420 let def_id = pick.item.def_id;
421 let method_predicates = self.tcx.predicates_of(def_id).instantiate(self.tcx, all_substs);
423 debug!("method_predicates after subst = {:?}", method_predicates);
425 let sig = self.tcx.fn_sig(def_id);
427 // Instantiate late-bound regions and substitute the trait
428 // parameters into the method type to get the actual method type.
430 // N.B., instantiate late-bound regions first so that
431 // `instantiate_type_scheme` can normalize associated types that
432 // may reference those regions.
433 let method_sig = self.replace_bound_vars_with_fresh_vars(sig);
434 debug!("late-bound lifetimes from method instantiated, method_sig={:?}", method_sig);
436 let method_sig = method_sig.subst(self.tcx, all_substs);
437 debug!("type scheme substituted, method_sig={:?}", method_sig);
439 (method_sig, method_predicates)
445 all_substs: SubstsRef<'tcx>,
446 method_predicates: ty::InstantiatedPredicates<'tcx>,
449 "add_obligations: fty={:?} all_substs={:?} method_predicates={:?}",
450 fty, all_substs, method_predicates
453 self.add_obligations_for_parameters(
454 traits::ObligationCause::misc(self.span, self.body_id),
458 // this is a projection from a trait reference, so we have to
459 // make sure that the trait reference inputs are well-formed.
460 self.add_wf_bounds(all_substs, self.call_expr);
462 // the function type must also be well-formed (this is not
463 // implied by the substs being well-formed because of inherent
464 // impls and late-bound regions - see issue #28609).
465 self.register_wf_obligation(fty.into(), self.span, traits::MiscObligation);
468 ///////////////////////////////////////////////////////////////////////////
471 fn predicates_require_illegal_sized_bound(
473 predicates: &ty::InstantiatedPredicates<'tcx>,
475 let sized_def_id = match self.tcx.lang_items().sized_trait() {
476 Some(def_id) => def_id,
480 traits::elaborate_predicates(self.tcx, predicates.predicates.iter().copied())
481 // We don't care about regions here.
482 .filter_map(|obligation| match obligation.predicate.kind().skip_binder() {
483 ty::PredicateKind::Trait(trait_pred, _) if trait_pred.def_id() == sized_def_id => {
484 let span = predicates
487 .zip(predicates.spans.iter())
490 if *p == obligation.predicate { Some(*span) } else { None }
493 .unwrap_or(rustc_span::DUMMY_SP);
494 Some((trait_pred, span))
498 .find_map(|(trait_pred, span)| match trait_pred.self_ty().kind() {
499 ty::Dynamic(..) => Some(span),
504 fn enforce_illegal_method_limitations(&self, pick: &probe::Pick<'_>) {
505 // Disallow calls to the method `drop` defined in the `Drop` trait.
506 match pick.item.container {
507 ty::TraitContainer(trait_def_id) => callee::check_legal_trait_for_method_call(
510 Some(self.self_expr.span),
514 ty::ImplContainer(..) => {}
520 source_trait_ref: ty::PolyTraitRef<'tcx>,
521 target_trait_def_id: DefId,
522 ) -> ty::PolyTraitRef<'tcx> {
523 let upcast_trait_refs =
524 traits::upcast_choices(self.tcx, source_trait_ref, target_trait_def_id);
526 // must be exactly one trait ref or we'd get an ambig error etc
527 if upcast_trait_refs.len() != 1 {
530 "cannot uniquely upcast `{:?}` to `{:?}`: `{:?}`",
537 upcast_trait_refs.into_iter().next().unwrap()
540 fn replace_bound_vars_with_fresh_vars<T>(&self, value: ty::Binder<T>) -> T
542 T: TypeFoldable<'tcx>,
544 self.fcx.replace_bound_vars_with_fresh_vars(self.span, infer::FnCall, value).0