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 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_compatible_variants(err, expr, expected, expr_ty)
44 || self.suggest_non_zero_new_unwrap(err, expr, expected, expr_ty)
45 || self.suggest_calling_boxed_future_when_appropriate(err, expr, expected, expr_ty)
46 || self.suggest_no_capture_closure(err, expected, expr_ty)
47 || self.suggest_boxing_when_appropriate(
48 err,
49 expr.peel_blocks().span,
50 expr.hir_id,
51 expected,
52 expr_ty,
53 )
54 || self.suggest_block_to_brackets_peeling_refs(err, expr, expr_ty, expected)
55 || self.suggest_copied_cloned_or_as_ref(err, expr, expr_ty, expected)
56 || self.suggest_clone_for_ref(err, expr, expr_ty, expected)
57 || self.suggest_into(err, expr, expr_ty, expected)
58 || self.suggest_floating_point_literal(err, expr, expected)
59 || self.suggest_null_ptr_for_literal_zero_given_to_ptr_arg(err, expr, expected)
60 || self.suggest_coercing_result_via_try_operator(err, expr, expected, expr_ty)
61 || self.suggest_returning_value_after_loop(err, expr, expected);
62
63 if !suggested {
64 self.note_source_of_type_mismatch_constraint(
65 err,
66 expr,
67 TypeMismatchSource::Ty(expected),
68 );
69 }
70 }
71
72 pub(crate) fn emit_coerce_suggestions(
73 &self,
74 err: &mut Diag<'_>,
75 expr: &hir::Expr<'tcx>,
76 expr_ty: Ty<'tcx>,
77 expected: Ty<'tcx>,
78 expected_ty_expr: Option<&'tcx hir::Expr<'tcx>>,
79 error: Option<TypeError<'tcx>>,
80 ) {
81 if expr_ty == expected {
82 return;
83 }
84
85 self.annotate_expected_due_to_let_ty(err, expr, error);
86 self.annotate_loop_expected_due_to_inference(err, expr, error);
87 if self.annotate_mut_binding_to_immutable_binding(err, expr, expr_ty, expected, error) {
88 return;
89 }
90
91 if #[allow(non_exhaustive_omitted_patterns)] match error {
Some(TypeError::RegionsInsufficientlyPolymorphic(..)) => true,
_ => false,
}matches!(error, Some(TypeError::RegionsInsufficientlyPolymorphic(..))) {
95 return;
96 }
97
98 if self.is_destruct_assignment_desugaring(expr) {
99 return;
100 }
101 self.emit_type_mismatch_suggestions(err, expr, expr_ty, expected, expected_ty_expr, error);
102 self.note_type_is_not_clone(err, expected, expr_ty, expr);
103 self.note_internal_mutation_in_method(err, expr, Some(expected), expr_ty);
104 self.suggest_method_call_on_range_literal(err, expr, expr_ty, expected);
105 self.suggest_return_binding_for_missing_tail_expr(err, expr, expr_ty, expected);
106 self.note_wrong_return_ty_due_to_generic_arg(err, expr, expr_ty);
107 }
108
109 fn adjust_expr_for_assert_eq_macro(
112 &self,
113 found_expr: &mut &'tcx hir::Expr<'tcx>,
114 expected_expr: &mut Option<&'tcx hir::Expr<'tcx>>,
115 ) {
116 let Some(expected_expr) = expected_expr else {
117 return;
118 };
119
120 if !found_expr.span.eq_ctxt(expected_expr.span) {
121 return;
122 }
123
124 if !found_expr
125 .span
126 .ctxt()
127 .outer_expn_data()
128 .macro_def_id
129 .is_some_and(|def_id| self.tcx.is_diagnostic_item(sym::assert_eq_macro, def_id))
130 {
131 return;
132 }
133
134 let hir::ExprKind::Unary(
135 hir::UnOp::Deref,
136 hir::Expr { kind: hir::ExprKind::Path(found_path), .. },
137 ) = found_expr.kind
138 else {
139 return;
140 };
141 let hir::ExprKind::Unary(
142 hir::UnOp::Deref,
143 hir::Expr { kind: hir::ExprKind::Path(expected_path), .. },
144 ) = expected_expr.kind
145 else {
146 return;
147 };
148
149 for (path, name, idx, var) in [
150 (expected_path, "left_val", 0, expected_expr),
151 (found_path, "right_val", 1, found_expr),
152 ] {
153 if let hir::QPath::Resolved(_, path) = path
154 && let [segment] = path.segments
155 && segment.ident.name.as_str() == name
156 && let Res::Local(hir_id) = path.res
157 && let Some((_, hir::Node::Expr(match_expr))) =
158 self.tcx.hir_parent_iter(hir_id).nth(2)
159 && let hir::ExprKind::Match(scrutinee, _, _) = match_expr.kind
160 && let hir::ExprKind::Tup(exprs) = scrutinee.kind
161 && let hir::ExprKind::AddrOf(_, _, macro_arg) = exprs[idx].kind
162 {
163 *var = macro_arg;
164 }
165 }
166 }
167
168 pub(crate) fn demand_suptype(&self, sp: Span, expected: Ty<'tcx>, actual: Ty<'tcx>) {
171 if let Err(e) = self.demand_suptype_diag(sp, expected, actual) {
172 e.emit();
173 }
174 }
175
176 pub(crate) fn demand_suptype_diag(
177 &'a self,
178 sp: Span,
179 expected: Ty<'tcx>,
180 actual: Ty<'tcx>,
181 ) -> Result<(), Diag<'a>> {
182 self.demand_suptype_with_origin(&self.misc(sp), expected, actual)
183 }
184
185 #[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(185u32),
::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")]
186 pub(crate) fn demand_suptype_with_origin(
187 &'a self,
188 cause: &ObligationCause<'tcx>,
189 expected: Ty<'tcx>,
190 actual: Ty<'tcx>,
191 ) -> Result<(), Diag<'a>> {
192 self.at(cause, self.param_env)
193 .sup(DefineOpaqueTypes::Yes, expected, actual)
194 .map(|infer_ok| self.register_infer_ok_obligations(infer_ok))
195 .map_err(|e| {
196 self.err_ctxt().report_mismatched_types(cause, self.param_env, expected, actual, e)
197 })
198 }
199
200 pub(crate) fn demand_eqtype(&self, sp: Span, expected: Ty<'tcx>, actual: Ty<'tcx>) {
201 if let Err(err) = self.demand_eqtype_diag(sp, expected, actual) {
202 err.emit();
203 }
204 }
205
206 pub(crate) fn demand_eqtype_diag(
207 &'a self,
208 sp: Span,
209 expected: Ty<'tcx>,
210 actual: Ty<'tcx>,
211 ) -> Result<(), Diag<'a>> {
212 self.demand_eqtype_with_origin(&self.misc(sp), expected, actual)
213 }
214
215 pub(crate) fn demand_eqtype_with_origin(
216 &'a self,
217 cause: &ObligationCause<'tcx>,
218 expected: Ty<'tcx>,
219 actual: Ty<'tcx>,
220 ) -> Result<(), Diag<'a>> {
221 self.at(cause, self.param_env)
222 .eq(DefineOpaqueTypes::Yes, expected, actual)
223 .map(|infer_ok| self.register_infer_ok_obligations(infer_ok))
224 .map_err(|e| {
225 self.err_ctxt().report_mismatched_types(cause, self.param_env, expected, actual, e)
226 })
227 }
228
229 pub(crate) fn demand_coerce(
230 &self,
231 expr: &'tcx hir::Expr<'tcx>,
232 checked_ty: Ty<'tcx>,
233 expected: Ty<'tcx>,
234 expected_ty_expr: Option<&'tcx hir::Expr<'tcx>>,
235 allow_two_phase: AllowTwoPhase,
236 ) -> Ty<'tcx> {
237 match self.demand_coerce_diag(expr, checked_ty, expected, expected_ty_expr, allow_two_phase)
238 {
239 Ok(ty) => ty,
240 Err(err) => {
241 err.emit();
242 expected
246 }
247 }
248 }
249
250 #[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(254u32),
::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))]
255 pub(crate) fn demand_coerce_diag(
256 &'a self,
257 mut expr: &'tcx hir::Expr<'tcx>,
258 checked_ty: Ty<'tcx>,
259 expected: Ty<'tcx>,
260 mut expected_ty_expr: Option<&'tcx hir::Expr<'tcx>>,
261 allow_two_phase: AllowTwoPhase,
262 ) -> Result<Ty<'tcx>, Diag<'a>> {
263 let expected = self.resolve_vars_with_obligations(expected);
264
265 let e = match self.coerce(expr, checked_ty, expected, allow_two_phase, None) {
266 Ok(ty) => return Ok(ty),
267 Err(e) => e,
268 };
269
270 self.adjust_expr_for_assert_eq_macro(&mut expr, &mut expected_ty_expr);
271
272 self.set_tainted_by_errors(self.dcx().span_delayed_bug(
273 expr.span,
274 "`TypeError` when attempting coercion but no error emitted",
275 ));
276 let expr = expr.peel_drop_temps();
277 let cause = self.misc(expr.span);
278 let expr_ty = self.resolve_vars_if_possible(checked_ty);
279 let mut err =
280 self.err_ctxt().report_mismatched_types(&cause, self.param_env, expected, expr_ty, e);
281
282 self.emit_coerce_suggestions(&mut err, expr, expr_ty, expected, expected_ty_expr, Some(e));
283
284 Err(err)
285 }
286
287 pub(crate) fn note_source_of_type_mismatch_constraint(
290 &self,
291 err: &mut Diag<'_>,
292 expr: &hir::Expr<'_>,
293 source: TypeMismatchSource<'tcx>,
294 ) -> bool {
295 let hir::ExprKind::Path(hir::QPath::Resolved(None, p)) = expr.kind else {
296 return false;
297 };
298 let [hir::PathSegment { ident, args: None, .. }] = p.segments else {
299 return false;
300 };
301 let hir::def::Res::Local(local_hir_id) = p.res else {
302 return false;
303 };
304 let hir::Node::Pat(pat) = self.tcx.hir_node(local_hir_id) else {
305 return false;
306 };
307 let (init_ty_hir_id, init) = match self.tcx.parent_hir_node(pat.hir_id) {
308 hir::Node::LetStmt(hir::LetStmt { ty: Some(ty), init, .. }) => (ty.hir_id, *init),
309 hir::Node::LetStmt(hir::LetStmt { init: Some(init), .. }) => (init.hir_id, Some(*init)),
310 _ => return false,
311 };
312 let Some(init_ty) = self.node_ty_opt(init_ty_hir_id) else {
313 return false;
314 };
315
316 struct FindExprs<'tcx> {
318 hir_id: hir::HirId,
319 uses: Vec<&'tcx hir::Expr<'tcx>>,
320 }
321 impl<'tcx> Visitor<'tcx> for FindExprs<'tcx> {
322 fn visit_expr(&mut self, ex: &'tcx hir::Expr<'tcx>) {
323 if let hir::ExprKind::Path(hir::QPath::Resolved(None, path)) = ex.kind
324 && let hir::def::Res::Local(hir_id) = path.res
325 && hir_id == self.hir_id
326 {
327 self.uses.push(ex);
328 }
329 hir::intravisit::walk_expr(self, ex);
330 }
331 }
332
333 let mut expr_finder = FindExprs { hir_id: local_hir_id, uses: init.into_iter().collect() };
334 let body = self.tcx.hir_body_owned_by(self.body_id);
335 expr_finder.visit_expr(body.value);
336
337 let mut fudger = BottomUpFolder {
339 tcx: self.tcx,
340 ty_op: |ty| {
341 if let ty::Infer(infer) = ty.kind() {
342 match infer {
343 ty::TyVar(_) => self.next_ty_var(DUMMY_SP),
344 ty::IntVar(_) => self.next_int_var(),
345 ty::FloatVar(_) => self.next_float_var(DUMMY_SP, None),
346 ty::FreshTy(_) | ty::FreshIntTy(_) | ty::FreshFloatTy(_) => {
347 ::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")
348 }
349 }
350 } else {
351 ty
352 }
353 },
354 lt_op: |_| self.tcx.lifetimes.re_erased,
355 ct_op: |ct| {
356 if let ty::ConstKind::Infer(_) = ct.kind() {
357 self.next_const_var(DUMMY_SP)
358 } else {
359 ct
360 }
361 },
362 };
363
364 let expected_ty = match source {
365 TypeMismatchSource::Ty(expected_ty) => expected_ty,
366 TypeMismatchSource::Arg { call_expr, incompatible_arg: idx } => {
373 let hir::ExprKind::MethodCall(segment, _, args, _) = call_expr.kind else {
374 return false;
375 };
376 let Some(arg_ty) = self.node_ty_opt(args[idx].hir_id) else {
377 return false;
378 };
379 let possible_rcvr_ty = expr_finder.uses.iter().rev().find_map(|binding| {
380 let possible_rcvr_ty = self.node_ty_opt(binding.hir_id)?;
381 if possible_rcvr_ty.is_ty_var() {
382 return None;
383 }
384 let possible_rcvr_ty = possible_rcvr_ty.fold_with(&mut fudger);
386 let method = self
387 .lookup_method_for_diagnostic(
388 possible_rcvr_ty,
389 segment,
390 DUMMY_SP,
391 call_expr,
392 binding,
393 )
394 .ok()?;
395 if Some(method.def_id)
397 != self.typeck_results.borrow().type_dependent_def_id(call_expr.hir_id)
398 {
399 return None;
400 }
401 let _ = self
405 .at(&ObligationCause::dummy(), self.param_env)
406 .eq(DefineOpaqueTypes::Yes, method.sig.inputs()[idx + 1], arg_ty)
407 .ok()?;
408 self.select_obligations_where_possible(|errs| {
409 errs.clear();
411 });
412 Some(self.resolve_vars_if_possible(possible_rcvr_ty))
413 });
414 let Some(rcvr_ty) = possible_rcvr_ty else { return false };
415 rcvr_ty
416 }
417 };
418
419 if !self.can_eq(self.param_env, expected_ty, init_ty.fold_with(&mut fudger)) {
422 return false;
423 }
424
425 for window in expr_finder.uses.windows(2) {
426 let [binding, next_usage] = *window else {
430 continue;
431 };
432
433 if binding.hir_id == expr.hir_id {
435 break;
436 }
437
438 let Some(next_use_ty) = self.node_ty_opt(next_usage.hir_id) else {
439 continue;
440 };
441
442 if self.can_eq(self.param_env, expected_ty, next_use_ty.fold_with(&mut fudger)) {
445 continue;
446 }
447
448 if let hir::Node::Expr(parent_expr) = self.tcx.parent_hir_node(binding.hir_id)
449 && let hir::ExprKind::MethodCall(segment, rcvr, args, _) = parent_expr.kind
450 && rcvr.hir_id == binding.hir_id
451 {
452 let Some(rcvr_ty) = self.node_ty_opt(rcvr.hir_id) else {
456 continue;
457 };
458 let rcvr_ty = rcvr_ty.fold_with(&mut fudger);
459 let Ok(method) = self.lookup_method_for_diagnostic(
460 rcvr_ty,
461 segment,
462 DUMMY_SP,
463 parent_expr,
464 rcvr,
465 ) else {
466 continue;
467 };
468 if Some(method.def_id)
470 != self.typeck_results.borrow().type_dependent_def_id(parent_expr.hir_id)
471 {
472 continue;
473 }
474
475 let ideal_rcvr_ty = rcvr_ty.fold_with(&mut fudger);
476 let ideal_method = self
477 .lookup_method_for_diagnostic(
478 ideal_rcvr_ty,
479 segment,
480 DUMMY_SP,
481 parent_expr,
482 rcvr,
483 )
484 .ok()
485 .and_then(|method| {
486 let _ = self
487 .at(&ObligationCause::dummy(), self.param_env)
488 .eq(DefineOpaqueTypes::Yes, ideal_rcvr_ty, expected_ty)
489 .ok()?;
490 Some(method)
491 });
492
493 for (idx, (expected_arg_ty, arg_expr)) in
496 std::iter::zip(&method.sig.inputs()[1..], args).enumerate()
497 {
498 let Some(arg_ty) = self.node_ty_opt(arg_expr.hir_id) else {
499 continue;
500 };
501 let arg_ty = arg_ty.fold_with(&mut fudger);
502 let _ =
503 self.coerce(arg_expr, arg_ty, *expected_arg_ty, AllowTwoPhase::No, None);
504 self.select_obligations_where_possible(|errs| {
505 errs.clear();
507 });
508 if self.can_eq(self.param_env, rcvr_ty, expected_ty) {
512 continue;
513 }
514 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}`..."));
515 err.span_label(
516 binding.span,
517 ::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}`"),
518 );
519 if #[allow(non_exhaustive_omitted_patterns)] match source {
TypeMismatchSource::Ty(_) => true,
_ => false,
}matches!(source, TypeMismatchSource::Ty(_))
528 && let Some(ideal_method) = ideal_method
529 && Some(ideal_method.def_id)
530 == self
531 .typeck_results
532 .borrow()
533 .type_dependent_def_id(parent_expr.hir_id)
534 && let ideal_arg_ty =
535 self.resolve_vars_if_possible(ideal_method.sig.inputs()[idx + 1])
536 && !ideal_arg_ty.has_non_region_infer()
537 {
538 self.emit_type_mismatch_suggestions(
539 err,
540 arg_expr,
541 arg_ty,
542 ideal_arg_ty,
543 None,
544 None,
545 );
546 }
547 return true;
548 }
549 }
550 err.span_label(
551 binding.span,
552 ::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}`"),
553 );
554 return true;
555 }
556
557 false
559 }
560
561 pub(crate) fn annotate_loop_expected_due_to_inference(
564 &self,
565 err: &mut Diag<'_>,
566 expr: &hir::Expr<'_>,
567 error: Option<TypeError<'tcx>>,
568 ) {
569 let Some(TypeError::Sorts(ExpectedFound { expected, .. })) = error else {
570 return;
571 };
572 let mut parent_id = self.tcx.parent_hir_id(expr.hir_id);
573 let mut parent;
574 'outer: loop {
575 let (hir::Node::Stmt(&hir::Stmt { kind: hir::StmtKind::Semi(p), .. })
577 | hir::Node::Block(&hir::Block { expr: Some(p), .. })
578 | hir::Node::Expr(p)) = self.tcx.hir_node(parent_id)
579 else {
580 break;
581 };
582 parent = p;
583 parent_id = self.tcx.parent_hir_id(parent_id);
584 let hir::ExprKind::Break(destination, _) = parent.kind else {
585 continue;
586 };
587 let mut parent_id = parent_id;
588 let mut direct = false;
589 loop {
590 let parent = match self.tcx.hir_node(parent_id) {
592 hir::Node::Expr(parent) => {
593 parent_id = self.tcx.parent_hir_id(parent.hir_id);
594 parent
595 }
596 hir::Node::Stmt(hir::Stmt {
597 hir_id,
598 kind: hir::StmtKind::Semi(parent) | hir::StmtKind::Expr(parent),
599 ..
600 }) => {
601 parent_id = self.tcx.parent_hir_id(*hir_id);
602 parent
603 }
604 hir::Node::Stmt(hir::Stmt { hir_id, kind: hir::StmtKind::Let(_), .. }) => {
605 parent_id = self.tcx.parent_hir_id(*hir_id);
606 parent
607 }
608 hir::Node::LetStmt(hir::LetStmt { hir_id, .. }) => {
609 parent_id = self.tcx.parent_hir_id(*hir_id);
610 parent
611 }
612 hir::Node::Block(_) => {
613 parent_id = self.tcx.parent_hir_id(parent_id);
614 parent
615 }
616 _ => break,
617 };
618 if let hir::ExprKind::Loop(..) = parent.kind {
619 direct = !direct;
622 }
623 if let hir::ExprKind::Loop(block, label, _, span) = parent.kind
624 && (destination.label == label || direct)
625 {
626 if let Some((reason_span, message)) =
627 self.maybe_get_coercion_reason(parent_id, parent.span)
628 {
629 err.span_label(reason_span, message);
630 err.span_label(
631 span,
632 ::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}`"),
633 );
634 break 'outer;
635 } else {
636 struct FindBreaks<'tcx> {
639 label: Option<rustc_ast::Label>,
640 uses: Vec<&'tcx hir::Expr<'tcx>>,
641 nest_depth: usize,
642 }
643 impl<'tcx> Visitor<'tcx> for FindBreaks<'tcx> {
644 fn visit_expr(&mut self, ex: &'tcx hir::Expr<'tcx>) {
645 let nest_depth = self.nest_depth;
646 if let hir::ExprKind::Loop(_, label, _, _) = ex.kind {
647 if label == self.label {
648 return;
650 }
651 self.nest_depth += 1;
652 }
653 if let hir::ExprKind::Break(destination, _) = ex.kind
654 && (self.label == destination.label
655 || destination.label.is_none() && self.nest_depth == 0)
657 {
658 self.uses.push(ex);
659 }
660 hir::intravisit::walk_expr(self, ex);
661 self.nest_depth = nest_depth;
662 }
663 }
664 let mut expr_finder = FindBreaks { label, uses: ::alloc::vec::Vec::new()vec![], nest_depth: 0 };
665 expr_finder.visit_block(block);
666 let mut exit = false;
667 for ex in expr_finder.uses {
668 let hir::ExprKind::Break(_, val) = ex.kind else {
669 continue;
670 };
671 let ty = match val {
672 Some(val) => {
673 match self.typeck_results.borrow().expr_ty_adjusted_opt(val) {
674 None => continue,
675 Some(ty) => ty,
676 }
677 }
678 None => self.tcx.types.unit,
679 };
680 if self.can_eq(self.param_env, ty, expected) {
681 err.span_label(ex.span, "expected because of this `break`");
682 exit = true;
683 }
684 }
685 if exit {
686 break 'outer;
687 }
688 }
689 }
690 }
691 }
692 }
693
694 fn annotate_expected_due_to_let_ty(
695 &self,
696 err: &mut Diag<'_>,
697 expr: &hir::Expr<'_>,
698 error: Option<TypeError<'tcx>>,
699 ) {
700 match (self.tcx.parent_hir_node(expr.hir_id), error) {
701 (hir::Node::LetStmt(hir::LetStmt { ty: Some(ty), init: Some(init), .. }), _)
702 if init.hir_id == expr.hir_id && !ty.span.source_equal(init.span) =>
703 {
704 err.span_label(ty.span, "expected due to this");
706 }
707 (
708 hir::Node::Expr(hir::Expr { kind: hir::ExprKind::Assign(lhs, rhs, _), .. }),
709 Some(TypeError::Sorts(ExpectedFound { expected, .. })),
710 ) if rhs.hir_id == expr.hir_id && !expected.is_closure() => {
711 let mut primary_span = lhs.span;
714 let mut secondary_span = lhs.span;
715 let mut post_message = "";
716 match lhs.kind {
717 hir::ExprKind::Path(hir::QPath::Resolved(
718 None,
719 hir::Path {
720 res:
721 hir::def::Res::Def(
722 hir::def::DefKind::Static { .. }
723 | hir::def::DefKind::Const { .. },
724 def_id,
725 ),
726 ..
727 },
728 )) => {
729 if let Some(hir::Node::Item(hir::Item {
730 kind:
731 hir::ItemKind::Static(_, ident, ty, _)
732 | hir::ItemKind::Const(ident, _, ty, _),
733 ..
734 })) = self.tcx.hir_get_if_local(*def_id)
735 {
736 primary_span = ty.span;
737 secondary_span = ident.span;
738 post_message = " type";
739 }
740 }
741 hir::ExprKind::Path(hir::QPath::Resolved(
742 None,
743 hir::Path { res: hir::def::Res::Local(hir_id), .. },
744 )) => {
745 if let hir::Node::Pat(pat) = self.tcx.hir_node(*hir_id) {
746 primary_span = pat.span;
747 secondary_span = pat.span;
748 match self.tcx.parent_hir_node(pat.hir_id) {
749 hir::Node::LetStmt(hir::LetStmt { ty: Some(ty), .. }) => {
750 primary_span = ty.span;
751 post_message = " type";
752 }
753 hir::Node::LetStmt(hir::LetStmt { init: Some(init), .. }) => {
754 primary_span = init.span;
755 post_message = " value";
756 }
757 hir::Node::Param(hir::Param { ty_span, .. }) => {
758 primary_span = *ty_span;
759 post_message = " parameter type";
760 }
761 _ => {}
762 }
763 }
764 }
765 _ => {}
766 }
767
768 if primary_span != secondary_span
769 && self
770 .tcx
771 .sess
772 .source_map()
773 .is_multiline(secondary_span.shrink_to_hi().until(primary_span))
774 {
775 err.span_label(secondary_span, "expected due to the type of this binding");
778 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}"));
779 } else if post_message.is_empty() {
780 err.span_label(primary_span, "expected due to the type of this binding");
782 } else {
783 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}"));
785 }
786
787 if !lhs.is_syntactic_place_expr() {
788 err.downgrade_to_delayed_bug();
791 }
792 }
793 (
794 hir::Node::Expr(hir::Expr { kind: hir::ExprKind::Binary(_, lhs, rhs), .. }),
795 Some(TypeError::Sorts(ExpectedFound { expected, .. })),
796 ) if rhs.hir_id == expr.hir_id
797 && self.typeck_results.borrow().expr_ty_adjusted_opt(lhs) == Some(expected)
798 && !#[allow(non_exhaustive_omitted_patterns)] match lhs.kind {
hir::ExprKind::Let(..) => true,
_ => false,
}matches!(lhs.kind, hir::ExprKind::Let(..)) =>
800 {
801 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}`"));
802 }
803 _ => {}
804 }
805 }
806
807 fn annotate_mut_binding_to_immutable_binding(
826 &self,
827 err: &mut Diag<'_>,
828 expr: &hir::Expr<'_>,
829 expr_ty: Ty<'tcx>,
830 expected: Ty<'tcx>,
831 error: Option<TypeError<'tcx>>,
832 ) -> bool {
833 if let Some(TypeError::Sorts(ExpectedFound { .. })) = error
834 && let ty::Ref(_, inner, hir::Mutability::Not) = expected.kind()
835
836 && self.can_eq(self.param_env, *inner, expr_ty)
838
839 && let hir::Node::Expr(hir::Expr { kind: hir::ExprKind::Assign(lhs, rhs, _), .. }) =
841 self.tcx.parent_hir_node(expr.hir_id)
842 && rhs.hir_id == expr.hir_id
843
844 && let hir::ExprKind::Path(hir::QPath::Resolved(
846 None,
847 hir::Path { res: hir::def::Res::Local(hir_id), .. },
848 )) = lhs.kind
849 && let hir::Node::Pat(pat) = self.tcx.hir_node(*hir_id)
850
851 && let hir::Node::Param(hir::Param { ty_span, .. }) =
853 self.tcx.parent_hir_node(pat.hir_id)
854 && let item = self.tcx.hir_get_parent_item(pat.hir_id)
855 && let item = self.tcx.hir_owner_node(item)
856 && let Some(fn_decl) = item.fn_decl()
857
858 && let hir::PatKind::Binding(hir::BindingMode::MUT, _hir_id, ident, _) = pat.kind
860
861 && let Some(ty_ref) = fn_decl
863 .inputs
864 .iter()
865 .filter_map(|ty| match ty.kind {
866 hir::TyKind::Ref(lt, mut_ty) if ty.span == *ty_span => Some((lt, mut_ty)),
867 _ => None,
868 })
869 .next()
870 {
871 let mut sugg = if ty_ref.1.mutbl.is_mut() {
872 ::alloc::vec::Vec::new()vec![]
874 } else {
875 ::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![(
877 ty_ref.1.ty.span.shrink_to_lo(),
878 format!("{}mut ", if ty_ref.0.ident.span.is_empty() { "" } else { " " },),
879 )]
880 };
881 sugg.extend([
882 (pat.span.until(ident.span), String::new()),
883 (lhs.span.shrink_to_lo(), "*".to_string()),
884 ]);
885 err.multipart_suggestion(
888 "you might have meant to mutate the pointed at value being passed in, instead of \
889 changing the reference in the local binding",
890 sugg,
891 Applicability::MaybeIncorrect,
892 );
893 return true;
894 }
895 false
896 }
897
898 fn annotate_alternative_method_deref(
899 &self,
900 err: &mut Diag<'_>,
901 expr: &hir::Expr<'_>,
902 error: Option<TypeError<'tcx>>,
903 ) {
904 let Some(TypeError::Sorts(ExpectedFound { expected, .. })) = error else {
905 return;
906 };
907 let hir::Node::Expr(hir::Expr { kind: hir::ExprKind::Assign(lhs, rhs, _), .. }) =
908 self.tcx.parent_hir_node(expr.hir_id)
909 else {
910 return;
911 };
912 if rhs.hir_id != expr.hir_id || expected.is_closure() {
913 return;
914 }
915 let hir::ExprKind::Unary(hir::UnOp::Deref, deref) = lhs.kind else {
916 return;
917 };
918 let hir::ExprKind::MethodCall(path, base, args, _) = deref.kind else {
919 return;
920 };
921 let Some(self_ty) = self.typeck_results.borrow().expr_ty_adjusted_opt(base) else {
922 return;
923 };
924
925 let Ok(pick) = self.lookup_probe_for_diagnostic(
926 path.ident,
927 self_ty,
928 deref,
929 probe::ProbeScope::TraitsInScope,
930 None,
931 ) else {
932 return;
933 };
934
935 let Ok(in_scope_methods) = self.probe_for_name_many(
936 probe::Mode::MethodCall,
937 path.ident,
938 Some(expected),
939 probe::IsSuggestion(true),
940 self_ty,
941 deref.hir_id,
942 probe::ProbeScope::TraitsInScope,
943 ) else {
944 return;
945 };
946
947 let other_methods_in_scope: Vec<_> =
948 in_scope_methods.iter().filter(|c| c.item.def_id != pick.item.def_id).collect();
949
950 let Ok(all_methods) = self.probe_for_name_many(
951 probe::Mode::MethodCall,
952 path.ident,
953 Some(expected),
954 probe::IsSuggestion(true),
955 self_ty,
956 deref.hir_id,
957 probe::ProbeScope::AllTraits,
958 ) else {
959 return;
960 };
961
962 let suggestions: Vec<_> = all_methods
963 .into_iter()
964 .filter(|c| c.item.def_id != pick.item.def_id)
965 .map(|c| {
966 let m = c.item;
967 let generic_args = ty::GenericArgs::for_item(self.tcx, m.def_id, |param, _| {
968 self.var_for_def(deref.span, param)
969 });
970 let mutability =
971 match self.tcx.fn_sig(m.def_id).skip_binder().input(0).skip_binder().kind() {
972 ty::Ref(_, _, hir::Mutability::Mut) => "&mut ",
973 ty::Ref(_, _, _) => "&",
974 _ => "",
975 };
976 ::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![
977 (
978 deref.span.until(base.span),
979 format!(
980 "{}({}",
981 with_no_trimmed_paths!(
982 self.tcx.def_path_str_with_args(m.def_id, generic_args,)
983 ),
984 mutability,
985 ),
986 ),
987 match &args {
988 [] => (base.span.shrink_to_hi().with_hi(deref.span.hi()), ")".to_string()),
989 [first, ..] => (base.span.between(first.span), ", ".to_string()),
990 },
991 ]
992 })
993 .collect();
994 if suggestions.is_empty() {
995 return;
996 }
997 let mut path_span: MultiSpan = path.ident.span.into();
998 path_span.push_span_label(
999 path.ident.span,
1000 {
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!(
1001 "refers to `{}`",
1002 self.tcx.def_path_str(pick.item.def_id),
1003 )),
1004 );
1005 let container_id = pick.item.container_id(self.tcx);
1006 let container = { let _guard = NoTrimmedGuard::new(); self.tcx.def_path_str(container_id) }with_no_trimmed_paths!(self.tcx.def_path_str(container_id));
1007 for &def_id in pick.import_ids {
1008 let hir_id = self.tcx.local_def_id_to_hir_id(def_id);
1009 path_span
1010 .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"));
1011 }
1012 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[..] {
1013 [] => return,
1014 [candidate] => format!(
1015 "the method of the same name on {} `{}`",
1016 match candidate.kind {
1017 probe::CandidateKind::InherentImplCandidate { .. } => "the inherent impl for",
1018 _ => "trait",
1019 },
1020 self.tcx.def_path_str(candidate.item.container_id(self.tcx))
1021 ),
1022 _ if other_methods_in_scope.len() < 5 => {
1023 format!(
1024 "the methods of the same name on {}",
1025 listify(
1026 &other_methods_in_scope[..other_methods_in_scope.len() - 1],
1027 |c| format!("`{}`", self.tcx.def_path_str(c.item.container_id(self.tcx)))
1028 )
1029 .unwrap_or_default(),
1030 )
1031 }
1032 _ => format!(
1033 "the methods of the same name on {} other traits",
1034 other_methods_in_scope.len()
1035 ),
1036 });
1037 err.span_note(
1038 path_span,
1039 ::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!(
1040 "the `{}` call is resolved to the method in `{container}`, shadowing {tail}",
1041 path.ident,
1042 ),
1043 );
1044 if suggestions.len() > other_methods_in_scope.len() {
1045 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!(
1046 "additionally, there are {} other available methods that aren't in scope",
1047 suggestions.len() - other_methods_in_scope.len()
1048 ));
1049 }
1050 err.multipart_suggestions(
1051 ::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!(
1052 "you might have meant to call {}; you can use the fully-qualified path to call {} \
1053 explicitly",
1054 if suggestions.len() == 1 {
1055 "the other method"
1056 } else {
1057 "one of the other methods"
1058 },
1059 if suggestions.len() == 1 { "it" } else { "one of them" },
1060 ),
1061 suggestions,
1062 Applicability::MaybeIncorrect,
1063 );
1064 }
1065
1066 pub(crate) fn get_conversion_methods_for_diagnostic(
1067 &self,
1068 span: Span,
1069 expected: Ty<'tcx>,
1070 checked_ty: Ty<'tcx>,
1071 hir_id: hir::HirId,
1072 ) -> Vec<AssocItem> {
1073 let methods = self.probe_for_return_type_for_diagnostic(
1074 span,
1075 probe::Mode::MethodCall,
1076 expected,
1077 checked_ty,
1078 hir_id,
1079 |m| {
1080 self.has_only_self_parameter(m)
1081 && {
{
'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)
1092 },
1093 );
1094
1095 methods
1096 }
1097
1098 fn has_only_self_parameter(&self, method: &AssocItem) -> bool {
1100 method.is_method()
1101 && self.tcx.fn_sig(method.def_id).skip_binder().inputs().skip_binder().len() == 1
1102 }
1103
1104 pub(crate) fn maybe_get_block_expr(
1106 &self,
1107 expr: &hir::Expr<'tcx>,
1108 ) -> Option<&'tcx hir::Expr<'tcx>> {
1109 match expr {
1110 hir::Expr { kind: hir::ExprKind::Block(block, ..), .. } => block.expr,
1111 _ => None,
1112 }
1113 }
1114
1115 pub(crate) fn is_destruct_assignment_desugaring(&self, expr: &hir::Expr<'_>) -> bool {
1120 if let hir::ExprKind::Path(hir::QPath::Resolved(
1121 _,
1122 hir::Path { res: hir::def::Res::Local(bind_hir_id), .. },
1123 )) = expr.kind
1124 && let bind = self.tcx.hir_node(*bind_hir_id)
1125 && let parent = self.tcx.parent_hir_node(*bind_hir_id)
1126 && let hir::Node::Pat(hir::Pat {
1127 kind: hir::PatKind::Binding(_, _hir_id, _, _), ..
1128 }) = bind
1129 && let hir::Node::Pat(hir::Pat { default_binding_modes: false, .. }) = parent
1130 {
1131 true
1132 } else {
1133 false
1134 }
1135 }
1136
1137 fn explain_self_literal(
1138 &self,
1139 err: &mut Diag<'_>,
1140 expr: &hir::Expr<'tcx>,
1141 expected: Ty<'tcx>,
1142 found: Ty<'tcx>,
1143 ) {
1144 match expr.peel_drop_temps().kind {
1145 hir::ExprKind::Struct(
1146 hir::QPath::Resolved(
1147 None,
1148 hir::Path { res: hir::def::Res::SelfTyAlias { alias_to, .. }, span, .. },
1149 ),
1150 ..,
1151 )
1152 | hir::ExprKind::Call(
1153 hir::Expr {
1154 kind:
1155 hir::ExprKind::Path(hir::QPath::Resolved(
1156 None,
1157 hir::Path {
1158 res: hir::def::Res::SelfTyAlias { alias_to, .. },
1159 span,
1160 ..
1161 },
1162 )),
1163 ..
1164 },
1165 ..,
1166 ) => {
1167 if let Some(hir::Node::Item(hir::Item {
1168 kind: hir::ItemKind::Impl(hir::Impl { self_ty, .. }),
1169 ..
1170 })) = self.tcx.hir_get_if_local(*alias_to)
1171 {
1172 err.span_label(self_ty.span, "this is the type of the `Self` literal");
1173 }
1174 if let ty::Adt(e_def, e_args) = expected.kind()
1175 && let ty::Adt(f_def, _f_args) = found.kind()
1176 && e_def == f_def
1177 {
1178 err.span_suggestion_verbose(
1179 *span,
1180 "use the type name directly",
1181 self.tcx.value_path_str_with_args(e_def.did(), e_args),
1182 Applicability::MaybeIncorrect,
1183 );
1184 }
1185 }
1186 _ => {}
1187 }
1188 }
1189
1190 fn note_wrong_return_ty_due_to_generic_arg(
1191 &self,
1192 err: &mut Diag<'_>,
1193 expr: &hir::Expr<'_>,
1194 checked_ty: Ty<'tcx>,
1195 ) {
1196 let hir::Node::Expr(parent_expr) = self.tcx.parent_hir_node(expr.hir_id) else {
1197 return;
1198 };
1199 if parent_expr.span.desugaring_kind().is_some() {
1200 return;
1201 }
1202 enum CallableKind {
1203 Function,
1204 Method,
1205 Constructor,
1206 }
1207 let mut maybe_emit_help = |def_id: hir::def_id::DefId,
1208 callable: Ident,
1209 args: &[hir::Expr<'_>],
1210 kind: CallableKind| {
1211 let arg_idx = args.iter().position(|a| a.hir_id == expr.hir_id).unwrap();
1212 let fn_ty = self.tcx.type_of(def_id).skip_binder();
1213 if !fn_ty.is_fn() {
1214 return;
1215 }
1216 let fn_sig = fn_ty.fn_sig(self.tcx).skip_binder();
1217 let Some(&arg) = fn_sig
1218 .inputs()
1219 .get(arg_idx + if #[allow(non_exhaustive_omitted_patterns)] match kind {
CallableKind::Method => true,
_ => false,
}matches!(kind, CallableKind::Method) { 1 } else { 0 })
1220 else {
1221 return;
1222 };
1223 if #[allow(non_exhaustive_omitted_patterns)] match arg.kind() {
ty::Param(_) => true,
_ => false,
}matches!(arg.kind(), ty::Param(_))
1224 && fn_sig.output().contains(arg)
1225 && self.node_ty(args[arg_idx].hir_id) == checked_ty
1226 {
1227 let mut multi_span: MultiSpan = parent_expr.span.into();
1228 multi_span.push_span_label(
1229 args[arg_idx].span,
1230 ::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!(
1231 "this argument influences the {} of `{}`",
1232 if matches!(kind, CallableKind::Constructor) {
1233 "type"
1234 } else {
1235 "return type"
1236 },
1237 callable
1238 ),
1239 );
1240 err.span_help(
1241 multi_span,
1242 ::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!(
1243 "the {} `{}` due to the type of the argument passed",
1244 match kind {
1245 CallableKind::Function => "return type of this call is",
1246 CallableKind::Method => "return type of this call is",
1247 CallableKind::Constructor => "type constructed contains",
1248 },
1249 checked_ty
1250 ),
1251 );
1252 }
1253 };
1254 match parent_expr.kind {
1255 hir::ExprKind::Call(fun, args) => {
1256 let hir::ExprKind::Path(hir::QPath::Resolved(_, path)) = fun.kind else {
1257 return;
1258 };
1259 let hir::def::Res::Def(kind, def_id) = path.res else {
1260 return;
1261 };
1262 let callable_kind = if #[allow(non_exhaustive_omitted_patterns)] match kind {
hir::def::DefKind::Ctor(_, _) => true,
_ => false,
}matches!(kind, hir::def::DefKind::Ctor(_, _)) {
1263 CallableKind::Constructor
1264 } else {
1265 CallableKind::Function
1266 };
1267 maybe_emit_help(def_id, path.segments.last().unwrap().ident, args, callable_kind);
1268 }
1269 hir::ExprKind::MethodCall(method, _receiver, args, _span) => {
1270 let Some(def_id) =
1271 self.typeck_results.borrow().type_dependent_def_id(parent_expr.hir_id)
1272 else {
1273 return;
1274 };
1275 maybe_emit_help(def_id, method.ident, args, CallableKind::Method)
1276 }
1277 _ => return,
1278 }
1279 }
1280}
1281
1282pub(crate) enum TypeMismatchSource<'tcx> {
1283 Ty(Ty<'tcx>),
1286 Arg { call_expr: &'tcx hir::Expr<'tcx>, incompatible_arg: usize },
1290}