Skip to main content

rustc_hir_typeck/
demand.rs

1use rustc_errors::{Applicability, Diag, MultiSpan, listify};
2use rustc_hir::def::Res;
3use rustc_hir::intravisit::Visitor;
4use rustc_hir::{self as hir, find_attr};
5use rustc_infer::infer::DefineOpaqueTypes;
6use rustc_middle::bug;
7use rustc_middle::ty::adjustment::AllowTwoPhase;
8use rustc_middle::ty::error::{ExpectedFound, TypeError};
9use rustc_middle::ty::print::with_no_trimmed_paths;
10use rustc_middle::ty::{self, AssocItem, BottomUpFolder, Ty, TypeFoldable, TypeVisitableExt};
11use rustc_span::{DUMMY_SP, Ident, Span, sym};
12use rustc_trait_selection::infer::InferCtxtExt;
13use rustc_trait_selection::traits::ObligationCause;
14use tracing::instrument;
15
16use super::method::probe;
17use crate::FnCtxt;
18
19impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
20    pub(crate) fn emit_type_mismatch_suggestions(
21        &self,
22        err: &mut Diag<'_>,
23        expr: &hir::Expr<'tcx>,
24        expr_ty: Ty<'tcx>,
25        expected: Ty<'tcx>,
26        expected_ty_expr: Option<&'tcx hir::Expr<'tcx>>,
27        error: Option<TypeError<'tcx>>,
28    ) {
29        if expr_ty == expected {
30            return;
31        }
32        self.annotate_alternative_method_deref(err, expr, error);
33        self.explain_self_literal(err, expr, expected, expr_ty);
34
35        // Use `||` to give these suggestions a precedence
36        let suggested = self.suggest_missing_parentheses(err, expr)
37            || self.suggest_missing_unwrap_expect(err, expr, expected, expr_ty)
38            || self.suggest_remove_last_method_call(err, expr, expected)
39            || self.suggest_associated_const(err, expr, expected)
40            || self.suggest_semicolon_in_repeat_expr(err, expr, expr_ty)
41            || self.suggest_deref_ref_or_into(err, expr, expected, expr_ty, expected_ty_expr)
42            || self.suggest_option_to_bool(err, expr, expr_ty, expected)
43            || self.suggest_collect(err, expr, expected, expr_ty)
44            || self.suggest_compatible_variants(err, expr, expected, expr_ty)
45            || self.suggest_non_zero_new_unwrap(err, expr, expected, expr_ty)
46            || self.suggest_calling_boxed_future_when_appropriate(err, expr, expected, expr_ty)
47            || self.suggest_no_capture_closure(err, expected, expr_ty)
48            || self.suggest_boxing_when_appropriate(
49                err,
50                expr.peel_blocks().span,
51                expr.hir_id,
52                expected,
53                expr_ty,
54            )
55            || self.suggest_block_to_brackets_peeling_refs(err, expr, expr_ty, expected)
56            || self.suggest_copied_cloned_or_as_ref(err, expr, expr_ty, expected)
57            || self.suggest_clone_for_ref(err, expr, expr_ty, expected)
58            || self.suggest_into(err, expr, expr_ty, expected)
59            || self.suggest_floating_point_literal(err, expr, expected)
60            || self.suggest_null_ptr_for_literal_zero_given_to_ptr_arg(err, expr, expected)
61            || self.suggest_coercing_result_via_try_operator(err, expr, expected, expr_ty)
62            || self.suggest_returning_value_after_loop(err, expr, expected);
63
64        if !suggested {
65            self.note_source_of_type_mismatch_constraint(
66                err,
67                expr,
68                TypeMismatchSource::Ty(expected),
69            );
70        }
71    }
72
73    pub(crate) fn emit_coerce_suggestions(
74        &self,
75        err: &mut Diag<'_>,
76        expr: &hir::Expr<'tcx>,
77        expr_ty: Ty<'tcx>,
78        expected: Ty<'tcx>,
79        expected_ty_expr: Option<&'tcx hir::Expr<'tcx>>,
80        error: Option<TypeError<'tcx>>,
81    ) {
82        if expr_ty == expected {
83            return;
84        }
85
86        self.annotate_expected_due_to_let_ty(err, expr, error);
87        self.annotate_loop_expected_due_to_inference(err, expr, error);
88        if self.annotate_mut_binding_to_immutable_binding(err, expr, expr_ty, expected, error) {
89            return;
90        }
91
92        // FIXME(#73154): For now, we do leak check when coercing function
93        // pointers in typeck, instead of only during borrowck. This can lead
94        // to these `RegionsInsufficientlyPolymorphic` errors that aren't helpful.
95        if #[allow(non_exhaustive_omitted_patterns)] match error {
    Some(TypeError::RegionsInsufficientlyPolymorphic(..)) => true,
    _ => false,
}matches!(error, Some(TypeError::RegionsInsufficientlyPolymorphic(..))) {
96            return;
97        }
98
99        if self.is_destruct_assignment_desugaring(expr) {
100            return;
101        }
102        self.emit_type_mismatch_suggestions(err, expr, expr_ty, expected, expected_ty_expr, error);
103        self.note_type_is_not_clone(err, expected, expr_ty, expr);
104        self.note_internal_mutation_in_method(err, expr, Some(expected), expr_ty);
105        self.suggest_method_call_on_range_literal(err, expr, expr_ty, expected);
106        self.suggest_return_binding_for_missing_tail_expr(err, expr, expr_ty, expected);
107        self.note_wrong_return_ty_due_to_generic_arg(err, expr, expr_ty);
108    }
109
110    /// Really hacky heuristic to remap an `assert_eq!` error to the user
111    /// expressions provided to the macro.
112    fn adjust_expr_for_assert_eq_macro(
113        &self,
114        found_expr: &mut &'tcx hir::Expr<'tcx>,
115        expected_expr: &mut Option<&'tcx hir::Expr<'tcx>>,
116    ) {
117        let Some(expected_expr) = expected_expr else {
118            return;
119        };
120
121        if !found_expr.span.eq_ctxt(expected_expr.span) {
122            return;
123        }
124
125        if !found_expr
126            .span
127            .ctxt()
128            .outer_expn_data()
129            .macro_def_id
130            .is_some_and(|def_id| self.tcx.is_diagnostic_item(sym::assert_eq_macro, def_id))
131        {
132            return;
133        }
134
135        let hir::ExprKind::Unary(
136            hir::UnOp::Deref,
137            hir::Expr { kind: hir::ExprKind::Path(found_path), .. },
138        ) = found_expr.kind
139        else {
140            return;
141        };
142        let hir::ExprKind::Unary(
143            hir::UnOp::Deref,
144            hir::Expr { kind: hir::ExprKind::Path(expected_path), .. },
145        ) = expected_expr.kind
146        else {
147            return;
148        };
149
150        for (path, name, idx, var) in [
151            (expected_path, "left_val", 0, expected_expr),
152            (found_path, "right_val", 1, found_expr),
153        ] {
154            if let hir::QPath::Resolved(_, path) = path
155                && let [segment] = path.segments
156                && segment.ident.name.as_str() == name
157                && let Res::Local(hir_id) = path.res
158                && let Some((_, hir::Node::Expr(match_expr))) =
159                    self.tcx.hir_parent_iter(hir_id).nth(2)
160                && let hir::ExprKind::Match(scrutinee, _, _) = match_expr.kind
161                && let hir::ExprKind::Tup(exprs) = scrutinee.kind
162                && let hir::ExprKind::AddrOf(_, _, macro_arg) = exprs[idx].kind
163            {
164                *var = macro_arg;
165            }
166        }
167    }
168
169    /// Requires that the two types unify, and prints an error message if
170    /// they don't.
171    pub(crate) fn demand_suptype(&self, sp: Span, expected: Ty<'tcx>, actual: Ty<'tcx>) {
172        if let Err(e) = self.demand_suptype_diag(sp, expected, actual) {
173            e.emit();
174        }
175    }
176
177    pub(crate) fn demand_suptype_diag(
178        &'a self,
179        sp: Span,
180        expected: Ty<'tcx>,
181        actual: Ty<'tcx>,
182    ) -> Result<(), Diag<'a>> {
183        self.demand_suptype_with_origin(&self.misc(sp), expected, actual)
184    }
185
186    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() ||
            { false } {
        __tracing_attr_span =
            {
                use ::tracing::__macro_support::Callsite as _;
                static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                    {
                        static META: ::tracing::Metadata<'static> =
                            {
                                ::tracing_core::metadata::Metadata::new("demand_suptype_with_origin",
                                    "rustc_hir_typeck::demand", ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/demand.rs"),
                                    ::tracing_core::__macro_support::Option::Some(186u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::demand"),
                                    ::tracing_core::field::FieldSet::new(&["cause", "expected",
                                                    "actual"],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::DEBUG <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::DEBUG <=
                                    ::tracing::level_filters::LevelFilter::current() &&
                            { interest = __CALLSITE.interest(); !interest.is_never() }
                        &&
                        ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                            interest) {
                    let meta = __CALLSITE.metadata();
                    ::tracing::Span::new(meta,
                        &{
                                #[allow(unused_imports)]
                                use ::tracing::field::{debug, display, Value};
                                let mut iter = meta.fields().iter();
                                meta.fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                    ::tracing::__macro_support::Option::Some(&::tracing::field::debug(&cause)
                                                            as &dyn Value)),
                                                (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                    ::tracing::__macro_support::Option::Some(&::tracing::field::debug(&expected)
                                                            as &dyn Value)),
                                                (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                    ::tracing::__macro_support::Option::Some(&::tracing::field::debug(&actual)
                                                            as &dyn Value))])
                            })
                } else {
                    let span =
                        ::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
                    {};
                    span
                }
            };
        __tracing_attr_guard = __tracing_attr_span.enter();
    }

    #[warn(clippy :: suspicious_else_formatting)]
    {

        #[allow(unknown_lints, unreachable_code, clippy ::
        diverging_sub_expression, clippy :: empty_loop, clippy ::
        let_unit_value, clippy :: let_with_type_underscore, clippy ::
        needless_return, clippy :: unreachable)]
        if false {
            let __tracing_attr_fake_return: Result<(), Diag<'a>> = loop {};
            return __tracing_attr_fake_return;
        }
        {
            self.at(cause,
                            self.param_env).sup(DefineOpaqueTypes::Yes, expected,
                        actual).map(|infer_ok|
                        self.register_infer_ok_obligations(infer_ok)).map_err(|e|
                    {
                        self.err_ctxt().report_mismatched_types(cause,
                            self.param_env, expected, actual, e)
                    })
        }
    }
}#[instrument(skip(self), level = "debug")]
187    pub(crate) fn demand_suptype_with_origin(
188        &'a self,
189        cause: &ObligationCause<'tcx>,
190        expected: Ty<'tcx>,
191        actual: Ty<'tcx>,
192    ) -> Result<(), Diag<'a>> {
193        self.at(cause, self.param_env)
194            .sup(DefineOpaqueTypes::Yes, expected, actual)
195            .map(|infer_ok| self.register_infer_ok_obligations(infer_ok))
196            .map_err(|e| {
197                self.err_ctxt().report_mismatched_types(cause, self.param_env, expected, actual, e)
198            })
199    }
200
201    pub(crate) fn demand_eqtype(&self, sp: Span, expected: Ty<'tcx>, actual: Ty<'tcx>) {
202        if let Err(err) = self.demand_eqtype_diag(sp, expected, actual) {
203            err.emit();
204        }
205    }
206
207    pub(crate) fn demand_eqtype_diag(
208        &'a self,
209        sp: Span,
210        expected: Ty<'tcx>,
211        actual: Ty<'tcx>,
212    ) -> Result<(), Diag<'a>> {
213        self.demand_eqtype_with_origin(&self.misc(sp), expected, actual)
214    }
215
216    pub(crate) fn demand_eqtype_with_origin(
217        &'a self,
218        cause: &ObligationCause<'tcx>,
219        expected: Ty<'tcx>,
220        actual: Ty<'tcx>,
221    ) -> Result<(), Diag<'a>> {
222        self.at(cause, self.param_env)
223            .eq(DefineOpaqueTypes::Yes, expected, actual)
224            .map(|infer_ok| self.register_infer_ok_obligations(infer_ok))
225            .map_err(|e| {
226                self.err_ctxt().report_mismatched_types(cause, self.param_env, expected, actual, e)
227            })
228    }
229
230    pub(crate) fn demand_coerce(
231        &self,
232        expr: &'tcx hir::Expr<'tcx>,
233        checked_ty: Ty<'tcx>,
234        expected: Ty<'tcx>,
235        expected_ty_expr: Option<&'tcx hir::Expr<'tcx>>,
236        allow_two_phase: AllowTwoPhase,
237    ) -> Ty<'tcx> {
238        match self.demand_coerce_diag(expr, checked_ty, expected, expected_ty_expr, allow_two_phase)
239        {
240            Ok(ty) => ty,
241            Err(err) => {
242                err.emit();
243                // Return the original type instead of an error type here, otherwise the type of `x` in
244                // `let x: u32 = ();` will be a type error, causing all subsequent usages of `x` to not
245                // report errors, even though `x` is definitely `u32`.
246                expected
247            }
248        }
249    }
250
251    /// Checks that the type of `expr` can be coerced to `expected`.
252    ///
253    /// N.B., this code relies on `self.diverges` to be accurate. In particular, assignments to `!`
254    /// will be permitted if the diverges flag is currently "always".
255    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() ||
            { false } {
        __tracing_attr_span =
            {
                use ::tracing::__macro_support::Callsite as _;
                static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                    {
                        static META: ::tracing::Metadata<'static> =
                            {
                                ::tracing_core::metadata::Metadata::new("demand_coerce_diag",
                                    "rustc_hir_typeck::demand", ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/demand.rs"),
                                    ::tracing_core::__macro_support::Option::Some(255u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::demand"),
                                    ::tracing_core::field::FieldSet::new(&["checked_ty",
                                                    "expected"],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::DEBUG <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::DEBUG <=
                                    ::tracing::level_filters::LevelFilter::current() &&
                            { interest = __CALLSITE.interest(); !interest.is_never() }
                        &&
                        ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                            interest) {
                    let meta = __CALLSITE.metadata();
                    ::tracing::Span::new(meta,
                        &{
                                #[allow(unused_imports)]
                                use ::tracing::field::{debug, display, Value};
                                let mut iter = meta.fields().iter();
                                meta.fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                    ::tracing::__macro_support::Option::Some(&::tracing::field::debug(&checked_ty)
                                                            as &dyn Value)),
                                                (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                    ::tracing::__macro_support::Option::Some(&::tracing::field::debug(&expected)
                                                            as &dyn Value))])
                            })
                } else {
                    let span =
                        ::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
                    {};
                    span
                }
            };
        __tracing_attr_guard = __tracing_attr_span.enter();
    }

    #[warn(clippy :: suspicious_else_formatting)]
    {

        #[allow(unknown_lints, unreachable_code, clippy ::
        diverging_sub_expression, clippy :: empty_loop, clippy ::
        let_unit_value, clippy :: let_with_type_underscore, clippy ::
        needless_return, clippy :: unreachable)]
        if false {
            let __tracing_attr_fake_return: Result<Ty<'tcx>, Diag<'a>> =
                loop {};
            return __tracing_attr_fake_return;
        }
        {
            let expected = self.resolve_vars_with_obligations(expected);
            let e =
                match self.coerce(expr, checked_ty, expected, allow_two_phase,
                        None) {
                    Ok(ty) => return Ok(ty),
                    Err(e) => e,
                };
            self.adjust_expr_for_assert_eq_macro(&mut expr,
                &mut expected_ty_expr);
            self.set_tainted_by_errors(self.dcx().span_delayed_bug(expr.span,
                    "`TypeError` when attempting coercion but no error emitted"));
            let expr = expr.peel_drop_temps();
            let cause = self.misc(expr.span);
            let expr_ty = self.resolve_vars_if_possible(checked_ty);
            let mut err =
                self.err_ctxt().report_mismatched_types(&cause,
                    self.param_env, expected, expr_ty, e);
            self.emit_coerce_suggestions(&mut err, expr, expr_ty, expected,
                expected_ty_expr, Some(e));
            Err(err)
        }
    }
}#[instrument(level = "debug", skip(self, expr, expected_ty_expr, allow_two_phase))]
256    pub(crate) fn demand_coerce_diag(
257        &'a self,
258        mut expr: &'tcx hir::Expr<'tcx>,
259        checked_ty: Ty<'tcx>,
260        expected: Ty<'tcx>,
261        mut expected_ty_expr: Option<&'tcx hir::Expr<'tcx>>,
262        allow_two_phase: AllowTwoPhase,
263    ) -> Result<Ty<'tcx>, Diag<'a>> {
264        let expected = self.resolve_vars_with_obligations(expected);
265
266        let e = match self.coerce(expr, checked_ty, expected, allow_two_phase, None) {
267            Ok(ty) => return Ok(ty),
268            Err(e) => e,
269        };
270
271        self.adjust_expr_for_assert_eq_macro(&mut expr, &mut expected_ty_expr);
272
273        self.set_tainted_by_errors(self.dcx().span_delayed_bug(
274            expr.span,
275            "`TypeError` when attempting coercion but no error emitted",
276        ));
277        let expr = expr.peel_drop_temps();
278        let cause = self.misc(expr.span);
279        let expr_ty = self.resolve_vars_if_possible(checked_ty);
280        let mut err =
281            self.err_ctxt().report_mismatched_types(&cause, self.param_env, expected, expr_ty, e);
282
283        self.emit_coerce_suggestions(&mut err, expr, expr_ty, expected, expected_ty_expr, Some(e));
284
285        Err(err)
286    }
287
288    /// Notes the point at which a variable is constrained to some type incompatible
289    /// with some expectation given by `source`.
290    pub(crate) fn note_source_of_type_mismatch_constraint(
291        &self,
292        err: &mut Diag<'_>,
293        expr: &hir::Expr<'_>,
294        source: TypeMismatchSource<'tcx>,
295    ) -> bool {
296        let hir::ExprKind::Path(hir::QPath::Resolved(None, p)) = expr.kind else {
297            return false;
298        };
299        let [hir::PathSegment { ident, args: None, .. }] = p.segments else {
300            return false;
301        };
302        let hir::def::Res::Local(local_hir_id) = p.res else {
303            return false;
304        };
305        let hir::Node::Pat(pat) = self.tcx.hir_node(local_hir_id) else {
306            return false;
307        };
308        let (init_ty_hir_id, init) = match self.tcx.parent_hir_node(pat.hir_id) {
309            hir::Node::LetStmt(hir::LetStmt { ty: Some(ty), init, .. }) => (ty.hir_id, *init),
310            hir::Node::LetStmt(hir::LetStmt { init: Some(init), .. }) => (init.hir_id, Some(*init)),
311            _ => return false,
312        };
313        let Some(init_ty) = self.node_ty_opt(init_ty_hir_id) else {
314            return false;
315        };
316
317        // Locate all the usages of the relevant binding.
318        struct FindExprs<'tcx> {
319            hir_id: hir::HirId,
320            uses: Vec<&'tcx hir::Expr<'tcx>>,
321        }
322        impl<'tcx> Visitor<'tcx> for FindExprs<'tcx> {
323            fn visit_expr(&mut self, ex: &'tcx hir::Expr<'tcx>) {
324                if let hir::ExprKind::Path(hir::QPath::Resolved(None, path)) = ex.kind
325                    && let hir::def::Res::Local(hir_id) = path.res
326                    && hir_id == self.hir_id
327                {
328                    self.uses.push(ex);
329                }
330                hir::intravisit::walk_expr(self, ex);
331            }
332        }
333
334        let mut expr_finder = FindExprs { hir_id: local_hir_id, uses: init.into_iter().collect() };
335        let body = self.tcx.hir_body_owned_by(self.body_id);
336        expr_finder.visit_expr(body.value);
337
338        // Replaces all of the variables in the given type with a fresh inference variable.
339        let mut fudger = BottomUpFolder {
340            tcx: self.tcx,
341            ty_op: |ty| {
342                if let ty::Infer(infer) = ty.kind() {
343                    match infer {
344                        ty::TyVar(_) => self.next_ty_var(DUMMY_SP),
345                        ty::IntVar(_) => self.next_int_var(),
346                        ty::FloatVar(_) => self.next_float_var(DUMMY_SP, None),
347                        ty::FreshTy(_) | ty::FreshIntTy(_) | ty::FreshFloatTy(_) => {
348                            ::rustc_middle::util::bug::bug_fmt(format_args!("unexpected fresh ty outside of the trait solver"))bug!("unexpected fresh ty outside of the trait solver")
349                        }
350                    }
351                } else {
352                    ty
353                }
354            },
355            lt_op: |_| self.tcx.lifetimes.re_erased,
356            ct_op: |ct| {
357                if let ty::ConstKind::Infer(_) = ct.kind() {
358                    self.next_const_var(DUMMY_SP)
359                } else {
360                    ct
361                }
362            },
363        };
364
365        let expected_ty = match source {
366            TypeMismatchSource::Ty(expected_ty) => expected_ty,
367            // Try to deduce what the possible value of `expr` would be if the
368            // incompatible arg were compatible. For example, given `Vec<i32>`
369            // and `vec.push(1u32)`, we ideally want to deduce that the type of
370            // `vec` *should* have been `Vec<u32>`. This will allow us to then
371            // run the subsequent code with this expectation, finding out exactly
372            // when this type diverged from our expectation.
373            TypeMismatchSource::Arg { call_expr, incompatible_arg: idx } => {
374                let hir::ExprKind::MethodCall(segment, _, args, _) = call_expr.kind else {
375                    return false;
376                };
377                let Some(arg_ty) = self.node_ty_opt(args[idx].hir_id) else {
378                    return false;
379                };
380                let possible_rcvr_ty = expr_finder.uses.iter().rev().find_map(|binding| {
381                    let possible_rcvr_ty = self.node_ty_opt(binding.hir_id)?;
382                    if possible_rcvr_ty.is_ty_var() {
383                        return None;
384                    }
385                    // Fudge the receiver, so we can do new inference on it.
386                    let possible_rcvr_ty = possible_rcvr_ty.fold_with(&mut fudger);
387                    let method = self
388                        .lookup_method_for_diagnostic(
389                            possible_rcvr_ty,
390                            segment,
391                            DUMMY_SP,
392                            call_expr,
393                            binding,
394                        )
395                        .ok()?;
396                    // Make sure we select the same method that we started with...
397                    if Some(method.def_id)
398                        != self.typeck_results.borrow().type_dependent_def_id(call_expr.hir_id)
399                    {
400                        return None;
401                    }
402                    // Unify the method signature with our incompatible arg, to
403                    // do inference in the *opposite* direction and to find out
404                    // what our ideal rcvr ty would look like.
405                    let _ = self
406                        .at(&ObligationCause::dummy(), self.param_env)
407                        .eq(DefineOpaqueTypes::Yes, method.sig.inputs()[idx + 1], arg_ty)
408                        .ok()?;
409                    self.select_obligations_where_possible(|errs| {
410                        // Yeet the errors, we're already reporting errors.
411                        errs.clear();
412                    });
413                    Some(self.resolve_vars_if_possible(possible_rcvr_ty))
414                });
415                let Some(rcvr_ty) = possible_rcvr_ty else { return false };
416                rcvr_ty
417            }
418        };
419
420        // If our expected_ty does not equal init_ty, then it *began* as incompatible.
421        // No need to note in this case...
422        if !self.can_eq(self.param_env, expected_ty, init_ty.fold_with(&mut fudger)) {
423            return false;
424        }
425
426        for window in expr_finder.uses.windows(2) {
427            // Bindings always update their recorded type after the fact, so we
428            // need to look at the *following* usage's type to see when the
429            // binding became incompatible.
430            let [binding, next_usage] = *window else {
431                continue;
432            };
433
434            // Don't go past the binding (always gonna be a nonsense label if so)
435            if binding.hir_id == expr.hir_id {
436                break;
437            }
438
439            let Some(next_use_ty) = self.node_ty_opt(next_usage.hir_id) else {
440                continue;
441            };
442
443            // If the type is not constrained in a way making it not possible to
444            // equate with `expected_ty` by this point, skip.
445            if self.can_eq(self.param_env, expected_ty, next_use_ty.fold_with(&mut fudger)) {
446                continue;
447            }
448
449            if let hir::Node::Expr(parent_expr) = self.tcx.parent_hir_node(binding.hir_id)
450                && let hir::ExprKind::MethodCall(segment, rcvr, args, _) = parent_expr.kind
451                && rcvr.hir_id == binding.hir_id
452            {
453                // If our binding became incompatible while it was a receiver
454                // to a method call, we may be able to make a better guess to
455                // the source of a type mismatch.
456                let Some(rcvr_ty) = self.node_ty_opt(rcvr.hir_id) else {
457                    continue;
458                };
459                let rcvr_ty = rcvr_ty.fold_with(&mut fudger);
460                let Ok(method) = self.lookup_method_for_diagnostic(
461                    rcvr_ty,
462                    segment,
463                    DUMMY_SP,
464                    parent_expr,
465                    rcvr,
466                ) else {
467                    continue;
468                };
469                // Make sure we select the same method that we started with...
470                if Some(method.def_id)
471                    != self.typeck_results.borrow().type_dependent_def_id(parent_expr.hir_id)
472                {
473                    continue;
474                }
475
476                let ideal_rcvr_ty = rcvr_ty.fold_with(&mut fudger);
477                let ideal_method = self
478                    .lookup_method_for_diagnostic(
479                        ideal_rcvr_ty,
480                        segment,
481                        DUMMY_SP,
482                        parent_expr,
483                        rcvr,
484                    )
485                    .ok()
486                    .and_then(|method| {
487                        let _ = self
488                            .at(&ObligationCause::dummy(), self.param_env)
489                            .eq(DefineOpaqueTypes::Yes, ideal_rcvr_ty, expected_ty)
490                            .ok()?;
491                        Some(method)
492                    });
493
494                // Find what argument caused our rcvr to become incompatible
495                // with the expected ty.
496                for (idx, (expected_arg_ty, arg_expr)) in
497                    std::iter::zip(&method.sig.inputs()[1..], args).enumerate()
498                {
499                    let Some(arg_ty) = self.node_ty_opt(arg_expr.hir_id) else {
500                        continue;
501                    };
502                    let arg_ty = arg_ty.fold_with(&mut fudger);
503                    let _ =
504                        self.coerce(arg_expr, arg_ty, *expected_arg_ty, AllowTwoPhase::No, None);
505                    self.select_obligations_where_possible(|errs| {
506                        // Yeet the errors, we're already reporting errors.
507                        errs.clear();
508                    });
509                    // If our rcvr, after inference due to unifying the signature
510                    // with the expected argument type, is still compatible with
511                    // the rcvr, then it must've not been the source of blame.
512                    if self.can_eq(self.param_env, rcvr_ty, expected_ty) {
513                        continue;
514                    }
515                    err.span_label(arg_expr.span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("this argument has type `{0}`...",
                arg_ty))
    })format!("this argument has type `{arg_ty}`..."));
516                    err.span_label(
517                        binding.span,
518                        ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("... which causes `{0}` to have type `{1}`",
                ident, next_use_ty))
    })format!("... which causes `{ident}` to have type `{next_use_ty}`"),
519                    );
520                    // Using our "ideal" method signature, suggest a fix to this
521                    // blame arg, if possible. Don't do this if we're coming from
522                    // arg mismatch code, because we'll possibly suggest a mutually
523                    // incompatible fix at the original mismatch site.
524                    // HACK(compiler-errors): We don't actually consider the implications
525                    // of our inference guesses in `emit_type_mismatch_suggestions`, so
526                    // only suggest things when we know our type error is precisely due to
527                    // a type mismatch, and not via some projection or something. See #116155.
528                    if #[allow(non_exhaustive_omitted_patterns)] match source {
    TypeMismatchSource::Ty(_) => true,
    _ => false,
}matches!(source, TypeMismatchSource::Ty(_))
529                        && let Some(ideal_method) = ideal_method
530                        && Some(ideal_method.def_id)
531                            == self
532                                .typeck_results
533                                .borrow()
534                                .type_dependent_def_id(parent_expr.hir_id)
535                        && let ideal_arg_ty =
536                            self.resolve_vars_if_possible(ideal_method.sig.inputs()[idx + 1])
537                        && !ideal_arg_ty.has_non_region_infer()
538                    {
539                        self.emit_type_mismatch_suggestions(
540                            err,
541                            arg_expr,
542                            arg_ty,
543                            ideal_arg_ty,
544                            None,
545                            None,
546                        );
547                    }
548                    return true;
549                }
550            }
551            err.span_label(
552                binding.span,
553                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("here the type of `{0}` is inferred to be `{1}`",
                ident, next_use_ty))
    })format!("here the type of `{ident}` is inferred to be `{next_use_ty}`"),
554            );
555            return true;
556        }
557
558        // We must've not found something that constrained the expr.
559        false
560    }
561
562    // When encountering a type error on the value of a `break`, try to point at the reason for the
563    // expected type.
564    pub(crate) fn annotate_loop_expected_due_to_inference(
565        &self,
566        err: &mut Diag<'_>,
567        expr: &hir::Expr<'_>,
568        error: Option<TypeError<'tcx>>,
569    ) {
570        let Some(TypeError::Sorts(ExpectedFound { expected, .. })) = error else {
571            return;
572        };
573        let mut parent_id = self.tcx.parent_hir_id(expr.hir_id);
574        let mut parent;
575        'outer: loop {
576            // Climb the HIR tree to see if the current `Expr` is part of a `break;` statement.
577            let (hir::Node::Stmt(&hir::Stmt { kind: hir::StmtKind::Semi(p), .. })
578            | hir::Node::Block(&hir::Block { expr: Some(p), .. })
579            | hir::Node::Expr(p)) = self.tcx.hir_node(parent_id)
580            else {
581                break;
582            };
583            parent = p;
584            parent_id = self.tcx.parent_hir_id(parent_id);
585            let hir::ExprKind::Break(destination, _) = parent.kind else {
586                continue;
587            };
588            let mut parent_id = parent_id;
589            let mut direct = false;
590            loop {
591                // Climb the HIR tree to find the (desugared) `loop` this `break` corresponds to.
592                let parent = match self.tcx.hir_node(parent_id) {
593                    hir::Node::Expr(parent) => {
594                        parent_id = self.tcx.parent_hir_id(parent.hir_id);
595                        parent
596                    }
597                    hir::Node::Stmt(hir::Stmt {
598                        hir_id,
599                        kind: hir::StmtKind::Semi(parent) | hir::StmtKind::Expr(parent),
600                        ..
601                    }) => {
602                        parent_id = self.tcx.parent_hir_id(*hir_id);
603                        parent
604                    }
605                    hir::Node::Stmt(hir::Stmt { hir_id, kind: hir::StmtKind::Let(_), .. }) => {
606                        parent_id = self.tcx.parent_hir_id(*hir_id);
607                        parent
608                    }
609                    hir::Node::LetStmt(hir::LetStmt { hir_id, .. }) => {
610                        parent_id = self.tcx.parent_hir_id(*hir_id);
611                        parent
612                    }
613                    hir::Node::Block(_) => {
614                        parent_id = self.tcx.parent_hir_id(parent_id);
615                        parent
616                    }
617                    _ => break,
618                };
619                if let hir::ExprKind::Loop(..) = parent.kind {
620                    // When you have `'a: loop { break; }`, the `break` corresponds to the labeled
621                    // loop, so we need to account for that.
622                    direct = !direct;
623                }
624                if let hir::ExprKind::Loop(block, label, _, span) = parent.kind
625                    && (destination.label == label || direct)
626                {
627                    if let Some((reason_span, message)) =
628                        self.maybe_get_coercion_reason(parent_id, parent.span)
629                    {
630                        err.span_label(reason_span, message);
631                        err.span_label(
632                            span,
633                            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("this loop is expected to be of type `{0}`",
                expected))
    })format!("this loop is expected to be of type `{expected}`"),
634                        );
635                        break 'outer;
636                    } else {
637                        // Locate all other `break` statements within the same `loop` that might
638                        // have affected inference.
639                        struct FindBreaks<'tcx> {
640                            label: Option<rustc_ast::Label>,
641                            uses: Vec<&'tcx hir::Expr<'tcx>>,
642                            nest_depth: usize,
643                        }
644                        impl<'tcx> Visitor<'tcx> for FindBreaks<'tcx> {
645                            fn visit_expr(&mut self, ex: &'tcx hir::Expr<'tcx>) {
646                                let nest_depth = self.nest_depth;
647                                if let hir::ExprKind::Loop(_, label, _, _) = ex.kind {
648                                    if label == self.label {
649                                        // Account for `'a: loop { 'a: loop {...} }`.
650                                        return;
651                                    }
652                                    self.nest_depth += 1;
653                                }
654                                if let hir::ExprKind::Break(destination, _) = ex.kind
655                                    && (self.label == destination.label
656                                        // Account for `loop { 'a: loop { loop { break; } } }`.
657                                        || destination.label.is_none() && self.nest_depth == 0)
658                                {
659                                    self.uses.push(ex);
660                                }
661                                hir::intravisit::walk_expr(self, ex);
662                                self.nest_depth = nest_depth;
663                            }
664                        }
665                        let mut expr_finder = FindBreaks { label, uses: ::alloc::vec::Vec::new()vec![], nest_depth: 0 };
666                        expr_finder.visit_block(block);
667                        let mut exit = false;
668                        for ex in expr_finder.uses {
669                            let hir::ExprKind::Break(_, val) = ex.kind else {
670                                continue;
671                            };
672                            let ty = match val {
673                                Some(val) => {
674                                    match self.typeck_results.borrow().expr_ty_adjusted_opt(val) {
675                                        None => continue,
676                                        Some(ty) => ty,
677                                    }
678                                }
679                                None => self.tcx.types.unit,
680                            };
681                            if self.can_eq(self.param_env, ty, expected) {
682                                err.span_label(ex.span, "expected because of this `break`");
683                                exit = true;
684                            }
685                        }
686                        if exit {
687                            break 'outer;
688                        }
689                    }
690                }
691            }
692        }
693    }
694
695    fn annotate_expected_due_to_let_ty(
696        &self,
697        err: &mut Diag<'_>,
698        expr: &hir::Expr<'_>,
699        error: Option<TypeError<'tcx>>,
700    ) {
701        match (self.tcx.parent_hir_node(expr.hir_id), error) {
702            (hir::Node::LetStmt(hir::LetStmt { ty: Some(ty), init: Some(init), .. }), _)
703                if init.hir_id == expr.hir_id && !ty.span.source_equal(init.span) =>
704            {
705                // Point at `let` assignment type.
706                err.span_label(ty.span, "expected due to this");
707            }
708            (
709                hir::Node::Expr(hir::Expr { kind: hir::ExprKind::Assign(lhs, rhs, _), .. }),
710                Some(TypeError::Sorts(ExpectedFound { expected, .. })),
711            ) if rhs.hir_id == expr.hir_id && !expected.is_closure() => {
712                // We ignore closures explicitly because we already point at them elsewhere.
713                // Point at the assigned-to binding.
714                let mut primary_span = lhs.span;
715                let mut secondary_span = lhs.span;
716                let mut post_message = "";
717                match lhs.kind {
718                    hir::ExprKind::Path(hir::QPath::Resolved(
719                        None,
720                        hir::Path {
721                            res:
722                                hir::def::Res::Def(
723                                    hir::def::DefKind::Static { .. }
724                                    | hir::def::DefKind::Const { .. },
725                                    def_id,
726                                ),
727                            ..
728                        },
729                    )) => {
730                        if let Some(hir::Node::Item(hir::Item {
731                            kind:
732                                hir::ItemKind::Static(_, ident, ty, _)
733                                | hir::ItemKind::Const(ident, _, ty, _),
734                            ..
735                        })) = self.tcx.hir_get_if_local(*def_id)
736                        {
737                            primary_span = ty.span;
738                            secondary_span = ident.span;
739                            post_message = " type";
740                        }
741                    }
742                    hir::ExprKind::Path(hir::QPath::Resolved(
743                        None,
744                        hir::Path { res: hir::def::Res::Local(hir_id), .. },
745                    )) => {
746                        if let hir::Node::Pat(pat) = self.tcx.hir_node(*hir_id) {
747                            primary_span = pat.span;
748                            secondary_span = pat.span;
749                            match self.tcx.parent_hir_node(pat.hir_id) {
750                                hir::Node::LetStmt(hir::LetStmt { ty: Some(ty), .. }) => {
751                                    primary_span = ty.span;
752                                    post_message = " type";
753                                }
754                                hir::Node::LetStmt(hir::LetStmt { init: Some(init), .. }) => {
755                                    primary_span = init.span;
756                                    post_message = " value";
757                                }
758                                hir::Node::Param(hir::Param { ty_span, .. }) => {
759                                    primary_span = *ty_span;
760                                    post_message = " parameter type";
761                                }
762                                _ => {}
763                            }
764                        }
765                    }
766                    _ => {}
767                }
768
769                if primary_span != secondary_span
770                    && self
771                        .tcx
772                        .sess
773                        .source_map()
774                        .is_multiline(secondary_span.shrink_to_hi().until(primary_span))
775                {
776                    // We are pointing at the binding's type or initializer value, but it's pattern
777                    // is in a different line, so we point at both.
778                    err.span_label(secondary_span, "expected due to the type of this binding");
779                    err.span_label(primary_span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("expected due to this{0}",
                post_message))
    })format!("expected due to this{post_message}"));
780                } else if post_message.is_empty() {
781                    // We are pointing at either the assignment lhs or the binding def pattern.
782                    err.span_label(primary_span, "expected due to the type of this binding");
783                } else {
784                    // We are pointing at the binding's type or initializer value.
785                    err.span_label(primary_span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("expected due to this{0}",
                post_message))
    })format!("expected due to this{post_message}"));
786                }
787
788                if !lhs.is_syntactic_place_expr() {
789                    // We already emitted E0070 "invalid left-hand side of assignment", so we
790                    // silence this.
791                    err.downgrade_to_delayed_bug();
792                }
793            }
794            (
795                hir::Node::Expr(hir::Expr { kind: hir::ExprKind::Binary(_, lhs, rhs), .. }),
796                Some(TypeError::Sorts(ExpectedFound { expected, .. })),
797            ) if rhs.hir_id == expr.hir_id
798                && self.typeck_results.borrow().expr_ty_adjusted_opt(lhs) == Some(expected)
799                // let expressions being marked as `bool` is confusing (see issue #147665)
800                && !#[allow(non_exhaustive_omitted_patterns)] match lhs.kind {
    hir::ExprKind::Let(..) => true,
    _ => false,
}matches!(lhs.kind, hir::ExprKind::Let(..)) =>
801            {
802                err.span_label(lhs.span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("expected because this is `{0}`",
                expected))
    })format!("expected because this is `{expected}`"));
803            }
804            _ => {}
805        }
806    }
807
808    /// Detect the following case
809    ///
810    /// ```text
811    /// fn change_object(mut b: &Ty) {
812    ///     let a = Ty::new();
813    ///     b = a;
814    /// }
815    /// ```
816    ///
817    /// where the user likely meant to modify the value behind there reference, use `b` as an out
818    /// parameter, instead of mutating the local binding. When encountering this we suggest:
819    ///
820    /// ```text
821    /// fn change_object(b: &'_ mut Ty) {
822    ///     let a = Ty::new();
823    ///     *b = a;
824    /// }
825    /// ```
826    fn annotate_mut_binding_to_immutable_binding(
827        &self,
828        err: &mut Diag<'_>,
829        expr: &hir::Expr<'_>,
830        expr_ty: Ty<'tcx>,
831        expected: Ty<'tcx>,
832        error: Option<TypeError<'tcx>>,
833    ) -> bool {
834        if let Some(TypeError::Sorts(ExpectedFound { .. })) = error
835            && let ty::Ref(_, inner, hir::Mutability::Not) = expected.kind()
836
837            // The difference between the expected and found values is one level of borrowing.
838            && self.can_eq(self.param_env, *inner, expr_ty)
839
840            // We have an `ident = expr;` assignment.
841            && let hir::Node::Expr(hir::Expr { kind: hir::ExprKind::Assign(lhs, rhs, _), .. }) =
842                self.tcx.parent_hir_node(expr.hir_id)
843            && rhs.hir_id == expr.hir_id
844
845            // We are assigning to some binding.
846            && let hir::ExprKind::Path(hir::QPath::Resolved(
847                None,
848                hir::Path { res: hir::def::Res::Local(hir_id), .. },
849            )) = lhs.kind
850            && let hir::Node::Pat(pat) = self.tcx.hir_node(*hir_id)
851
852            // The pattern we have is an fn argument.
853            && let hir::Node::Param(hir::Param { ty_span, .. }) =
854                self.tcx.parent_hir_node(pat.hir_id)
855            && let item = self.tcx.hir_get_parent_item(pat.hir_id)
856            && let item = self.tcx.hir_owner_node(item)
857            && let Some(fn_decl) = item.fn_decl()
858
859            // We have a mutable binding in the argument.
860            && let hir::PatKind::Binding(hir::BindingMode::MUT, _hir_id, ident, _) = pat.kind
861
862            // Look for the type corresponding to the argument pattern we have in the argument list.
863            && let Some(ty_ref) = fn_decl
864                .inputs
865                .iter()
866                .filter_map(|ty| match ty.kind {
867                    hir::TyKind::Ref(lt, mut_ty) if ty.span == *ty_span => Some((lt, mut_ty)),
868                    _ => None,
869                })
870                .next()
871        {
872            let mut sugg = if ty_ref.1.mutbl.is_mut() {
873                // Leave `&'name mut Ty` and `&mut Ty` as they are (#136028).
874                ::alloc::vec::Vec::new()vec![]
875            } else {
876                // `&'name Ty` -> `&'name mut Ty` or `&Ty` -> `&mut Ty`
877                ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(ty_ref.1.ty.span.shrink_to_lo(),
                    ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!("{0}mut ",
                                    if ty_ref.0.ident.span.is_empty() { "" } else { " " }))
                        }))]))vec![(
878                    ty_ref.1.ty.span.shrink_to_lo(),
879                    format!("{}mut ", if ty_ref.0.ident.span.is_empty() { "" } else { " " },),
880                )]
881            };
882            sugg.extend([
883                (pat.span.until(ident.span), String::new()),
884                (lhs.span.shrink_to_lo(), "*".to_string()),
885            ]);
886            // We suggest changing the argument from `mut ident: &Ty` to `ident: &'_ mut Ty` and the
887            // assignment from `ident = val;` to `*ident = val;`.
888            err.multipart_suggestion(
889                "you might have meant to mutate the pointed at value being passed in, instead of \
890                changing the reference in the local binding",
891                sugg,
892                Applicability::MaybeIncorrect,
893            );
894            return true;
895        }
896        false
897    }
898
899    fn annotate_alternative_method_deref(
900        &self,
901        err: &mut Diag<'_>,
902        expr: &hir::Expr<'_>,
903        error: Option<TypeError<'tcx>>,
904    ) {
905        let Some(TypeError::Sorts(ExpectedFound { expected, .. })) = error else {
906            return;
907        };
908        let hir::Node::Expr(hir::Expr { kind: hir::ExprKind::Assign(lhs, rhs, _), .. }) =
909            self.tcx.parent_hir_node(expr.hir_id)
910        else {
911            return;
912        };
913        if rhs.hir_id != expr.hir_id || expected.is_closure() {
914            return;
915        }
916        let hir::ExprKind::Unary(hir::UnOp::Deref, deref) = lhs.kind else {
917            return;
918        };
919        let hir::ExprKind::MethodCall(path, base, args, _) = deref.kind else {
920            return;
921        };
922        let Some(self_ty) = self.typeck_results.borrow().expr_ty_adjusted_opt(base) else {
923            return;
924        };
925
926        let Ok(pick) = self.lookup_probe_for_diagnostic(
927            path.ident,
928            self_ty,
929            deref,
930            probe::ProbeScope::TraitsInScope,
931            None,
932        ) else {
933            return;
934        };
935
936        let Ok(in_scope_methods) = self.probe_for_name_many(
937            probe::Mode::MethodCall,
938            path.ident,
939            Some(expected),
940            probe::IsSuggestion(true),
941            self_ty,
942            deref.hir_id,
943            probe::ProbeScope::TraitsInScope,
944        ) else {
945            return;
946        };
947
948        let other_methods_in_scope: Vec<_> =
949            in_scope_methods.iter().filter(|c| c.item.def_id != pick.item.def_id).collect();
950
951        let Ok(all_methods) = self.probe_for_name_many(
952            probe::Mode::MethodCall,
953            path.ident,
954            Some(expected),
955            probe::IsSuggestion(true),
956            self_ty,
957            deref.hir_id,
958            probe::ProbeScope::AllTraits,
959        ) else {
960            return;
961        };
962
963        let suggestions: Vec<_> = all_methods
964            .into_iter()
965            .filter(|c| c.item.def_id != pick.item.def_id)
966            .map(|c| {
967                let m = c.item;
968                let generic_args = ty::GenericArgs::for_item(self.tcx, m.def_id, |param, _| {
969                    self.var_for_def(deref.span, param)
970                });
971                let mutability =
972                    match self.tcx.fn_sig(m.def_id).skip_binder().input(0).skip_binder().kind() {
973                        ty::Ref(_, _, hir::Mutability::Mut) => "&mut ",
974                        ty::Ref(_, _, _) => "&",
975                        _ => "",
976                    };
977                ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [(deref.span.until(base.span),
                    ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!("{0}({1}",
                                    {
                                        let _guard = NoTrimmedGuard::new();
                                        self.tcx.def_path_str_with_args(m.def_id, generic_args)
                                    }, mutability))
                        })),
                match &args {
                    [] =>
                        (base.span.shrink_to_hi().with_hi(deref.span.hi()),
                            ")".to_string()),
                    [first, ..] =>
                        (base.span.between(first.span), ", ".to_string()),
                }]))vec![
978                    (
979                        deref.span.until(base.span),
980                        format!(
981                            "{}({}",
982                            with_no_trimmed_paths!(
983                                self.tcx.def_path_str_with_args(m.def_id, generic_args,)
984                            ),
985                            mutability,
986                        ),
987                    ),
988                    match &args {
989                        [] => (base.span.shrink_to_hi().with_hi(deref.span.hi()), ")".to_string()),
990                        [first, ..] => (base.span.between(first.span), ", ".to_string()),
991                    },
992                ]
993            })
994            .collect();
995        if suggestions.is_empty() {
996            return;
997        }
998        let mut path_span: MultiSpan = path.ident.span.into();
999        path_span.push_span_label(
1000            path.ident.span,
1001            {
    let _guard = NoTrimmedGuard::new();
    ::alloc::__export::must_use({
            ::alloc::fmt::format(format_args!("refers to `{0}`",
                    self.tcx.def_path_str(pick.item.def_id)))
        })
}with_no_trimmed_paths!(format!(
1002                "refers to `{}`",
1003                self.tcx.def_path_str(pick.item.def_id),
1004            )),
1005        );
1006        let container_id = pick.item.container_id(self.tcx);
1007        let container = { let _guard = NoTrimmedGuard::new(); self.tcx.def_path_str(container_id) }with_no_trimmed_paths!(self.tcx.def_path_str(container_id));
1008        for &def_id in pick.import_ids {
1009            let hir_id = self.tcx.local_def_id_to_hir_id(def_id);
1010            path_span
1011                .push_span_label(self.tcx.hir_span(hir_id), ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` imported here", container))
    })format!("`{container}` imported here"));
1012        }
1013        let tail = {
    let _guard = NoTrimmedGuard::new();
    match &other_methods_in_scope[..] {
        [] => return,
        [candidate] =>
            ::alloc::__export::must_use({
                    ::alloc::fmt::format(format_args!("the method of the same name on {0} `{1}`",
                            match candidate.kind {
                                probe::CandidateKind::InherentImplCandidate { .. } =>
                                    "the inherent impl for",
                                _ => "trait",
                            },
                            self.tcx.def_path_str(candidate.item.container_id(self.tcx))))
                }),
        _ if other_methods_in_scope.len() < 5 => {
            ::alloc::__export::must_use({
                    ::alloc::fmt::format(format_args!("the methods of the same name on {0}",
                            listify(&other_methods_in_scope[..other_methods_in_scope.len()
                                                    - 1],
                                    |c|
                                        ::alloc::__export::must_use({
                                                ::alloc::fmt::format(format_args!("`{0}`",
                                                        self.tcx.def_path_str(c.item.container_id(self.tcx))))
                                            })).unwrap_or_default()))
                })
        }
        _ =>
            ::alloc::__export::must_use({
                    ::alloc::fmt::format(format_args!("the methods of the same name on {0} other traits",
                            other_methods_in_scope.len()))
                }),
    }
}with_no_trimmed_paths!(match &other_methods_in_scope[..] {
1014            [] => return,
1015            [candidate] => format!(
1016                "the method of the same name on {} `{}`",
1017                match candidate.kind {
1018                    probe::CandidateKind::InherentImplCandidate { .. } => "the inherent impl for",
1019                    _ => "trait",
1020                },
1021                self.tcx.def_path_str(candidate.item.container_id(self.tcx))
1022            ),
1023            _ if other_methods_in_scope.len() < 5 => {
1024                format!(
1025                    "the methods of the same name on {}",
1026                    listify(
1027                        &other_methods_in_scope[..other_methods_in_scope.len() - 1],
1028                        |c| format!("`{}`", self.tcx.def_path_str(c.item.container_id(self.tcx)))
1029                    )
1030                    .unwrap_or_default(),
1031                )
1032            }
1033            _ => format!(
1034                "the methods of the same name on {} other traits",
1035                other_methods_in_scope.len()
1036            ),
1037        });
1038        err.span_note(
1039            path_span,
1040            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("the `{0}` call is resolved to the method in `{1}`, shadowing {2}",
                path.ident, container, tail))
    })format!(
1041                "the `{}` call is resolved to the method in `{container}`, shadowing {tail}",
1042                path.ident,
1043            ),
1044        );
1045        if suggestions.len() > other_methods_in_scope.len() {
1046            err.note(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("additionally, there are {0} other available methods that aren\'t in scope",
                suggestions.len() - other_methods_in_scope.len()))
    })format!(
1047                "additionally, there are {} other available methods that aren't in scope",
1048                suggestions.len() - other_methods_in_scope.len()
1049            ));
1050        }
1051        err.multipart_suggestions(
1052            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("you might have meant to call {0}; you can use the fully-qualified path to call {1} explicitly",
                if suggestions.len() == 1 {
                    "the other method"
                } else { "one of the other methods" },
                if suggestions.len() == 1 { "it" } else { "one of them" }))
    })format!(
1053                "you might have meant to call {}; you can use the fully-qualified path to call {} \
1054                 explicitly",
1055                if suggestions.len() == 1 {
1056                    "the other method"
1057                } else {
1058                    "one of the other methods"
1059                },
1060                if suggestions.len() == 1 { "it" } else { "one of them" },
1061            ),
1062            suggestions,
1063            Applicability::MaybeIncorrect,
1064        );
1065    }
1066
1067    pub(crate) fn get_conversion_methods_for_diagnostic(
1068        &self,
1069        span: Span,
1070        expected: Ty<'tcx>,
1071        checked_ty: Ty<'tcx>,
1072        hir_id: hir::HirId,
1073    ) -> Vec<AssocItem> {
1074        let methods = self.probe_for_return_type_for_diagnostic(
1075            span,
1076            probe::Mode::MethodCall,
1077            expected,
1078            checked_ty,
1079            hir_id,
1080            |m| {
1081                self.has_only_self_parameter(m)
1082                // This special internal attribute is used to permit
1083                // "identity-like" conversion methods to be suggested here.
1084                //
1085                // FIXME (#46459 and #46460): ideally
1086                // `std::convert::Into::into` and `std::borrow:ToOwned` would
1087                // also be `#[rustc_conversion_suggestion]`, if not for
1088                // method-probing false-positives and -negatives (respectively).
1089                //
1090                // FIXME? Other potential candidate methods: `as_ref` and
1091                // `as_mut`?
1092                && {
        {
            'done:
                {
                for i in
                    ::rustc_hir::attrs::HasAttrs::get_attrs(m.def_id, &self.tcx)
                    {
                    #[allow(unused_imports)]
                    use rustc_hir::attrs::AttributeKind::*;
                    let i: &rustc_hir::Attribute = i;
                    match i {
                        rustc_hir::Attribute::Parsed(RustcConversionSuggestion) => {
                            break 'done Some(());
                        }
                        rustc_hir::Attribute::Unparsed(..) =>
                            {}
                            #[deny(unreachable_patterns)]
                            _ => {}
                    }
                }
                None
            }
        }
    }.is_some()find_attr!(self.tcx, m.def_id, RustcConversionSuggestion)
1093            },
1094        );
1095
1096        methods
1097    }
1098
1099    /// This function checks whether the method is not static and does not accept other parameters than `self`.
1100    fn has_only_self_parameter(&self, method: &AssocItem) -> bool {
1101        method.is_method()
1102            && self.tcx.fn_sig(method.def_id).skip_binder().inputs().skip_binder().len() == 1
1103    }
1104
1105    /// If the given `HirId` corresponds to a block with a trailing expression, return that expression
1106    pub(crate) fn maybe_get_block_expr(
1107        &self,
1108        expr: &hir::Expr<'tcx>,
1109    ) -> Option<&'tcx hir::Expr<'tcx>> {
1110        match expr {
1111            hir::Expr { kind: hir::ExprKind::Block(block, ..), .. } => block.expr,
1112            _ => None,
1113        }
1114    }
1115
1116    /// Returns whether the given expression is a destruct assignment desugaring.
1117    /// For example, `(a, b) = (1, &2);`
1118    /// Here we try to find the pattern binding of the expression,
1119    /// `default_binding_modes` is false only for destruct assignment desugaring.
1120    pub(crate) fn is_destruct_assignment_desugaring(&self, expr: &hir::Expr<'_>) -> bool {
1121        if let hir::ExprKind::Path(hir::QPath::Resolved(
1122            _,
1123            hir::Path { res: hir::def::Res::Local(bind_hir_id), .. },
1124        )) = expr.kind
1125            && let bind = self.tcx.hir_node(*bind_hir_id)
1126            && let parent = self.tcx.parent_hir_node(*bind_hir_id)
1127            && let hir::Node::Pat(hir::Pat {
1128                kind: hir::PatKind::Binding(_, _hir_id, _, _), ..
1129            }) = bind
1130            && let hir::Node::Pat(hir::Pat { default_binding_modes: false, .. }) = parent
1131        {
1132            true
1133        } else {
1134            false
1135        }
1136    }
1137
1138    fn explain_self_literal(
1139        &self,
1140        err: &mut Diag<'_>,
1141        expr: &hir::Expr<'tcx>,
1142        expected: Ty<'tcx>,
1143        found: Ty<'tcx>,
1144    ) {
1145        match expr.peel_drop_temps().kind {
1146            hir::ExprKind::Struct(
1147                hir::QPath::Resolved(
1148                    None,
1149                    hir::Path { res: hir::def::Res::SelfTyAlias { alias_to, .. }, span, .. },
1150                ),
1151                ..,
1152            )
1153            | hir::ExprKind::Call(
1154                hir::Expr {
1155                    kind:
1156                        hir::ExprKind::Path(hir::QPath::Resolved(
1157                            None,
1158                            hir::Path {
1159                                res: hir::def::Res::SelfTyAlias { alias_to, .. },
1160                                span,
1161                                ..
1162                            },
1163                        )),
1164                    ..
1165                },
1166                ..,
1167            ) => {
1168                if let Some(hir::Node::Item(hir::Item {
1169                    kind: hir::ItemKind::Impl(hir::Impl { self_ty, .. }),
1170                    ..
1171                })) = self.tcx.hir_get_if_local(*alias_to)
1172                {
1173                    err.span_label(self_ty.span, "this is the type of the `Self` literal");
1174                }
1175                if let ty::Adt(e_def, e_args) = expected.kind()
1176                    && let ty::Adt(f_def, _f_args) = found.kind()
1177                    && e_def == f_def
1178                {
1179                    err.span_suggestion_verbose(
1180                        *span,
1181                        "use the type name directly",
1182                        self.tcx.value_path_str_with_args(e_def.did(), e_args),
1183                        Applicability::MaybeIncorrect,
1184                    );
1185                }
1186            }
1187            _ => {}
1188        }
1189    }
1190
1191    fn note_wrong_return_ty_due_to_generic_arg(
1192        &self,
1193        err: &mut Diag<'_>,
1194        expr: &hir::Expr<'_>,
1195        checked_ty: Ty<'tcx>,
1196    ) {
1197        let hir::Node::Expr(parent_expr) = self.tcx.parent_hir_node(expr.hir_id) else {
1198            return;
1199        };
1200        if parent_expr.span.desugaring_kind().is_some() {
1201            return;
1202        }
1203        enum CallableKind {
1204            Function,
1205            Method,
1206            Constructor,
1207        }
1208        let mut maybe_emit_help = |def_id: hir::def_id::DefId,
1209                                   callable: Ident,
1210                                   args: &[hir::Expr<'_>],
1211                                   kind: CallableKind| {
1212            let arg_idx = args.iter().position(|a| a.hir_id == expr.hir_id).unwrap();
1213            let fn_ty = self.tcx.type_of(def_id).skip_binder();
1214            if !fn_ty.is_fn() {
1215                return;
1216            }
1217            let fn_sig = fn_ty.fn_sig(self.tcx).skip_binder();
1218            let Some(&arg) = fn_sig
1219                .inputs()
1220                .get(arg_idx + if #[allow(non_exhaustive_omitted_patterns)] match kind {
    CallableKind::Method => true,
    _ => false,
}matches!(kind, CallableKind::Method) { 1 } else { 0 })
1221            else {
1222                return;
1223            };
1224            if #[allow(non_exhaustive_omitted_patterns)] match arg.kind() {
    ty::Param(_) => true,
    _ => false,
}matches!(arg.kind(), ty::Param(_))
1225                && fn_sig.output().contains(arg)
1226                && self.node_ty(args[arg_idx].hir_id) == checked_ty
1227            {
1228                let mut multi_span: MultiSpan = parent_expr.span.into();
1229                multi_span.push_span_label(
1230                    args[arg_idx].span,
1231                    ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("this argument influences the {0} of `{1}`",
                if #[allow(non_exhaustive_omitted_patterns)] match kind {
                        CallableKind::Constructor => true,
                        _ => false,
                    } {
                    "type"
                } else { "return type" }, callable))
    })format!(
1232                        "this argument influences the {} of `{}`",
1233                        if matches!(kind, CallableKind::Constructor) {
1234                            "type"
1235                        } else {
1236                            "return type"
1237                        },
1238                        callable
1239                    ),
1240                );
1241                err.span_help(
1242                    multi_span,
1243                    ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("the {0} `{1}` due to the type of the argument passed",
                match kind {
                    CallableKind::Function => "return type of this call is",
                    CallableKind::Method => "return type of this call is",
                    CallableKind::Constructor => "type constructed contains",
                }, checked_ty))
    })format!(
1244                        "the {} `{}` due to the type of the argument passed",
1245                        match kind {
1246                            CallableKind::Function => "return type of this call is",
1247                            CallableKind::Method => "return type of this call is",
1248                            CallableKind::Constructor => "type constructed contains",
1249                        },
1250                        checked_ty
1251                    ),
1252                );
1253            }
1254        };
1255        match parent_expr.kind {
1256            hir::ExprKind::Call(fun, args) => {
1257                let hir::ExprKind::Path(hir::QPath::Resolved(_, path)) = fun.kind else {
1258                    return;
1259                };
1260                let hir::def::Res::Def(kind, def_id) = path.res else {
1261                    return;
1262                };
1263                let callable_kind = if #[allow(non_exhaustive_omitted_patterns)] match kind {
    hir::def::DefKind::Ctor(_, _) => true,
    _ => false,
}matches!(kind, hir::def::DefKind::Ctor(_, _)) {
1264                    CallableKind::Constructor
1265                } else {
1266                    CallableKind::Function
1267                };
1268                maybe_emit_help(def_id, path.segments.last().unwrap().ident, args, callable_kind);
1269            }
1270            hir::ExprKind::MethodCall(method, _receiver, args, _span) => {
1271                let Some(def_id) =
1272                    self.typeck_results.borrow().type_dependent_def_id(parent_expr.hir_id)
1273                else {
1274                    return;
1275                };
1276                maybe_emit_help(def_id, method.ident, args, CallableKind::Method)
1277            }
1278            _ => return,
1279        }
1280    }
1281}
1282
1283pub(crate) enum TypeMismatchSource<'tcx> {
1284    /// Expected the binding to have the given type, but it was found to have
1285    /// a different type. Find out when that type first became incompatible.
1286    Ty(Ty<'tcx>),
1287    /// When we fail during method argument checking, try to find out if a previous
1288    /// expression has constrained the method's receiver in a way that makes the
1289    /// argument's type incompatible.
1290    Arg { call_expr: &'tcx hir::Expr<'tcx>, incompatible_arg: usize },
1291}