1use std::borrow::Cow;
49use std::ops::ControlFlow;
50use std::path::PathBuf;
51use std::{cmp, fmt, iter};
52
53use rustc_abi::ExternAbi;
54use rustc_data_structures::fx::{FxHashSet, FxIndexMap, FxIndexSet};
55use rustc_errors::{Applicability, Diag, DiagStyledString, IntoDiagArg, StringPart, pluralize};
56use rustc_hir::attrs::diagnostic::{CustomDiagnostic, Directive, FormatArgs};
57use rustc_hir::def_id::{CRATE_DEF_ID, DefId};
58use rustc_hir::intravisit::Visitor;
59use rustc_hir::lang_items::LangItem;
60use rustc_hir::{self as hir, find_attr};
61use rustc_infer::infer::DefineOpaqueTypes;
62use rustc_macros::extension;
63use rustc_middle::bug;
64use rustc_middle::traits::PatternOriginExpr;
65use rustc_middle::ty::error::{ExpectedFound, TypeError, TypeErrorToStringExt};
66use rustc_middle::ty::print::{PrintTraitRefExt as _, WrapBinderMode, with_forced_trimmed_paths};
67use rustc_middle::ty::{
68 self, List, ParamEnv, Region, Ty, TyCtxt, TypeFoldable, TypeSuperVisitable, TypeVisitable,
69 TypeVisitableExt, Unnormalized,
70};
71use rustc_span::{BytePos, DUMMY_SP, DesugaringKind, Pos, Span, sym};
72use thin_vec::ThinVec;
73use tracing::{debug, instrument};
74
75use crate::diagnostics::{ObligationCauseFailureCode, TypeErrorAdditionalDiags};
76use crate::error_reporting::TypeErrCtxt;
77use crate::error_reporting::traits::ambiguity::{
78 CandidateSource, compute_applicable_impls_for_diagnostics,
79};
80use crate::infer;
81use crate::infer::relate::{self, RelateResult, TypeRelation};
82use crate::infer::{InferCtxt, InferCtxtExt as _, TypeTrace, ValuePairs};
83use crate::solve::deeply_normalize_for_diagnostics;
84use crate::traits::{
85 MatchExpressionArmCause, Obligation, ObligationCause, ObligationCauseCode, ObligationCtxt,
86 specialization_graph,
87};
88
89mod note_and_explain;
90mod suggest;
91
92pub mod need_type_info;
93pub mod nice_region_error;
94pub mod region;
95
96fn escape_literal(s: &str) -> String {
99 let mut escaped = String::with_capacity(s.len());
100 let mut chrs = s.chars().peekable();
101 while let Some(first) = chrs.next() {
102 match (first, chrs.peek()) {
103 ('\\', Some(&delim @ '"') | Some(&delim @ '\'')) => {
104 escaped.push('\\');
105 escaped.push(delim);
106 chrs.next();
107 }
108 ('"' | '\'', _) => {
109 escaped.push('\\');
110 escaped.push(first)
111 }
112 (c, _) => escaped.push(c),
113 };
114 }
115 escaped
116}
117
118impl<'a, 'tcx> TypeErrCtxt<'a, 'tcx> {
119 fn normalize_fn_sig(
120 &self,
121 fn_sig: Unnormalized<'tcx, ty::PolyFnSig<'tcx>>,
122 ) -> ty::PolyFnSig<'tcx> {
123 let Some(param_env) = self.param_env else {
124 return fn_sig.skip_normalization();
125 };
126
127 if fn_sig.skip_normalization().has_escaping_bound_vars() {
128 return fn_sig.skip_normalization();
129 }
130
131 self.probe(|_| {
132 let ocx = ObligationCtxt::new(self);
133 let normalized_fn_sig = ocx.normalize(&ObligationCause::dummy(), param_env, fn_sig);
134 if ocx.evaluate_obligations_error_on_ambiguity().is_empty() {
135 let normalized_fn_sig = self.resolve_vars_if_possible(normalized_fn_sig);
136 if !normalized_fn_sig.has_infer() {
137 return normalized_fn_sig;
138 }
139 }
140 fn_sig.skip_normalization()
141 })
142 }
143
144 pub fn type_error_struct_with_diag<M>(
155 &self,
156 sp: Span,
157 mk_diag: M,
158 actual_ty: Ty<'tcx>,
159 ) -> Diag<'a>
160 where
161 M: FnOnce(String) -> Diag<'a>,
162 {
163 let actual_ty = self.resolve_vars_if_possible(actual_ty);
164 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs:164",
"rustc_trait_selection::error_reporting::infer",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs"),
::tracing_core::__macro_support::Option::Some(164u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::error_reporting::infer"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("type_error_struct_with_diag({0:?}, {1:?})",
sp, actual_ty) as &dyn Value))])
});
} else { ; }
};debug!("type_error_struct_with_diag({:?}, {:?})", sp, actual_ty);
165
166 let mut err = mk_diag(self.ty_to_string(actual_ty));
167
168 if actual_ty.references_error() {
170 err.downgrade_to_delayed_bug();
171 }
172
173 err
174 }
175
176 pub fn report_mismatched_types(
177 &self,
178 cause: &ObligationCause<'tcx>,
179 param_env: ty::ParamEnv<'tcx>,
180 expected: Ty<'tcx>,
181 actual: Ty<'tcx>,
182 err: TypeError<'tcx>,
183 ) -> Diag<'a> {
184 let mut diag = self.report_and_explain_type_error(
185 TypeTrace::types(cause, expected, actual),
186 param_env,
187 err,
188 );
189
190 self.suggest_param_env_shadowing(&mut diag, expected, actual, param_env);
191
192 diag
193 }
194
195 pub fn report_mismatched_consts(
196 &self,
197 cause: &ObligationCause<'tcx>,
198 param_env: ty::ParamEnv<'tcx>,
199 expected: ty::Const<'tcx>,
200 actual: ty::Const<'tcx>,
201 err: TypeError<'tcx>,
202 ) -> Diag<'a> {
203 self.report_and_explain_type_error(
204 TypeTrace::consts(cause, expected, actual),
205 param_env,
206 err,
207 )
208 }
209
210 pub fn get_impl_future_output_ty(&self, ty: Ty<'tcx>) -> Option<Ty<'tcx>> {
211 let (def_id, args) = match *ty.kind() {
212 ty::Alias(ty::AliasTy { kind: ty::Opaque { def_id }, args, .. }) => (def_id, args),
213 ty::Alias(ty::AliasTy { kind, args, .. })
214 if self.tcx.is_impl_trait_in_trait(kind.def_id()) =>
215 {
216 (kind.def_id(), args)
217 }
218 _ => return None,
219 };
220
221 let future_trait = self.tcx.require_lang_item(LangItem::Future, DUMMY_SP);
222 let item_def_id = self.tcx.associated_item_def_ids(future_trait)[0];
223
224 self.tcx
225 .explicit_item_self_bounds(def_id)
226 .iter_instantiated_copied(self.tcx, args)
227 .map(Unnormalized::skip_norm_wip)
228 .find_map(|(predicate, _)| {
229 predicate
230 .kind()
231 .map_bound(|kind| match kind {
232 ty::ClauseKind::Projection(projection_predicate)
233 if projection_predicate.def_id() == item_def_id =>
234 {
235 projection_predicate.term.as_type()
236 }
237 _ => None,
238 })
239 .no_bound_vars()
240 .flatten()
241 })
242 }
243
244 fn check_and_note_conflicting_crates(&self, err: &mut Diag<'_>, terr: TypeError<'tcx>) -> bool {
246 match terr {
247 TypeError::Sorts(ref exp_found) => {
248 if let (&ty::Adt(exp_adt, _), &ty::Adt(found_adt, _)) =
251 (exp_found.expected.kind(), exp_found.found.kind())
252 {
253 return self.check_same_definition_different_crate(
254 err,
255 exp_adt.did(),
256 [found_adt.did()].into_iter(),
257 |did| ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[self.tcx.def_span(did)]))vec![self.tcx.def_span(did)],
258 "type",
259 );
260 }
261 }
262 TypeError::Traits(ref exp_found) => {
263 return self.check_same_definition_different_crate(
264 err,
265 exp_found.expected,
266 [exp_found.found].into_iter(),
267 |did| ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[self.tcx.def_span(did)]))vec![self.tcx.def_span(did)],
268 "trait",
269 );
270 }
271 _ => (), }
273 false
274 }
275
276 fn suggest_param_env_shadowing(
277 &self,
278 diag: &mut Diag<'_>,
279 expected: Ty<'tcx>,
280 found: Ty<'tcx>,
281 param_env: ty::ParamEnv<'tcx>,
282 ) {
283 let (alias, &def_id, concrete) = match (expected.kind(), found.kind()) {
284 (ty::Alias(proj @ ty::AliasTy { kind: ty::Projection { def_id }, .. }), _) => {
285 (proj, def_id, found)
286 }
287 (_, ty::Alias(proj @ ty::AliasTy { kind: ty::Projection { def_id }, .. })) => {
288 (proj, def_id, expected)
289 }
290 _ => return,
291 };
292
293 let tcx = self.tcx;
294
295 let trait_ref = alias.trait_ref(tcx);
296 let obligation =
297 Obligation::new(tcx, ObligationCause::dummy(), param_env, ty::Binder::dummy(trait_ref));
298
299 let applicable_impls = compute_applicable_impls_for_diagnostics(self.infcx, &obligation);
300
301 for candidate in applicable_impls {
302 let impl_def_id = match candidate {
303 CandidateSource::DefId(did) => did,
304 CandidateSource::ParamEnv(_) => continue,
305 };
306
307 let is_shadowed = self.infcx.probe(|_| {
308 let impl_substs = self.infcx.fresh_args_for_item(DUMMY_SP, impl_def_id);
309 let impl_trait_ref =
310 tcx.impl_trait_ref(impl_def_id).instantiate(tcx, impl_substs).skip_norm_wip();
311
312 let expected_trait_ref = alias.trait_ref(tcx);
313
314 if let Err(_) = self.infcx.at(&ObligationCause::dummy(), param_env).eq(
315 DefineOpaqueTypes::No,
316 expected_trait_ref,
317 impl_trait_ref,
318 ) {
319 return false;
320 }
321
322 let leaf_def = match specialization_graph::assoc_def(tcx, impl_def_id, def_id) {
323 Ok(leaf) => leaf,
324 Err(_) => return false,
325 };
326
327 let trait_def_id = alias.trait_def_id(tcx);
328 let rebased_args = alias.args.rebase_onto(tcx, trait_def_id, impl_substs);
329
330 if !leaf_def.item.defaultness(tcx).has_value() {
333 return false;
334 }
335
336 let impl_item_def_id = leaf_def.item.def_id;
337 if !tcx.check_args_compatible(impl_item_def_id, rebased_args) {
338 return false;
339 }
340 let impl_assoc_ty =
341 tcx.type_of(impl_item_def_id).instantiate(tcx, rebased_args).skip_norm_wip();
342
343 self.infcx.can_eq(param_env, impl_assoc_ty, concrete)
344 });
345
346 if is_shadowed {
347 diag.note(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("the associated type `{0}` is defined as `{1}` in the implementation, but the where-bound `{2}` shadows this definition\nsee issue #152409 <https://github.com/rust-lang/rust/issues/152409> for more information",
self.ty_to_string(tcx.mk_ty_from_kind(ty::Alias(*alias))),
self.ty_to_string(concrete),
self.ty_to_string(alias.self_ty())))
})format!(
348 "the associated type `{}` is defined as `{}` in the implementation, \
349 but the where-bound `{}` shadows this definition\n\
350 see issue #152409 <https://github.com/rust-lang/rust/issues/152409> for more information",
351 self.ty_to_string(tcx.mk_ty_from_kind(ty::Alias(*alias))),
352 self.ty_to_string(concrete),
353 self.ty_to_string(alias.self_ty())
354 ));
355 return;
356 }
357 }
358 }
359
360 fn note_error_origin(
361 &self,
362 err: &mut Diag<'_>,
363 cause: &ObligationCause<'tcx>,
364 exp_found: Option<ty::error::ExpectedFound<Ty<'tcx>>>,
365 terr: TypeError<'tcx>,
366 param_env: Option<ParamEnv<'tcx>>,
367 ) {
368 match *cause.code() {
369 ObligationCauseCode::Pattern {
370 origin_expr: Some(origin_expr),
371 span: Some(span),
372 root_ty,
373 } => {
374 let expected_ty = self.resolve_vars_if_possible(root_ty);
375 if !#[allow(non_exhaustive_omitted_patterns)] match expected_ty.kind() {
ty::Infer(ty::InferTy::TyVar(_) | ty::InferTy::FreshTy(_)) => true,
_ => false,
}matches!(
376 expected_ty.kind(),
377 ty::Infer(ty::InferTy::TyVar(_) | ty::InferTy::FreshTy(_))
378 ) {
379 if span.desugaring_kind() == Some(DesugaringKind::ForLoop)
381 && let ty::Adt(def, args) = expected_ty.kind()
382 && Some(def.did()) == self.tcx.get_diagnostic_item(sym::Option)
383 {
384 err.span_label(
385 span,
386 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("this is an iterator with items of type `{0}`",
args.type_at(0)))
})format!("this is an iterator with items of type `{}`", args.type_at(0)),
387 );
388 } else if !span.overlaps(cause.span) {
389 let expected_ty = self.tcx.short_string(expected_ty, err.long_ty_path());
390 err.span_label(span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("this expression has type `{0}`",
expected_ty))
})format!("this expression has type `{expected_ty}`"));
391 }
392 }
393 if let Some(ty::error::ExpectedFound { found, .. }) = exp_found
394 && let Ok(mut peeled_snippet) =
395 self.tcx.sess.source_map().span_to_snippet(origin_expr.peeled_span)
396 {
397 if origin_expr.peeled_prefix_suggestion_parentheses {
402 peeled_snippet = ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("({0})", peeled_snippet))
})format!("({peeled_snippet})");
403 }
404
405 if expected_ty.boxed_ty() == Some(found) {
408 err.span_suggestion_verbose(
409 span,
410 "consider dereferencing the boxed value",
411 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("*{0}", peeled_snippet))
})format!("*{peeled_snippet}"),
412 Applicability::MachineApplicable,
413 );
414 } else if let Some(param_env) = param_env
415 && let Some(prefix) = self.should_deref_suggestion_on_mismatch(
416 param_env,
417 found,
418 expected_ty,
419 origin_expr,
420 )
421 {
422 err.span_suggestion_verbose(
423 span,
424 "consider dereferencing to access the inner value using the `Deref` trait",
425 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}{1}", prefix, peeled_snippet))
})format!("{prefix}{peeled_snippet}"),
426 Applicability::MaybeIncorrect,
427 );
428 }
429 }
430 }
431 ObligationCauseCode::Pattern { origin_expr: None, span: Some(span), .. } => {
432 err.span_label(span, "expected due to this");
433 }
434 ObligationCauseCode::BlockTailExpression(
435 _,
436 hir::MatchSource::TryDesugar(scrut_hir_id),
437 ) => {
438 if let Some(ty::error::ExpectedFound { expected, .. }) = exp_found {
439 let scrut_expr = self.tcx.hir_expect_expr(scrut_hir_id);
440 let scrut_ty = if let hir::ExprKind::Call(_, args) = &scrut_expr.kind {
441 let arg_expr = args.first().expect("try desugaring call w/out arg");
442 self.typeck_results
443 .as_ref()
444 .and_then(|typeck_results| typeck_results.expr_ty_opt(arg_expr))
445 } else {
446 ::rustc_middle::util::bug::bug_fmt(format_args!("try desugaring w/out call expr as scrutinee"));bug!("try desugaring w/out call expr as scrutinee");
447 };
448
449 match scrut_ty {
450 Some(ty) if expected == ty => {
451 let source_map = self.tcx.sess.source_map();
452 err.span_suggestion(
453 source_map.end_point(cause.span),
454 "try removing this `?`",
455 "",
456 Applicability::MachineApplicable,
457 );
458 }
459 _ => {}
460 }
461 }
462 }
463 ObligationCauseCode::MatchExpressionArm(MatchExpressionArmCause {
464 arm_block_id,
465 arm_span,
466 arm_ty,
467 prior_arm_block_id,
468 prior_arm_span,
469 prior_arm_ty,
470 source,
471 ref prior_non_diverging_arms,
472 scrut_span,
473 expr_span,
474 ..
475 }) => match source {
476 hir::MatchSource::TryDesugar(scrut_hir_id) => {
477 if let Some(ty::error::ExpectedFound { expected, .. }) = exp_found {
478 let scrut_expr = self.tcx.hir_expect_expr(scrut_hir_id);
479 let scrut_ty = if let hir::ExprKind::Call(_, args) = &scrut_expr.kind {
480 let arg_expr = args.first().expect("try desugaring call w/out arg");
481 self.typeck_results
482 .as_ref()
483 .and_then(|typeck_results| typeck_results.expr_ty_opt(arg_expr))
484 } else {
485 ::rustc_middle::util::bug::bug_fmt(format_args!("try desugaring w/out call expr as scrutinee"));bug!("try desugaring w/out call expr as scrutinee");
486 };
487
488 match scrut_ty {
489 Some(ty) if expected == ty => {
490 let source_map = self.tcx.sess.source_map();
491 err.span_suggestion(
492 source_map.end_point(cause.span),
493 "try removing this `?`",
494 "",
495 Applicability::MachineApplicable,
496 );
497 }
498 _ => {}
499 }
500 }
501 }
502 _ => {
503 let t = self.resolve_vars_if_possible(match exp_found {
505 Some(ty::error::ExpectedFound { expected, .. }) => expected,
506 _ => prior_arm_ty,
507 });
508 let source_map = self.tcx.sess.source_map();
509 let mut any_multiline_arm = source_map.is_multiline(arm_span);
510 if prior_non_diverging_arms.len() <= 4 {
511 for sp in prior_non_diverging_arms {
512 any_multiline_arm |= source_map.is_multiline(*sp);
513 err.span_label(*sp, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("this is found to be of type `{0}`",
t))
})format!("this is found to be of type `{t}`"));
514 }
515 } else if let Some(sp) = prior_non_diverging_arms.last() {
516 any_multiline_arm |= source_map.is_multiline(*sp);
517 err.span_label(
518 *sp,
519 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("this and all prior arms are found to be of type `{0}`",
t))
})format!("this and all prior arms are found to be of type `{t}`"),
520 );
521 }
522 let outer = if any_multiline_arm || !source_map.is_multiline(expr_span) {
523 expr_span.shrink_to_lo().to(scrut_span)
526 } else {
527 expr_span
528 };
529 let msg = "`match` arms have incompatible types";
530 err.span_label(outer, msg);
531 if let Some(subdiag) = self.suggest_remove_semi_or_return_binding(
532 prior_arm_block_id,
533 prior_arm_ty,
534 prior_arm_span,
535 arm_block_id,
536 arm_ty,
537 arm_span,
538 ) {
539 err.subdiagnostic(subdiag);
540 }
541 }
542 },
543 ObligationCauseCode::IfExpression { expr_id, .. } => {
544 let hir::Node::Expr(&hir::Expr {
545 kind: hir::ExprKind::If(cond_expr, then_expr, Some(else_expr)),
546 span: expr_span,
547 ..
548 }) = self.tcx.hir_node(expr_id)
549 else {
550 return;
551 };
552 let then_span = self.find_block_span_from_hir_id(then_expr.hir_id);
553 let then_ty = self
554 .typeck_results
555 .as_ref()
556 .expect("if expression only expected inside FnCtxt")
557 .expr_ty(then_expr);
558 let else_span = self.find_block_span_from_hir_id(else_expr.hir_id);
559 let else_ty = self
560 .typeck_results
561 .as_ref()
562 .expect("if expression only expected inside FnCtxt")
563 .expr_ty(else_expr);
564 if let hir::ExprKind::If(_cond, _then, None) = else_expr.kind
565 && else_ty.is_unit()
566 {
567 err.note("`if` expressions without `else` evaluate to `()`");
569 err.note("consider adding an `else` block that evaluates to the expected type");
570 }
571 err.span_label(then_span, "expected because of this");
572
573 let outer_span = if self.tcx.sess.source_map().is_multiline(expr_span) {
574 if then_span.hi() == expr_span.hi() || else_span.hi() == expr_span.hi() {
575 Some(expr_span.shrink_to_lo().to(cond_expr.peel_drop_temps().span))
578 } else {
579 Some(expr_span)
580 }
581 } else {
582 None
583 };
584 if let Some(sp) = outer_span {
585 err.span_label(sp, "`if` and `else` have incompatible types");
586 }
587
588 let then_id = if let hir::ExprKind::Block(then_blk, _) = then_expr.kind {
589 then_blk.hir_id
590 } else {
591 then_expr.hir_id
592 };
593 let else_id = if let hir::ExprKind::Block(else_blk, _) = else_expr.kind {
594 else_blk.hir_id
595 } else {
596 else_expr.hir_id
597 };
598 if let Some(subdiag) = self.suggest_remove_semi_or_return_binding(
599 Some(then_id),
600 then_ty,
601 then_span,
602 Some(else_id),
603 else_ty,
604 else_span,
605 ) {
606 err.subdiagnostic(subdiag);
607 }
608 }
609 ObligationCauseCode::LetElse => {
610 err.help("try adding a diverging expression, such as `return` or `panic!(..)`");
611 err.help("...or use `match` instead of `let...else`");
612 }
613 _ => {
614 if let ObligationCauseCode::WhereClause(_, span)
615 | ObligationCauseCode::WhereClauseInExpr(_, span, ..) =
616 cause.code().peel_derives()
617 && !span.is_dummy()
618 && let TypeError::RegionsPlaceholderMismatch = terr
619 {
620 err.span_note(*span, "the lifetime requirement is introduced here");
621 }
622 }
623 }
624 }
625
626 fn should_deref_suggestion_on_mismatch(
629 &self,
630 param_env: ParamEnv<'tcx>,
631 deref_to: Ty<'tcx>,
632 deref_from: Ty<'tcx>,
633 origin_expr: PatternOriginExpr,
634 ) -> Option<String> {
635 let Some((num_derefs, (after_deref_ty, _))) = (self.autoderef_steps)(deref_from)
643 .into_iter()
644 .enumerate()
645 .find(|(_, (ty, _))| self.infcx.can_eq(param_env, *ty, deref_to))
646 else {
647 return None;
648 };
649
650 if num_derefs <= origin_expr.peeled_count {
651 return None;
652 }
653
654 let deref_part = "*".repeat(num_derefs - origin_expr.peeled_count);
655
656 if deref_from.is_ref() && !after_deref_ty.is_ref() {
659 Some(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("&{0}", deref_part))
})format!("&{deref_part}"))
660 } else {
661 Some(deref_part)
662 }
663 }
664
665 fn highlight_outer(
679 &self,
680 value: &mut DiagStyledString,
681 other_value: &mut DiagStyledString,
682 name: String,
683 args: &[ty::GenericArg<'tcx>],
684 pos: usize,
685 other_ty: Ty<'tcx>,
686 ) {
687 value.push_highlighted(name);
690
691 if args.is_empty() {
692 return;
693 }
694 value.push_highlighted("<");
695
696 for (i, arg) in args.iter().enumerate() {
697 if i > 0 {
698 value.push_normal(", ");
699 }
700
701 match arg.kind() {
702 ty::GenericArgKind::Lifetime(lt) => {
703 let s = lt.to_string();
704 value.push_normal(if s.is_empty() { "'_" } else { &s });
705 }
706 ty::GenericArgKind::Const(ct) => {
707 value.push_normal(ct.to_string());
708 }
709 ty::GenericArgKind::Type(type_arg) => {
712 if i == pos {
713 let values = self.cmp(type_arg, other_ty);
714 value.0.extend((values.0).0);
715 other_value.0.extend((values.1).0);
716 } else {
717 value.push_highlighted(type_arg.to_string());
718 }
719 }
720 }
721 }
722
723 value.push_highlighted(">");
724 }
725
726 fn cmp_type_arg(
747 &self,
748 t1_out: &mut DiagStyledString,
749 t2_out: &mut DiagStyledString,
750 path: String,
751 args: &'tcx [ty::GenericArg<'tcx>],
752 other_path: String,
753 other_ty: Ty<'tcx>,
754 ) -> bool {
755 for (i, arg) in args.iter().enumerate() {
756 if let Some(ta) = arg.as_type() {
757 if ta == other_ty {
758 self.highlight_outer(t1_out, t2_out, path, args, i, other_ty);
759 return true;
760 }
761 if let ty::Adt(def, _) = ta.kind() {
762 let path_ = self.tcx.def_path_str(def.did());
763 if path_ == other_path {
764 self.highlight_outer(t1_out, t2_out, path, args, i, other_ty);
765 return true;
766 }
767 }
768 }
769 }
770 false
771 }
772
773 fn push_comma(
775 &self,
776 value: &mut DiagStyledString,
777 other_value: &mut DiagStyledString,
778 pos: usize,
779 ) {
780 if pos > 0 {
781 value.push_normal(", ");
782 other_value.push_normal(", ");
783 }
784 }
785
786 fn cmp_fn_sig(
788 &self,
789 sig1: ty::PolyFnSig<'tcx>,
790 fn_def1: Option<(DefId, Option<&'tcx [ty::GenericArg<'tcx>]>)>,
791 sig2: ty::PolyFnSig<'tcx>,
792 fn_def2: Option<(DefId, Option<&'tcx [ty::GenericArg<'tcx>]>)>,
793 ) -> (DiagStyledString, DiagStyledString) {
794 let sig1 = self.normalize_fn_sig(Unnormalized::new_wip(sig1));
795 let sig2 = self.normalize_fn_sig(Unnormalized::new_wip(sig2));
796
797 let get_lifetimes = |sig| {
798 use rustc_hir::def::Namespace;
799 let (sig, reg) = ty::print::FmtPrinter::new(self.tcx, Namespace::TypeNS)
800 .name_all_regions(&sig, WrapBinderMode::ForAll)
801 .unwrap();
802 let lts: Vec<String> =
803 reg.into_items().map(|(_, kind)| kind.to_string()).into_sorted_stable_ord();
804 (if lts.is_empty() { String::new() } else { ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("for<{0}> ", lts.join(", ")))
})format!("for<{}> ", lts.join(", ")) }, sig)
805 };
806
807 let (lt1, sig1) = get_lifetimes(sig1);
808 let (lt2, sig2) = get_lifetimes(sig2);
809
810 let mut values =
812 (DiagStyledString::normal("".to_string()), DiagStyledString::normal("".to_string()));
813
814 let fn_item_prefix_and_safety = |fn_def, sig: ty::FnSig<'_>| match fn_def {
817 None => ("", sig.safety().prefix_str()),
818 Some((did, _)) => {
819 if self.tcx.codegen_fn_attrs(did).safe_target_features {
820 ("#[target_features] ", "")
821 } else {
822 ("", sig.safety().prefix_str())
823 }
824 }
825 };
826 let (prefix1, safety1) = fn_item_prefix_and_safety(fn_def1, sig1);
827 let (prefix2, safety2) = fn_item_prefix_and_safety(fn_def2, sig2);
828 values.0.push(prefix1, prefix1 != prefix2);
829 values.1.push(prefix2, prefix1 != prefix2);
830
831 let lifetime_diff = lt1 != lt2;
834 values.0.push(lt1, lifetime_diff);
835 values.1.push(lt2, lifetime_diff);
836
837 values.0.push(safety1, safety1 != safety2);
840 values.1.push(safety2, safety1 != safety2);
841
842 if sig1.abi() != ExternAbi::Rust {
845 values.0.push(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("extern {0} ", sig1.abi()))
})format!("extern {} ", sig1.abi()), sig1.abi() != sig2.abi());
846 }
847 if sig2.abi() != ExternAbi::Rust {
848 values.1.push(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("extern {0} ", sig2.abi()))
})format!("extern {} ", sig2.abi()), sig1.abi() != sig2.abi());
849 }
850
851 values.0.push_normal("fn(");
854 values.1.push_normal("fn(");
855
856 let len1 = sig1.inputs().len();
859 let len2 = sig2.inputs().len();
860 if len1 == len2 {
861 for (i, (l, r)) in iter::zip(sig1.inputs(), sig2.inputs()).enumerate() {
862 self.push_comma(&mut values.0, &mut values.1, i);
863 let (x1, x2) = self.cmp(*l, *r);
864 (values.0).0.extend(x1.0);
865 (values.1).0.extend(x2.0);
866 }
867 } else {
868 for (i, l) in sig1.inputs().iter().enumerate() {
869 values.0.push_highlighted(l.to_string());
870 if i != len1 - 1 {
871 values.0.push_highlighted(", ");
872 }
873 }
874 for (i, r) in sig2.inputs().iter().enumerate() {
875 values.1.push_highlighted(r.to_string());
876 if i != len2 - 1 {
877 values.1.push_highlighted(", ");
878 }
879 }
880 }
881
882 if sig1.c_variadic() {
883 if len1 > 0 {
884 values.0.push_normal(", ");
885 }
886 values.0.push("...", !sig2.c_variadic());
887 }
888 if sig2.c_variadic() {
889 if len2 > 0 {
890 values.1.push_normal(", ");
891 }
892 values.1.push("...", !sig1.c_variadic());
893 }
894
895 values.0.push_normal(")");
898 values.1.push_normal(")");
899
900 let output1 = sig1.output();
903 let output2 = sig2.output();
904 let (x1, x2) = self.cmp(output1, output2);
905 let output_diff = x1 != x2;
906 if !output1.is_unit() || output_diff {
907 values.0.push_normal(" -> ");
908 (values.0).0.extend(x1.0);
909 }
910 if !output2.is_unit() || output_diff {
911 values.1.push_normal(" -> ");
912 (values.1).0.extend(x2.0);
913 }
914
915 let fmt = |did, args| ::alloc::__export::must_use({
::alloc::fmt::format(format_args!(" {{{0}}}",
self.tcx.def_path_str_with_args(did, args)))
})format!(" {{{}}}", self.tcx.def_path_str_with_args(did, args));
916
917 match (fn_def1, fn_def2) {
918 (Some((fn_def1, Some(fn_args1))), Some((fn_def2, Some(fn_args2)))) => {
919 let path1 = fmt(fn_def1, fn_args1);
920 let path2 = fmt(fn_def2, fn_args2);
921 let same_path = path1 == path2;
922 values.0.push(path1, !same_path);
923 values.1.push(path2, !same_path);
924 }
925 (Some((fn_def1, Some(fn_args1))), None) => {
926 values.0.push_highlighted(fmt(fn_def1, fn_args1));
927 }
928 (None, Some((fn_def2, Some(fn_args2)))) => {
929 values.1.push_highlighted(fmt(fn_def2, fn_args2));
930 }
931 _ => {}
932 }
933
934 values
935 }
936
937 pub fn cmp_traits(
938 &self,
939 def_id1: DefId,
940 args1: &[ty::GenericArg<'tcx>],
941 def_id2: DefId,
942 args2: &[ty::GenericArg<'tcx>],
943 ) -> (DiagStyledString, DiagStyledString) {
944 let mut values = (DiagStyledString::new(), DiagStyledString::new());
945
946 if def_id1 != def_id2 {
947 values.0.push_highlighted(self.tcx.def_path_str(def_id1).as_str());
948 values.1.push_highlighted(self.tcx.def_path_str(def_id2).as_str());
949 } else {
950 values.0.push_normal(self.tcx.item_name(def_id1).as_str());
951 values.1.push_normal(self.tcx.item_name(def_id2).as_str());
952 }
953
954 if args1.len() != args2.len() {
955 let (pre, post) = if args1.len() > 0 { ("<", ">") } else { ("", "") };
956 values.0.push_normal(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{1}{0}{2}",
args1.iter().map(|a|
a.to_string()).collect::<Vec<_>>().join(", "), pre, post))
})format!(
957 "{pre}{}{post}",
958 args1.iter().map(|a| a.to_string()).collect::<Vec<_>>().join(", ")
959 ));
960 let (pre, post) = if args2.len() > 0 { ("<", ">") } else { ("", "") };
961 values.1.push_normal(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{1}{0}{2}",
args2.iter().map(|a|
a.to_string()).collect::<Vec<_>>().join(", "), pre, post))
})format!(
962 "{pre}{}{post}",
963 args2.iter().map(|a| a.to_string()).collect::<Vec<_>>().join(", ")
964 ));
965 return values;
966 }
967
968 if args1.len() > 0 {
969 values.0.push_normal("<");
970 values.1.push_normal("<");
971 }
972 for (i, (a, b)) in std::iter::zip(args1, args2).enumerate() {
973 let a_str = a.to_string();
974 let b_str = b.to_string();
975 if let (Some(a), Some(b)) = (a.as_type(), b.as_type()) {
976 let (a, b) = self.cmp(a, b);
977 values.0.0.extend(a.0);
978 values.1.0.extend(b.0);
979 } else if a_str != b_str {
980 values.0.push_highlighted(a_str);
981 values.1.push_highlighted(b_str);
982 } else {
983 values.0.push_normal(a_str);
984 values.1.push_normal(b_str);
985 }
986 if i + 1 < args1.len() {
987 values.0.push_normal(", ");
988 values.1.push_normal(", ");
989 }
990 }
991 if args1.len() > 0 {
992 values.0.push_normal(">");
993 values.1.push_normal(">");
994 }
995 values
996 }
997
998 pub fn cmp(&self, t1: Ty<'tcx>, t2: Ty<'tcx>) -> (DiagStyledString, DiagStyledString) {
1001 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs:1001",
"rustc_trait_selection::error_reporting::infer",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs"),
::tracing_core::__macro_support::Option::Some(1001u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::error_reporting::infer"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("cmp(t1={0}, t1.kind={1:?}, t2={2}, t2.kind={3:?})",
t1, t1.kind(), t2, t2.kind()) as &dyn Value))])
});
} else { ; }
};debug!("cmp(t1={}, t1.kind={:?}, t2={}, t2.kind={:?})", t1, t1.kind(), t2, t2.kind());
1002
1003 let recurse = |t1, t2, values: &mut (DiagStyledString, DiagStyledString)| {
1005 let (x1, x2) = self.cmp(t1, t2);
1006 (values.0).0.extend(x1.0);
1007 (values.1).0.extend(x2.0);
1008 };
1009
1010 fn fmt_region<'tcx>(region: ty::Region<'tcx>) -> String {
1011 let mut r = region.to_string();
1012 if r == "'_" {
1013 r.clear();
1014 } else {
1015 r.push(' ');
1016 }
1017 ::alloc::__export::must_use({ ::alloc::fmt::format(format_args!("&{0}", r)) })format!("&{r}")
1018 }
1019
1020 fn push_ref<'tcx>(
1021 region: ty::Region<'tcx>,
1022 mutbl: hir::Mutability,
1023 s: &mut DiagStyledString,
1024 ) {
1025 s.push_highlighted(fmt_region(region));
1026 s.push_highlighted(mutbl.prefix_str());
1027 }
1028
1029 fn maybe_highlight<T: Eq + ToString>(
1030 t1: T,
1031 t2: T,
1032 (buf1, buf2): &mut (DiagStyledString, DiagStyledString),
1033 tcx: TyCtxt<'_>,
1034 ) {
1035 let highlight = t1 != t2;
1036 let (t1, t2) = if highlight || tcx.sess.opts.verbose {
1037 (t1.to_string(), t2.to_string())
1038 } else {
1039 ("_".into(), "_".into())
1041 };
1042 buf1.push(t1, highlight);
1043 buf2.push(t2, highlight);
1044 }
1045
1046 fn cmp_ty_refs<'tcx>(
1047 r1: ty::Region<'tcx>,
1048 mut1: hir::Mutability,
1049 r2: ty::Region<'tcx>,
1050 mut2: hir::Mutability,
1051 ss: &mut (DiagStyledString, DiagStyledString),
1052 ) {
1053 let (r1, r2) = (fmt_region(r1), fmt_region(r2));
1054 if r1 != r2 {
1055 ss.0.push_highlighted(r1);
1056 ss.1.push_highlighted(r2);
1057 } else {
1058 ss.0.push_normal(r1);
1059 ss.1.push_normal(r2);
1060 }
1061
1062 if mut1 != mut2 {
1063 ss.0.push_highlighted(mut1.prefix_str());
1064 ss.1.push_highlighted(mut2.prefix_str());
1065 } else {
1066 ss.0.push_normal(mut1.prefix_str());
1067 ss.1.push_normal(mut2.prefix_str());
1068 }
1069 }
1070
1071 match (t1.kind(), t2.kind()) {
1073 (&ty::Adt(def1, sub1), &ty::Adt(def2, sub2)) => {
1074 let did1 = def1.did();
1075 let did2 = def2.did();
1076
1077 let generics1 = self.tcx.generics_of(did1);
1078 let generics2 = self.tcx.generics_of(did2);
1079
1080 let non_default_after_default = generics1
1081 .check_concrete_type_after_default(self.tcx, sub1)
1082 || generics2.check_concrete_type_after_default(self.tcx, sub2);
1083 let sub_no_defaults_1 = if non_default_after_default {
1084 generics1.own_args(sub1)
1085 } else {
1086 generics1.own_args_no_defaults(self.tcx, sub1)
1087 };
1088 let sub_no_defaults_2 = if non_default_after_default {
1089 generics2.own_args(sub2)
1090 } else {
1091 generics2.own_args_no_defaults(self.tcx, sub2)
1092 };
1093 let mut values = (DiagStyledString::new(), DiagStyledString::new());
1094 let path1 = self.tcx.def_path_str(did1);
1095 let path2 = self.tcx.def_path_str(did2);
1096 if did1 == did2 {
1097 values.0.push_normal(path1);
1106 values.1.push_normal(path2);
1107
1108 let len1 = sub_no_defaults_1.len();
1111 let len2 = sub_no_defaults_2.len();
1112 let common_len = cmp::min(len1, len2);
1113 let remainder1 = &sub1[common_len..];
1114 let remainder2 = &sub2[common_len..];
1115 let common_default_params =
1116 iter::zip(remainder1.iter().rev(), remainder2.iter().rev())
1117 .filter(|(a, b)| a == b)
1118 .count();
1119 let len = sub1.len() - common_default_params;
1120
1121 if len > 0 {
1123 values.0.push_normal("<");
1124 values.1.push_normal("<");
1125 }
1126
1127 fn lifetime_display(lifetime: Region<'_>) -> String {
1128 let s = lifetime.to_string();
1129 if s.is_empty() { "'_".to_string() } else { s }
1130 }
1131
1132 for (i, (arg1, arg2)) in sub1.iter().zip(sub2).enumerate().take(len) {
1133 self.push_comma(&mut values.0, &mut values.1, i);
1134 match arg1.kind() {
1135 ty::GenericArgKind::Lifetime(l1) => {
1152 let l1_str = lifetime_display(l1);
1153 let l2 = arg2.expect_region();
1154 let l2_str = lifetime_display(l2);
1155 if l1 != l2 {
1156 values.0.push_highlighted(l1_str);
1157 values.1.push_highlighted(l2_str);
1158 } else if l1.is_bound() || self.tcx.sess.opts.verbose {
1159 values.0.push_normal(l1_str);
1160 values.1.push_normal(l2_str);
1161 } else {
1162 values.0.push_normal("'_");
1163 values.1.push_normal("'_");
1164 }
1165 }
1166 ty::GenericArgKind::Type(ta1) => {
1167 let ta2 = arg2.expect_ty();
1168 if ta1 == ta2 && !self.tcx.sess.opts.verbose {
1169 values.0.push_normal("_");
1170 values.1.push_normal("_");
1171 } else {
1172 recurse(ta1, ta2, &mut values);
1173 }
1174 }
1175 ty::GenericArgKind::Const(ca1) => {
1185 let ca2 = arg2.expect_const();
1186 maybe_highlight(ca1, ca2, &mut values, self.tcx);
1187 }
1188 }
1189 }
1190
1191 if len > 0 {
1194 values.0.push_normal(">");
1195 values.1.push_normal(">");
1196 }
1197 values
1198 } else {
1199 if self.cmp_type_arg(
1205 &mut values.0,
1206 &mut values.1,
1207 path1.clone(),
1208 sub_no_defaults_1,
1209 path2.clone(),
1210 t2,
1211 ) {
1212 return values;
1213 }
1214 if self.cmp_type_arg(
1220 &mut values.1,
1221 &mut values.0,
1222 path2,
1223 sub_no_defaults_2,
1224 path1,
1225 t1,
1226 ) {
1227 return values;
1228 }
1229
1230 let t1_str = t1.to_string();
1237 let t2_str = t2.to_string();
1238 let min_len = t1_str.len().min(t2_str.len());
1239
1240 const SEPARATOR: &str = "::";
1241 let separator_len = SEPARATOR.len();
1242 let split_idx: usize =
1243 iter::zip(t1_str.split(SEPARATOR), t2_str.split(SEPARATOR))
1244 .take_while(|(mod1_str, mod2_str)| mod1_str == mod2_str)
1245 .map(|(mod_str, _)| mod_str.len() + separator_len)
1246 .sum();
1247
1248 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs:1248",
"rustc_trait_selection::error_reporting::infer",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs"),
::tracing_core::__macro_support::Option::Some(1248u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::error_reporting::infer"),
::tracing_core::field::FieldSet::new(&["message",
"separator_len", "split_idx", "min_len"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("cmp")
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&separator_len)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&split_idx)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&min_len) as
&dyn Value))])
});
} else { ; }
};debug!(?separator_len, ?split_idx, ?min_len, "cmp");
1249
1250 if split_idx >= min_len {
1251 (
1253 DiagStyledString::highlighted(t1_str),
1254 DiagStyledString::highlighted(t2_str),
1255 )
1256 } else {
1257 let (common, uniq1) = t1_str.split_at(split_idx);
1258 let (_, uniq2) = t2_str.split_at(split_idx);
1259 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs:1259",
"rustc_trait_selection::error_reporting::infer",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs"),
::tracing_core::__macro_support::Option::Some(1259u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::error_reporting::infer"),
::tracing_core::field::FieldSet::new(&["message", "common",
"uniq1", "uniq2"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("cmp")
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&common) as
&dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&uniq1) as
&dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&uniq2) as
&dyn Value))])
});
} else { ; }
};debug!(?common, ?uniq1, ?uniq2, "cmp");
1260
1261 values.0.push_normal(common);
1262 values.0.push_highlighted(uniq1);
1263 values.1.push_normal(common);
1264 values.1.push_highlighted(uniq2);
1265
1266 values
1267 }
1268 }
1269 }
1270
1271 (&ty::Ref(r1, ref_ty1, mutbl1), &ty::Ref(r2, ref_ty2, mutbl2)) => {
1273 let mut values = (DiagStyledString::new(), DiagStyledString::new());
1274 cmp_ty_refs(r1, mutbl1, r2, mutbl2, &mut values);
1275 recurse(ref_ty1, ref_ty2, &mut values);
1276 values
1277 }
1278 (&ty::Ref(r1, ref_ty1, mutbl1), _) => {
1280 let mut values = (DiagStyledString::new(), DiagStyledString::new());
1281 push_ref(r1, mutbl1, &mut values.0);
1282 recurse(ref_ty1, t2, &mut values);
1283 values
1284 }
1285 (_, &ty::Ref(r2, ref_ty2, mutbl2)) => {
1286 let mut values = (DiagStyledString::new(), DiagStyledString::new());
1287 push_ref(r2, mutbl2, &mut values.1);
1288 recurse(t1, ref_ty2, &mut values);
1289 values
1290 }
1291
1292 (&ty::Tuple(args1), &ty::Tuple(args2)) if args1.len() == args2.len() => {
1294 let mut values = (DiagStyledString::normal("("), DiagStyledString::normal("("));
1295 let len = args1.len();
1296 for (i, (left, right)) in args1.iter().zip(args2).enumerate() {
1297 self.push_comma(&mut values.0, &mut values.1, i);
1298 recurse(left, right, &mut values);
1299 }
1300 if len == 1 {
1301 values.0.push_normal(",");
1303 values.1.push_normal(",");
1304 }
1305 values.0.push_normal(")");
1306 values.1.push_normal(")");
1307 values
1308 }
1309
1310 (ty::FnDef(did1, args1), ty::FnDef(did2, args2)) => {
1311 let sig1 = self.tcx.fn_sig(*did1).instantiate(self.tcx, args1).skip_norm_wip();
1312 let sig2 = self.tcx.fn_sig(*did2).instantiate(self.tcx, args2).skip_norm_wip();
1313 self.cmp_fn_sig(sig1, Some((*did1, Some(args1))), sig2, Some((*did2, Some(args2))))
1314 }
1315
1316 (ty::FnDef(did1, args1), ty::FnPtr(sig_tys2, hdr2)) => {
1317 let sig1 = self.tcx.fn_sig(*did1).instantiate(self.tcx, args1).skip_norm_wip();
1318 self.cmp_fn_sig(sig1, Some((*did1, Some(args1))), sig_tys2.with(*hdr2), None)
1319 }
1320
1321 (ty::FnPtr(sig_tys1, hdr1), ty::FnDef(did2, args2)) => {
1322 let sig2 = self.tcx.fn_sig(*did2).instantiate(self.tcx, args2).skip_norm_wip();
1323 self.cmp_fn_sig(sig_tys1.with(*hdr1), None, sig2, Some((*did2, Some(args2))))
1324 }
1325
1326 (ty::FnPtr(sig_tys1, hdr1), ty::FnPtr(sig_tys2, hdr2)) => {
1327 self.cmp_fn_sig(sig_tys1.with(*hdr1), None, sig_tys2.with(*hdr2), None)
1328 }
1329
1330 _ => {
1331 let mut strs = (DiagStyledString::new(), DiagStyledString::new());
1332 maybe_highlight(t1, t2, &mut strs, self.tcx);
1333 strs
1334 }
1335 }
1336 }
1337
1338 #[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("note_type_err",
"rustc_trait_selection::error_reporting::infer",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs"),
::tracing_core::__macro_support::Option::Some(1347u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::error_reporting::infer"),
::tracing_core::field::FieldSet::new(&["cause", "values",
"terr", "override_span"],
::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(&values)
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(&terr)
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(&override_span)
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: () = loop {};
return __tracing_attr_fake_return;
}
{
let span = override_span.unwrap_or(cause.span);
if let TypeError::CyclicTy(_) = terr { values = None; }
struct OpaqueTypesVisitor<'tcx> {
types: FxIndexMap<TyCategory, FxIndexSet<Span>>,
expected: FxIndexMap<TyCategory, FxIndexSet<Span>>,
found: FxIndexMap<TyCategory, FxIndexSet<Span>>,
ignore_span: Span,
tcx: TyCtxt<'tcx>,
}
impl<'tcx> OpaqueTypesVisitor<'tcx> {
fn visit_expected_found(tcx: TyCtxt<'tcx>,
expected: impl TypeVisitable<TyCtxt<'tcx>>,
found: impl TypeVisitable<TyCtxt<'tcx>>, ignore_span: Span)
-> Self {
let mut types_visitor =
OpaqueTypesVisitor {
types: Default::default(),
expected: Default::default(),
found: Default::default(),
ignore_span,
tcx,
};
expected.visit_with(&mut types_visitor);
std::mem::swap(&mut types_visitor.expected,
&mut types_visitor.types);
found.visit_with(&mut types_visitor);
std::mem::swap(&mut types_visitor.found,
&mut types_visitor.types);
types_visitor
}
fn report(&self, err: &mut Diag<'_>) {
self.add_labels_for_types(err, "expected", &self.expected);
self.add_labels_for_types(err, "found", &self.found);
}
fn add_labels_for_types(&self, err: &mut Diag<'_>,
target: &str,
types: &FxIndexMap<TyCategory, FxIndexSet<Span>>) {
for (kind, values) in types.iter() {
let count = values.len();
for &sp in values {
err.span_label(sp,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}{1} {2:#}{3}",
if count == 1 { "the " } else { "one of the " }, target,
kind, if count == 1 { "" } else { "s" }))
}));
}
}
}
}
impl<'tcx> ty::TypeVisitor<TyCtxt<'tcx>> for
OpaqueTypesVisitor<'tcx> {
fn visit_ty(&mut self, t: Ty<'tcx>) {
if let Some((kind, def_id)) =
TyCategory::from_ty(self.tcx, t) {
let span = self.tcx.def_span(def_id);
if !self.ignore_span.overlaps(span) &&
!span.is_desugaring(DesugaringKind::Async) {
self.types.entry(kind).or_default().insert(span);
}
}
t.super_visit_with(self)
}
}
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs:1458",
"rustc_trait_selection::error_reporting::infer",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs"),
::tracing_core::__macro_support::Option::Some(1458u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::error_reporting::infer"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("note_type_err(diag={0:?})",
diag) as &dyn Value))])
});
} else { ; }
};
enum Mismatch<'a> {
Variable(ty::error::ExpectedFound<Ty<'a>>),
Fixed(&'static str),
}
let (expected_found, exp_found, is_simple_error, values,
param_env) =
match values {
None => (None, Mismatch::Fixed("type"), false, None, None),
Some(ty::ParamEnvAnd { param_env, value: values }) => {
let mut values = self.resolve_vars_if_possible(values);
if self.next_trait_solver() {
values =
deeply_normalize_for_diagnostics(self, param_env, values);
}
let (is_simple_error, exp_found) =
match values {
ValuePairs::Terms(ExpectedFound { expected, found }) => {
match (expected.kind(), found.kind()) {
(ty::TermKind::Ty(expected), ty::TermKind::Ty(found)) => {
let is_simple_err =
expected.is_simple_text() && found.is_simple_text();
OpaqueTypesVisitor::visit_expected_found(self.tcx, expected,
found, span).report(diag);
(is_simple_err,
Mismatch::Variable(ExpectedFound { expected, found }))
}
(ty::TermKind::Const(_), ty::TermKind::Const(_)) => {
(false, Mismatch::Fixed("constant"))
}
_ => (false, Mismatch::Fixed("type")),
}
}
ValuePairs::PolySigs(ExpectedFound { expected, found }) => {
OpaqueTypesVisitor::visit_expected_found(self.tcx, expected,
found, span).report(diag);
(false, Mismatch::Fixed("signature"))
}
ValuePairs::TraitRefs(_) =>
(false, Mismatch::Fixed("trait")),
ValuePairs::Aliases(ExpectedFound { expected, .. }) => {
let def_id =
match expected.kind {
ty::AliasTermKind::ProjectionTy { def_id } => def_id.into(),
ty::AliasTermKind::InherentTy { def_id } => def_id.into(),
ty::AliasTermKind::OpaqueTy { def_id } => def_id.into(),
ty::AliasTermKind::FreeTy { def_id } => def_id.into(),
ty::AliasTermKind::AnonConst { def_id } => def_id.into(),
ty::AliasTermKind::ProjectionConst { def_id } =>
def_id.into(),
ty::AliasTermKind::FreeConst { def_id } => def_id.into(),
ty::AliasTermKind::InherentConst { def_id } =>
def_id.into(),
};
(false, Mismatch::Fixed(self.tcx.def_descr(def_id)))
}
ValuePairs::Regions(_) =>
(false, Mismatch::Fixed("lifetime")),
ValuePairs::ExistentialTraitRef(_) => {
(false, Mismatch::Fixed("existential trait ref"))
}
ValuePairs::ExistentialProjection(_) => {
(false, Mismatch::Fixed("existential projection"))
}
};
let Some(vals) =
self.values_str(values, cause,
diag.long_ty_path()) else {
diag.downgrade_to_delayed_bug();
return;
};
(Some(vals), exp_found, is_simple_error, Some(values),
Some(param_env))
}
};
let mut label_or_note =
|span: Span, msg: Cow<'static, str>|
{
if (prefer_label && is_simple_error) ||
&[span] == diag.span.primary_spans() {
diag.span_label(span, msg);
} else { diag.span_note(span, msg); }
};
if let Some((secondary_span, secondary_msg,
swap_secondary_and_primary)) = secondary_span {
if swap_secondary_and_primary {
let terr =
if let Some(infer::ValuePairs::Terms(ExpectedFound {
expected, .. })) = values {
Cow::from(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("expected this to be `{0}`",
expected))
}))
} else { terr.to_string(self.tcx) };
label_or_note(secondary_span, terr);
label_or_note(span, secondary_msg);
} else {
label_or_note(span, terr.to_string(self.tcx));
label_or_note(secondary_span, secondary_msg);
}
} else if let Some(values) = values &&
let Some((e, f)) = values.ty() &&
let TypeError::ArgumentSorts(..) | TypeError::Sorts(_) =
terr {
let e = self.tcx.erase_and_anonymize_regions(e);
let f = self.tcx.erase_and_anonymize_regions(f);
let expected =
{
let _guard = ForceTrimmedGuard::new();
e.sort_string(self.tcx)
};
let found =
{
let _guard = ForceTrimmedGuard::new();
f.sort_string(self.tcx)
};
if expected == found {
label_or_note(span, terr.to_string(self.tcx));
} else {
label_or_note(span,
Cow::from(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("expected {0}, found {1}",
expected, found))
})));
}
} else { label_or_note(span, terr.to_string(self.tcx)); }
if let Some(param_env) = param_env {
self.note_field_shadowed_by_private_candidate_in_cause(diag,
cause, param_env);
}
if self.check_and_note_conflicting_crates(diag, terr) { return; }
if let Some((expected, found)) = expected_found {
let (expected_label, found_label, exp_found) =
match exp_found {
Mismatch::Variable(ef) =>
(ef.expected.prefix_string(self.tcx),
ef.found.prefix_string(self.tcx), Some(ef)),
Mismatch::Fixed(s) => (s.into(), s.into(), None),
};
enum Similar<'tcx> {
Adts {
expected: ty::AdtDef<'tcx>,
found: ty::AdtDef<'tcx>,
},
PrimitiveFound {
expected: ty::AdtDef<'tcx>,
found: Ty<'tcx>,
},
PrimitiveExpected {
expected: Ty<'tcx>,
found: ty::AdtDef<'tcx>,
},
}
let similarity =
|ExpectedFound { expected, found }: ExpectedFound<Ty<'tcx>>|
{
if let ty::Adt(expected, _) = expected.kind() &&
let Some(primitive) = found.primitive_symbol() {
let path = self.tcx.def_path(expected.did()).data;
let name = path.last().unwrap().data.get_opt_name();
if name == Some(primitive) {
return Some(Similar::PrimitiveFound {
expected: *expected,
found,
});
}
} else if let Some(primitive) = expected.primitive_symbol()
&& let ty::Adt(found, _) = found.kind() {
let path = self.tcx.def_path(found.did()).data;
let name = path.last().unwrap().data.get_opt_name();
if name == Some(primitive) {
return Some(Similar::PrimitiveExpected {
expected,
found: *found,
});
}
} else if let ty::Adt(expected, _) = expected.kind() &&
let ty::Adt(found, _) = found.kind() {
if !expected.did().is_local() &&
expected.did().krate == found.did().krate {
return None;
}
let f_path = self.tcx.def_path(found.did()).data;
let e_path = self.tcx.def_path(expected.did()).data;
if let (Some(e_last), Some(f_last)) =
(e_path.last(), f_path.last()) && e_last == f_last {
return Some(Similar::Adts {
expected: *expected,
found: *found,
});
}
}
None
};
match terr {
TypeError::Sorts(values) if let Some(s) = similarity(values)
=> {
let diagnose_primitive =
|prim: Ty<'tcx>, shadow: Ty<'tcx>, defid: DefId,
diag: &mut Diag<'_>|
{
let name = shadow.sort_string(self.tcx);
diag.note(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` and {1} have similar names, but are actually distinct types",
prim, name))
}));
diag.note(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("one `{0}` is a primitive defined by the language",
prim))
}));
let def_span = self.tcx.def_span(defid);
let msg =
if defid.is_local() {
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("the other {0} is defined in the current crate",
name))
})
} else {
let crate_name = self.tcx.crate_name(defid.krate);
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("the other {0} is defined in crate `{1}`",
name, crate_name))
})
};
diag.span_note(def_span, msg);
};
let diagnose_adts =
|expected_adt: ty::AdtDef<'tcx>,
found_adt: ty::AdtDef<'tcx>, diag: &mut Diag<'_>|
{
let found_name = values.found.sort_string(self.tcx);
let expected_name = values.expected.sort_string(self.tcx);
let found_defid = found_adt.did();
let expected_defid = expected_adt.did();
diag.note(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} and {1} have similar names, but are actually distinct types",
found_name, expected_name))
}));
for (defid, name) in
[(found_defid, found_name), (expected_defid, expected_name)]
{
let def_span = self.tcx.def_span(defid);
let msg =
if found_defid.is_local() && expected_defid.is_local() {
let module =
self.tcx.parent_module_from_def_id(defid.expect_local()).to_def_id();
let module_name =
self.tcx.def_path(module).to_string_no_crate_verbose();
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} is defined in module `crate{1}` of the current crate",
name, module_name))
})
} else if defid.is_local() {
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} is defined in the current crate",
name))
})
} else {
let crate_name = self.tcx.crate_name(defid.krate);
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} is defined in crate `{1}`",
name, crate_name))
})
};
diag.span_note(def_span, msg);
}
};
match s {
Similar::Adts { expected, found } =>
diagnose_adts(expected, found, diag),
Similar::PrimitiveFound { expected, found: prim } => {
diagnose_primitive(prim, values.expected, expected.did(),
diag)
}
Similar::PrimitiveExpected { expected: prim, found } => {
diagnose_primitive(prim, values.found, found.did(), diag)
}
}
}
TypeError::Sorts(values) => {
let extra =
expected == found &&
values.expected.sort_string(self.tcx) !=
values.found.sort_string(self.tcx);
let sort_string =
|ty: Ty<'tcx>|
match (extra, ty.kind()) {
(true,
ty::Alias(ty::AliasTy { kind: ty::Opaque { def_id }, .. }))
=> {
let sm = self.tcx.sess.source_map();
let pos =
sm.lookup_char_pos(self.tcx.def_span(*def_id).lo());
DiagStyledString::normal(::alloc::__export::must_use({
::alloc::fmt::format(format_args!(" (opaque type at <{0}:{1}:{2}>)",
sm.filename_for_diagnostics(&pos.file.name), pos.line,
pos.col.to_usize() + 1))
}))
}
(true,
&ty::Alias(ty::AliasTy { kind: ty::Projection { def_id }, ..
})) if self.tcx.is_impl_trait_in_trait(def_id) => {
let sm = self.tcx.sess.source_map();
let pos =
sm.lookup_char_pos(self.tcx.def_span(def_id).lo());
DiagStyledString::normal(::alloc::__export::must_use({
::alloc::fmt::format(format_args!(" (trait associated opaque type at <{0}:{1}:{2}>)",
sm.filename_for_diagnostics(&pos.file.name), pos.line,
pos.col.to_usize() + 1))
}))
}
(true, _) => {
let mut s = DiagStyledString::normal(" (");
s.push_highlighted(ty.sort_string(self.tcx));
s.push_normal(")");
s
}
(false, _) => DiagStyledString::normal(""),
};
if !(values.expected.is_simple_text() &&
values.found.is_simple_text()) ||
(exp_found.is_some_and(|ef|
{
if !ef.expected.is_ty_or_numeric_infer() {
ef.expected != values.expected
} else if !ef.found.is_ty_or_numeric_infer() {
ef.found != values.found
} else { false }
})) {
if let Some(ExpectedFound { found: found_ty, .. }) =
exp_found && !self.tcx.ty_is_opaque_future(found_ty) {
diag.note_expected_found_extra(&expected_label, expected,
&found_label, found, sort_string(values.expected),
sort_string(values.found));
}
}
}
_ => {
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs:1774",
"rustc_trait_selection::error_reporting::infer",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs"),
::tracing_core::__macro_support::Option::Some(1774u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::error_reporting::infer"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("note_type_err: exp_found={0:?}, expected={1:?} found={2:?}",
exp_found, expected, found) as &dyn Value))])
});
} else { ; }
};
if !is_simple_error || terr.must_include_note() {
diag.note_expected_found(&expected_label, expected,
&found_label, found);
if let Some(ty::Closure(_, args)) =
exp_found.map(|expected_type_found|
expected_type_found.found.kind()) {
diag.highlighted_note(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[StringPart::normal("closure has signature: `"),
StringPart::highlighted(self.tcx.signature_unclosure(args.as_closure().sig(),
rustc_hir::Safety::Safe).to_string()),
StringPart::normal("`")])));
}
}
}
}
}
let exp_found =
match exp_found {
Mismatch::Variable(exp_found) => Some(exp_found),
Mismatch::Fixed(_) => None,
};
let exp_found =
match terr {
ty::error::TypeError::Sorts(terr) if
exp_found.is_some_and(|ef| terr.found == ef.found) => {
Some(terr)
}
_ => exp_found,
};
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs:1814",
"rustc_trait_selection::error_reporting::infer",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs"),
::tracing_core::__macro_support::Option::Some(1814u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::error_reporting::infer"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("exp_found {0:?} terr {1:?} cause.code {2:?}",
exp_found, terr, cause.code()) as &dyn Value))])
});
} else { ; }
};
if let Some(exp_found) = exp_found {
let should_suggest_fixes =
if let ObligationCauseCode::Pattern { root_ty, .. } =
cause.code() {
self.same_type_modulo_infer(*root_ty, exp_found.expected)
} else { true };
if should_suggest_fixes &&
!#[allow(non_exhaustive_omitted_patterns)] match terr {
TypeError::RegionsInsufficientlyPolymorphic(..) => true,
_ => false,
} {
self.suggest_tuple_pattern(cause, &exp_found, diag);
self.suggest_accessing_field_where_appropriate(cause,
&exp_found, diag);
self.suggest_await_on_expect_found(cause, span, &exp_found,
diag);
self.suggest_function_pointers(cause, span, &exp_found,
terr, diag);
self.suggest_turning_stmt_into_expr(cause, &exp_found,
diag);
}
}
let body_owner_def_id =
(cause.body_id !=
CRATE_DEF_ID).then(|| cause.body_id.to_def_id());
self.note_and_explain_type_err(diag, terr, cause, span,
body_owner_def_id);
if let Some(exp_found) = exp_found &&
let exp_found = TypeError::Sorts(exp_found) &&
exp_found != terr {
self.note_and_explain_type_err(diag, exp_found, cause, span,
body_owner_def_id);
}
if let Some(ValuePairs::TraitRefs(exp_found)) = values &&
let ty::Closure(def_id, _) =
exp_found.expected.self_ty().kind() &&
let Some(def_id) = def_id.as_local() &&
terr.involves_regions() {
let span = self.tcx.def_span(def_id);
diag.span_note(span,
"this closure does not fulfill the lifetime requirements");
self.suggest_for_all_lifetime_closure(span,
self.tcx.hir_node_by_def_id(def_id), &exp_found, diag);
}
self.note_error_origin(diag, cause, exp_found, terr, param_env);
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs:1867",
"rustc_trait_selection::error_reporting::infer",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs"),
::tracing_core::__macro_support::Option::Some(1867u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::error_reporting::infer"),
::tracing_core::field::FieldSet::new(&["diag"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&diag) as
&dyn Value))])
});
} else { ; }
};
}
}
}#[instrument(level = "debug", skip(self, diag, secondary_span, prefer_label))]
1348 pub fn note_type_err(
1349 &self,
1350 diag: &mut Diag<'_>,
1351 cause: &ObligationCause<'tcx>,
1352 secondary_span: Option<(Span, Cow<'static, str>, bool)>,
1353 mut values: Option<ty::ParamEnvAnd<'tcx, ValuePairs<'tcx>>>,
1354 terr: TypeError<'tcx>,
1355 prefer_label: bool,
1356 override_span: Option<Span>,
1357 ) {
1358 let span = override_span.unwrap_or(cause.span);
1363 if let TypeError::CyclicTy(_) = terr {
1366 values = None;
1367 }
1368 struct OpaqueTypesVisitor<'tcx> {
1369 types: FxIndexMap<TyCategory, FxIndexSet<Span>>,
1370 expected: FxIndexMap<TyCategory, FxIndexSet<Span>>,
1371 found: FxIndexMap<TyCategory, FxIndexSet<Span>>,
1372 ignore_span: Span,
1373 tcx: TyCtxt<'tcx>,
1374 }
1375
1376 impl<'tcx> OpaqueTypesVisitor<'tcx> {
1377 fn visit_expected_found(
1378 tcx: TyCtxt<'tcx>,
1379 expected: impl TypeVisitable<TyCtxt<'tcx>>,
1380 found: impl TypeVisitable<TyCtxt<'tcx>>,
1381 ignore_span: Span,
1382 ) -> Self {
1383 let mut types_visitor = OpaqueTypesVisitor {
1384 types: Default::default(),
1385 expected: Default::default(),
1386 found: Default::default(),
1387 ignore_span,
1388 tcx,
1389 };
1390 expected.visit_with(&mut types_visitor);
1394 std::mem::swap(&mut types_visitor.expected, &mut types_visitor.types);
1395 found.visit_with(&mut types_visitor);
1396 std::mem::swap(&mut types_visitor.found, &mut types_visitor.types);
1397 types_visitor
1398 }
1399
1400 fn report(&self, err: &mut Diag<'_>) {
1401 self.add_labels_for_types(err, "expected", &self.expected);
1402 self.add_labels_for_types(err, "found", &self.found);
1403 }
1404
1405 fn add_labels_for_types(
1406 &self,
1407 err: &mut Diag<'_>,
1408 target: &str,
1409 types: &FxIndexMap<TyCategory, FxIndexSet<Span>>,
1410 ) {
1411 for (kind, values) in types.iter() {
1412 let count = values.len();
1413 for &sp in values {
1414 err.span_label(
1415 sp,
1416 format!(
1417 "{}{} {:#}{}",
1418 if count == 1 { "the " } else { "one of the " },
1419 target,
1420 kind,
1421 pluralize!(count),
1422 ),
1423 );
1424 }
1425 }
1426 }
1427 }
1428
1429 impl<'tcx> ty::TypeVisitor<TyCtxt<'tcx>> for OpaqueTypesVisitor<'tcx> {
1430 fn visit_ty(&mut self, t: Ty<'tcx>) {
1431 if let Some((kind, def_id)) = TyCategory::from_ty(self.tcx, t) {
1432 let span = self.tcx.def_span(def_id);
1433 if !self.ignore_span.overlaps(span)
1449 && !span.is_desugaring(DesugaringKind::Async)
1450 {
1451 self.types.entry(kind).or_default().insert(span);
1452 }
1453 }
1454 t.super_visit_with(self)
1455 }
1456 }
1457
1458 debug!("note_type_err(diag={:?})", diag);
1459 enum Mismatch<'a> {
1460 Variable(ty::error::ExpectedFound<Ty<'a>>),
1461 Fixed(&'static str),
1462 }
1463 let (expected_found, exp_found, is_simple_error, values, param_env) = match values {
1464 None => (None, Mismatch::Fixed("type"), false, None, None),
1465 Some(ty::ParamEnvAnd { param_env, value: values }) => {
1466 let mut values = self.resolve_vars_if_possible(values);
1467 if self.next_trait_solver() {
1468 values = deeply_normalize_for_diagnostics(self, param_env, values);
1469 }
1470 let (is_simple_error, exp_found) = match values {
1471 ValuePairs::Terms(ExpectedFound { expected, found }) => {
1472 match (expected.kind(), found.kind()) {
1473 (ty::TermKind::Ty(expected), ty::TermKind::Ty(found)) => {
1474 let is_simple_err =
1475 expected.is_simple_text() && found.is_simple_text();
1476 OpaqueTypesVisitor::visit_expected_found(
1477 self.tcx, expected, found, span,
1478 )
1479 .report(diag);
1480
1481 (
1482 is_simple_err,
1483 Mismatch::Variable(ExpectedFound { expected, found }),
1484 )
1485 }
1486 (ty::TermKind::Const(_), ty::TermKind::Const(_)) => {
1487 (false, Mismatch::Fixed("constant"))
1488 }
1489 _ => (false, Mismatch::Fixed("type")),
1490 }
1491 }
1492 ValuePairs::PolySigs(ExpectedFound { expected, found }) => {
1493 OpaqueTypesVisitor::visit_expected_found(self.tcx, expected, found, span)
1494 .report(diag);
1495 (false, Mismatch::Fixed("signature"))
1496 }
1497 ValuePairs::TraitRefs(_) => (false, Mismatch::Fixed("trait")),
1498 ValuePairs::Aliases(ExpectedFound { expected, .. }) => {
1499 let def_id = match expected.kind {
1500 ty::AliasTermKind::ProjectionTy { def_id } => def_id.into(),
1501 ty::AliasTermKind::InherentTy { def_id } => def_id.into(),
1502 ty::AliasTermKind::OpaqueTy { def_id } => def_id.into(),
1503 ty::AliasTermKind::FreeTy { def_id } => def_id.into(),
1504 ty::AliasTermKind::AnonConst { def_id } => def_id.into(),
1505 ty::AliasTermKind::ProjectionConst { def_id } => def_id.into(),
1506 ty::AliasTermKind::FreeConst { def_id } => def_id.into(),
1507 ty::AliasTermKind::InherentConst { def_id } => def_id.into(),
1508 };
1509 (false, Mismatch::Fixed(self.tcx.def_descr(def_id)))
1510 }
1511 ValuePairs::Regions(_) => (false, Mismatch::Fixed("lifetime")),
1512 ValuePairs::ExistentialTraitRef(_) => {
1513 (false, Mismatch::Fixed("existential trait ref"))
1514 }
1515 ValuePairs::ExistentialProjection(_) => {
1516 (false, Mismatch::Fixed("existential projection"))
1517 }
1518 };
1519 let Some(vals) = self.values_str(values, cause, diag.long_ty_path()) else {
1520 diag.downgrade_to_delayed_bug();
1524 return;
1525 };
1526 (Some(vals), exp_found, is_simple_error, Some(values), Some(param_env))
1527 }
1528 };
1529
1530 let mut label_or_note = |span: Span, msg: Cow<'static, str>| {
1531 if (prefer_label && is_simple_error) || &[span] == diag.span.primary_spans() {
1532 diag.span_label(span, msg);
1533 } else {
1534 diag.span_note(span, msg);
1535 }
1536 };
1537 if let Some((secondary_span, secondary_msg, swap_secondary_and_primary)) = secondary_span {
1538 if swap_secondary_and_primary {
1539 let terr = if let Some(infer::ValuePairs::Terms(ExpectedFound {
1540 expected, ..
1541 })) = values
1542 {
1543 Cow::from(format!("expected this to be `{expected}`"))
1544 } else {
1545 terr.to_string(self.tcx)
1546 };
1547 label_or_note(secondary_span, terr);
1548 label_or_note(span, secondary_msg);
1549 } else {
1550 label_or_note(span, terr.to_string(self.tcx));
1551 label_or_note(secondary_span, secondary_msg);
1552 }
1553 } else if let Some(values) = values
1554 && let Some((e, f)) = values.ty()
1555 && let TypeError::ArgumentSorts(..) | TypeError::Sorts(_) = terr
1556 {
1557 let e = self.tcx.erase_and_anonymize_regions(e);
1558 let f = self.tcx.erase_and_anonymize_regions(f);
1559 let expected = with_forced_trimmed_paths!(e.sort_string(self.tcx));
1560 let found = with_forced_trimmed_paths!(f.sort_string(self.tcx));
1561 if expected == found {
1562 label_or_note(span, terr.to_string(self.tcx));
1563 } else {
1564 label_or_note(span, Cow::from(format!("expected {expected}, found {found}")));
1565 }
1566 } else {
1567 label_or_note(span, terr.to_string(self.tcx));
1568 }
1569
1570 if let Some(param_env) = param_env {
1571 self.note_field_shadowed_by_private_candidate_in_cause(diag, cause, param_env);
1572 }
1573
1574 if self.check_and_note_conflicting_crates(diag, terr) {
1575 return;
1576 }
1577
1578 if let Some((expected, found)) = expected_found {
1579 let (expected_label, found_label, exp_found) = match exp_found {
1580 Mismatch::Variable(ef) => (
1581 ef.expected.prefix_string(self.tcx),
1582 ef.found.prefix_string(self.tcx),
1583 Some(ef),
1584 ),
1585 Mismatch::Fixed(s) => (s.into(), s.into(), None),
1586 };
1587
1588 enum Similar<'tcx> {
1589 Adts { expected: ty::AdtDef<'tcx>, found: ty::AdtDef<'tcx> },
1590 PrimitiveFound { expected: ty::AdtDef<'tcx>, found: Ty<'tcx> },
1591 PrimitiveExpected { expected: Ty<'tcx>, found: ty::AdtDef<'tcx> },
1592 }
1593
1594 let similarity = |ExpectedFound { expected, found }: ExpectedFound<Ty<'tcx>>| {
1595 if let ty::Adt(expected, _) = expected.kind()
1596 && let Some(primitive) = found.primitive_symbol()
1597 {
1598 let path = self.tcx.def_path(expected.did()).data;
1599 let name = path.last().unwrap().data.get_opt_name();
1600 if name == Some(primitive) {
1601 return Some(Similar::PrimitiveFound { expected: *expected, found });
1602 }
1603 } else if let Some(primitive) = expected.primitive_symbol()
1604 && let ty::Adt(found, _) = found.kind()
1605 {
1606 let path = self.tcx.def_path(found.did()).data;
1607 let name = path.last().unwrap().data.get_opt_name();
1608 if name == Some(primitive) {
1609 return Some(Similar::PrimitiveExpected { expected, found: *found });
1610 }
1611 } else if let ty::Adt(expected, _) = expected.kind()
1612 && let ty::Adt(found, _) = found.kind()
1613 {
1614 if !expected.did().is_local() && expected.did().krate == found.did().krate {
1615 return None;
1619 }
1620 let f_path = self.tcx.def_path(found.did()).data;
1621 let e_path = self.tcx.def_path(expected.did()).data;
1622
1623 if let (Some(e_last), Some(f_last)) = (e_path.last(), f_path.last())
1624 && e_last == f_last
1625 {
1626 return Some(Similar::Adts { expected: *expected, found: *found });
1627 }
1628 }
1629 None
1630 };
1631
1632 match terr {
1633 TypeError::Sorts(values) if let Some(s) = similarity(values) => {
1635 let diagnose_primitive =
1636 |prim: Ty<'tcx>, shadow: Ty<'tcx>, defid: DefId, diag: &mut Diag<'_>| {
1637 let name = shadow.sort_string(self.tcx);
1638 diag.note(format!(
1639 "`{prim}` and {name} have similar names, but are actually distinct types"
1640 ));
1641 diag.note(format!(
1642 "one `{prim}` is a primitive defined by the language",
1643 ));
1644 let def_span = self.tcx.def_span(defid);
1645 let msg = if defid.is_local() {
1646 format!("the other {name} is defined in the current crate")
1647 } else {
1648 let crate_name = self.tcx.crate_name(defid.krate);
1649 format!("the other {name} is defined in crate `{crate_name}`")
1650 };
1651 diag.span_note(def_span, msg);
1652 };
1653
1654 let diagnose_adts =
1655 |expected_adt: ty::AdtDef<'tcx>,
1656 found_adt: ty::AdtDef<'tcx>,
1657 diag: &mut Diag<'_>| {
1658 let found_name = values.found.sort_string(self.tcx);
1659 let expected_name = values.expected.sort_string(self.tcx);
1660
1661 let found_defid = found_adt.did();
1662 let expected_defid = expected_adt.did();
1663
1664 diag.note(format!("{found_name} and {expected_name} have similar names, but are actually distinct types"));
1665 for (defid, name) in
1666 [(found_defid, found_name), (expected_defid, expected_name)]
1667 {
1668 let def_span = self.tcx.def_span(defid);
1669
1670 let msg = if found_defid.is_local() && expected_defid.is_local() {
1671 let module = self
1672 .tcx
1673 .parent_module_from_def_id(defid.expect_local())
1674 .to_def_id();
1675 let module_name =
1676 self.tcx.def_path(module).to_string_no_crate_verbose();
1677 format!(
1678 "{name} is defined in module `crate{module_name}` of the current crate"
1679 )
1680 } else if defid.is_local() {
1681 format!("{name} is defined in the current crate")
1682 } else {
1683 let crate_name = self.tcx.crate_name(defid.krate);
1684 format!("{name} is defined in crate `{crate_name}`")
1685 };
1686 diag.span_note(def_span, msg);
1687 }
1688 };
1689
1690 match s {
1691 Similar::Adts { expected, found } => diagnose_adts(expected, found, diag),
1692 Similar::PrimitiveFound { expected, found: prim } => {
1693 diagnose_primitive(prim, values.expected, expected.did(), diag)
1694 }
1695 Similar::PrimitiveExpected { expected: prim, found } => {
1696 diagnose_primitive(prim, values.found, found.did(), diag)
1697 }
1698 }
1699 }
1700 TypeError::Sorts(values) => {
1701 let extra = expected == found
1702 && values.expected.sort_string(self.tcx)
1706 != values.found.sort_string(self.tcx);
1707 let sort_string = |ty: Ty<'tcx>| match (extra, ty.kind()) {
1708 (true, ty::Alias(ty::AliasTy { kind: ty::Opaque { def_id }, .. })) => {
1709 let sm = self.tcx.sess.source_map();
1710 let pos = sm.lookup_char_pos(self.tcx.def_span(*def_id).lo());
1711 DiagStyledString::normal(format!(
1712 " (opaque type at <{}:{}:{}>)",
1713 sm.filename_for_diagnostics(&pos.file.name),
1714 pos.line,
1715 pos.col.to_usize() + 1,
1716 ))
1717 }
1718 (true, &ty::Alias(ty::AliasTy { kind: ty::Projection { def_id }, .. }))
1719 if self.tcx.is_impl_trait_in_trait(def_id) =>
1720 {
1721 let sm = self.tcx.sess.source_map();
1722 let pos = sm.lookup_char_pos(self.tcx.def_span(def_id).lo());
1723 DiagStyledString::normal(format!(
1724 " (trait associated opaque type at <{}:{}:{}>)",
1725 sm.filename_for_diagnostics(&pos.file.name),
1726 pos.line,
1727 pos.col.to_usize() + 1,
1728 ))
1729 }
1730 (true, _) => {
1731 let mut s = DiagStyledString::normal(" (");
1732 s.push_highlighted(ty.sort_string(self.tcx));
1733 s.push_normal(")");
1734 s
1735 }
1736 (false, _) => DiagStyledString::normal(""),
1737 };
1738 if !(values.expected.is_simple_text() && values.found.is_simple_text())
1739 || (exp_found.is_some_and(|ef| {
1740 if !ef.expected.is_ty_or_numeric_infer() {
1745 ef.expected != values.expected
1746 } else if !ef.found.is_ty_or_numeric_infer() {
1747 ef.found != values.found
1748 } else {
1749 false
1750 }
1751 }))
1752 {
1753 if let Some(ExpectedFound { found: found_ty, .. }) = exp_found
1754 && !self.tcx.ty_is_opaque_future(found_ty)
1755 {
1756 diag.note_expected_found_extra(
1763 &expected_label,
1764 expected,
1765 &found_label,
1766 found,
1767 sort_string(values.expected),
1768 sort_string(values.found),
1769 );
1770 }
1771 }
1772 }
1773 _ => {
1774 debug!(
1775 "note_type_err: exp_found={:?}, expected={:?} found={:?}",
1776 exp_found, expected, found
1777 );
1778 if !is_simple_error || terr.must_include_note() {
1779 diag.note_expected_found(&expected_label, expected, &found_label, found);
1780
1781 if let Some(ty::Closure(_, args)) =
1782 exp_found.map(|expected_type_found| expected_type_found.found.kind())
1783 {
1784 diag.highlighted_note(vec![
1785 StringPart::normal("closure has signature: `"),
1786 StringPart::highlighted(
1787 self.tcx
1788 .signature_unclosure(
1789 args.as_closure().sig(),
1790 rustc_hir::Safety::Safe,
1791 )
1792 .to_string(),
1793 ),
1794 StringPart::normal("`"),
1795 ]);
1796 }
1797 }
1798 }
1799 }
1800 }
1801 let exp_found = match exp_found {
1802 Mismatch::Variable(exp_found) => Some(exp_found),
1803 Mismatch::Fixed(_) => None,
1804 };
1805 let exp_found = match terr {
1806 ty::error::TypeError::Sorts(terr)
1808 if exp_found.is_some_and(|ef| terr.found == ef.found) =>
1809 {
1810 Some(terr)
1811 }
1812 _ => exp_found,
1813 };
1814 debug!("exp_found {:?} terr {:?} cause.code {:?}", exp_found, terr, cause.code());
1815 if let Some(exp_found) = exp_found {
1816 let should_suggest_fixes =
1817 if let ObligationCauseCode::Pattern { root_ty, .. } = cause.code() {
1818 self.same_type_modulo_infer(*root_ty, exp_found.expected)
1821 } else {
1822 true
1823 };
1824
1825 if should_suggest_fixes
1829 && !matches!(terr, TypeError::RegionsInsufficientlyPolymorphic(..))
1830 {
1831 self.suggest_tuple_pattern(cause, &exp_found, diag);
1832 self.suggest_accessing_field_where_appropriate(cause, &exp_found, diag);
1833 self.suggest_await_on_expect_found(cause, span, &exp_found, diag);
1834 self.suggest_function_pointers(cause, span, &exp_found, terr, diag);
1835 self.suggest_turning_stmt_into_expr(cause, &exp_found, diag);
1836 }
1837 }
1838
1839 let body_owner_def_id = (cause.body_id != CRATE_DEF_ID).then(|| cause.body_id.to_def_id());
1840 self.note_and_explain_type_err(diag, terr, cause, span, body_owner_def_id);
1841 if let Some(exp_found) = exp_found
1842 && let exp_found = TypeError::Sorts(exp_found)
1843 && exp_found != terr
1844 {
1845 self.note_and_explain_type_err(diag, exp_found, cause, span, body_owner_def_id);
1846 }
1847
1848 if let Some(ValuePairs::TraitRefs(exp_found)) = values
1849 && let ty::Closure(def_id, _) = exp_found.expected.self_ty().kind()
1850 && let Some(def_id) = def_id.as_local()
1851 && terr.involves_regions()
1852 {
1853 let span = self.tcx.def_span(def_id);
1854 diag.span_note(span, "this closure does not fulfill the lifetime requirements");
1855 self.suggest_for_all_lifetime_closure(
1856 span,
1857 self.tcx.hir_node_by_def_id(def_id),
1858 &exp_found,
1859 diag,
1860 );
1861 }
1862
1863 self.note_error_origin(diag, cause, exp_found, terr, param_env);
1866
1867 debug!(?diag);
1868 }
1869
1870 pub(crate) fn type_error_additional_suggestions(
1871 &self,
1872 trace: &TypeTrace<'tcx>,
1873 terr: TypeError<'tcx>,
1874 long_ty_path: &mut Option<PathBuf>,
1875 ) -> Vec<TypeErrorAdditionalDiags> {
1876 let mut suggestions = Vec::new();
1877 let span = trace.cause.span;
1878 let values = self.resolve_vars_if_possible(trace.values);
1879 if let Some((expected, found)) = values.ty() {
1880 match (expected.kind(), found.kind()) {
1881 (ty::Tuple(_), ty::Tuple(_)) => {}
1882 (ty::Tuple(fields), _) => {
1886 suggestions.extend(self.suggest_wrap_to_build_a_tuple(span, found, fields))
1887 }
1888 (ty::Uint(ty::UintTy::U8), ty::Char) => {
1892 if let Ok(code) = self.tcx.sess.source_map().span_to_snippet(span)
1893 && let Some(code) = code.strip_circumfix('\'', '\'')
1894 && !code.starts_with("\\u")
1896 && code.chars().next().is_some_and(|c| c.is_ascii())
1898 {
1899 suggestions.push(TypeErrorAdditionalDiags::MeantByteLiteral {
1900 span,
1901 code: escape_literal(code),
1902 })
1903 }
1904 }
1905 (ty::Char, ty::Ref(_, r, _)) if r.is_str() => {
1909 if let Ok(code) = self.tcx.sess.source_map().span_to_snippet(span)
1910 && let Some(code) = code.strip_circumfix('"', '"')
1911 && code.chars().count() == 1
1912 {
1913 suggestions.push(TypeErrorAdditionalDiags::MeantCharLiteral {
1914 span,
1915 code: escape_literal(code),
1916 })
1917 }
1918 }
1919 (ty::Ref(_, r, _), ty::Char) if r.is_str() => {
1922 if let Ok(code) = self.tcx.sess.source_map().span_to_snippet(span)
1923 && code.starts_with("'")
1924 && code.ends_with("'")
1925 {
1926 suggestions.push(TypeErrorAdditionalDiags::MeantStrLiteral {
1927 start: span.with_hi(span.lo() + BytePos(1)),
1928 end: span.with_lo(span.hi() - BytePos(1)),
1929 });
1930 }
1931 }
1932 (ty::Bool, ty::Tuple(list)) => {
1935 if list.len() == 0 {
1936 suggestions.extend(self.suggest_let_for_letchains(&trace.cause, span));
1937 }
1938 }
1939 (ty::Array(_, _), ty::Array(_, _)) => {
1940 suggestions.extend(self.suggest_specify_actual_length(terr, trace, span))
1941 }
1942 _ => {}
1943 }
1944 }
1945 let code = trace.cause.code();
1946 if let &(ObligationCauseCode::MatchExpressionArm(MatchExpressionArmCause {
1947 source, ..
1948 })
1949 | ObligationCauseCode::BlockTailExpression(.., source)) = code
1950 && let hir::MatchSource::TryDesugar(_) = source
1951 && let Some((expected_ty, found_ty)) =
1952 self.values_str(trace.values, &trace.cause, long_ty_path)
1953 {
1954 suggestions.push(TypeErrorAdditionalDiags::TryCannotConvert {
1955 found: found_ty.content(),
1956 expected: expected_ty.content(),
1957 });
1958 }
1959 suggestions
1960 }
1961
1962 fn suggest_specify_actual_length(
1963 &self,
1964 terr: TypeError<'tcx>,
1965 trace: &TypeTrace<'tcx>,
1966 span: Span,
1967 ) -> Option<TypeErrorAdditionalDiags> {
1968 let TypeError::ArraySize(sz) = terr else {
1969 return None;
1970 };
1971 let tykind = match self.tcx.hir_node_by_def_id(trace.cause.body_id) {
1972 hir::Node::Item(hir::Item {
1973 kind: hir::ItemKind::Fn { body: body_id, .. }, ..
1974 }) => {
1975 let body = self.tcx.hir_body(*body_id);
1976 struct LetVisitor {
1977 span: Span,
1978 }
1979 impl<'v> Visitor<'v> for LetVisitor {
1980 type Result = ControlFlow<&'v hir::TyKind<'v>>;
1981 fn visit_stmt(&mut self, s: &'v hir::Stmt<'v>) -> Self::Result {
1982 if let hir::Stmt {
1985 kind:
1986 hir::StmtKind::Let(hir::LetStmt {
1987 init: Some(hir::Expr { span: init_span, .. }),
1988 ty: Some(array_ty),
1989 ..
1990 }),
1991 ..
1992 } = s
1993 && init_span == &self.span
1994 {
1995 ControlFlow::Break(&array_ty.peel_refs().kind)
1996 } else {
1997 ControlFlow::Continue(())
1998 }
1999 }
2000 }
2001 LetVisitor { span }.visit_body(body).break_value()
2002 }
2003 hir::Node::Item(hir::Item { kind: hir::ItemKind::Const(_, _, ty, _), .. }) => {
2004 Some(&ty.peel_refs().kind)
2005 }
2006 _ => None,
2007 };
2008 if let Some(tykind) = tykind
2009 && let hir::TyKind::Array(_, length_arg) = tykind
2010 && let Some(length_val) = sz.found.try_to_target_usize(self.tcx)
2011 {
2012 Some(TypeErrorAdditionalDiags::ConsiderSpecifyingLength {
2013 span: length_arg.span,
2014 length: length_val,
2015 })
2016 } else {
2017 None
2018 }
2019 }
2020
2021 fn check_on_type_error_attribute(
2022 &self,
2023 expected_ty: Ty<'tcx>,
2024 found_ty: Ty<'tcx>,
2025 ) -> ThinVec<String> {
2026 let mut seen = FxHashSet::default();
2027 let mut unique_notes: ThinVec<String> = ThinVec::new();
2028
2029 if let ty::Adt(item_def, args) = found_ty.kind() {
2031 if let Some(Some(directive)) =
2032 {
{
'done:
{
for i in
::rustc_hir::attrs::HasAttrs::get_attrs(item_def.did(),
&self.tcx) {
#[allow(unused_imports)]
use rustc_hir::attrs::AttributeKind::*;
let i: &rustc_hir::Attribute = i;
match i {
rustc_hir::Attribute::Parsed(OnTypeError { directive, .. })
=> {
break 'done Some(directive);
}
rustc_hir::Attribute::Unparsed(..) =>
{}
#[deny(unreachable_patterns)]
_ => {}
}
}
None
}
}
}find_attr!(self.tcx, item_def.did(), OnTypeError { directive, .. } => directive)
2033 {
2034 let notes = self.format_on_type_error_notes(
2035 directive,
2036 args,
2037 item_def.clone(),
2038 expected_ty,
2039 found_ty,
2040 );
2041
2042 for note in notes {
2043 if seen.insert(note.clone()) {
2044 unique_notes.push(note);
2045 }
2046 }
2047 }
2048 }
2049
2050 if let ty::Adt(item_def, args) = expected_ty.kind() {
2052 if let Some(Some(directive)) =
2053 {
{
'done:
{
for i in
::rustc_hir::attrs::HasAttrs::get_attrs(item_def.did(),
&self.tcx) {
#[allow(unused_imports)]
use rustc_hir::attrs::AttributeKind::*;
let i: &rustc_hir::Attribute = i;
match i {
rustc_hir::Attribute::Parsed(OnTypeError { directive, .. })
=> {
break 'done Some(directive);
}
rustc_hir::Attribute::Unparsed(..) =>
{}
#[deny(unreachable_patterns)]
_ => {}
}
}
None
}
}
}find_attr!(self.tcx, item_def.did(), OnTypeError { directive, .. } => directive)
2054 {
2055 let notes = self.format_on_type_error_notes(
2056 directive,
2057 args,
2058 item_def.clone(),
2059 expected_ty,
2060 found_ty,
2061 );
2062
2063 for note in notes {
2064 if seen.insert(note.clone()) {
2065 unique_notes.push(note);
2066 }
2067 }
2068 }
2069 }
2070
2071 unique_notes
2072 }
2073
2074 fn format_on_type_error_notes(
2075 &self,
2076 directive: &Directive,
2077 args: &ty::GenericArgsRef<'tcx>,
2078 item_def: ty::AdtDef<'tcx>,
2079 expected_ty: Ty<'tcx>,
2080 found_ty: Ty<'tcx>,
2081 ) -> ThinVec<String> {
2082 let item_name = self.tcx.item_name(item_def.did()).to_string();
2083 let generic_args: Vec<_> = self
2084 .tcx
2085 .generics_of(item_def.did())
2086 .own_params
2087 .iter()
2088 .filter_map(|param| Some((param.name, args[param.index as usize].to_string())))
2089 .collect();
2090
2091 let format_args = FormatArgs {
2092 this: item_name,
2093 generic_args,
2094 found: found_ty.to_string(),
2095 expected: expected_ty.to_string(),
2096 ..
2097 };
2098 let CustomDiagnostic { notes, .. } = directive.eval(None, &format_args);
2099
2100 notes.into()
2101 }
2102
2103 pub fn report_and_explain_type_error(
2104 &self,
2105 trace: TypeTrace<'tcx>,
2106 param_env: ty::ParamEnv<'tcx>,
2107 terr: TypeError<'tcx>,
2108 ) -> Diag<'a> {
2109 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs:2109",
"rustc_trait_selection::error_reporting::infer",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/error_reporting/infer/mod.rs"),
::tracing_core::__macro_support::Option::Some(2109u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::error_reporting::infer"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("report_and_explain_type_error(trace={0:?}, terr={1:?})",
trace, terr) as &dyn Value))])
});
} else { ; }
};debug!("report_and_explain_type_error(trace={:?}, terr={:?})", trace, terr);
2110
2111 let span = trace.cause.span;
2112 let mut path = None;
2113
2114 let on_type_error_notes = if let Some((expected_ty, found_ty)) = trace.values.ty() {
2116 self.check_on_type_error_attribute(expected_ty, found_ty)
2117 } else {
2118 ThinVec::new()
2119 };
2120
2121 let failure_code = trace.cause.as_failure_code_diag(
2122 terr,
2123 span,
2124 self.type_error_additional_suggestions(&trace, terr, &mut path),
2125 );
2126 let mut diag = self.dcx().create_err(failure_code);
2127 *diag.long_ty_path() = path;
2128
2129 for note in on_type_error_notes {
2131 diag.note(note);
2132 }
2133
2134 self.note_type_err(
2135 &mut diag,
2136 &trace.cause,
2137 None,
2138 Some(param_env.and(trace.values)),
2139 terr,
2140 false,
2141 None,
2142 );
2143 diag
2144 }
2145
2146 fn suggest_wrap_to_build_a_tuple(
2147 &self,
2148 span: Span,
2149 found: Ty<'tcx>,
2150 expected_fields: &List<Ty<'tcx>>,
2151 ) -> Option<TypeErrorAdditionalDiags> {
2152 let [expected_tup_elem] = expected_fields[..] else { return None };
2153
2154 if !self.same_type_modulo_infer(expected_tup_elem, found) {
2155 return None;
2156 }
2157
2158 let Ok(code) = self.tcx.sess.source_map().span_to_snippet(span) else { return None };
2159
2160 let sugg = if code.starts_with('(') && code.ends_with(')') {
2161 let before_close = span.hi() - BytePos::from_u32(1);
2162 TypeErrorAdditionalDiags::TupleOnlyComma {
2163 span: span.with_hi(before_close).shrink_to_hi(),
2164 }
2165 } else {
2166 TypeErrorAdditionalDiags::TupleAlsoParentheses {
2167 span_low: span.shrink_to_lo(),
2168 span_high: span.shrink_to_hi(),
2169 }
2170 };
2171 Some(sugg)
2172 }
2173
2174 fn values_str(
2175 &self,
2176 values: ValuePairs<'tcx>,
2177 cause: &ObligationCause<'tcx>,
2178 long_ty_path: &mut Option<PathBuf>,
2179 ) -> Option<(DiagStyledString, DiagStyledString)> {
2180 match values {
2181 ValuePairs::Regions(exp_found) => self.expected_found_str(exp_found),
2182 ValuePairs::Terms(exp_found) => self.expected_found_str_term(exp_found, long_ty_path),
2183 ValuePairs::Aliases(exp_found) => self.expected_found_str(exp_found),
2184 ValuePairs::ExistentialTraitRef(exp_found) => self.expected_found_str(exp_found),
2185 ValuePairs::ExistentialProjection(exp_found) => self.expected_found_str(exp_found),
2186 ValuePairs::TraitRefs(exp_found) => {
2187 let pretty_exp_found = ty::error::ExpectedFound {
2188 expected: exp_found.expected.print_trait_sugared(),
2189 found: exp_found.found.print_trait_sugared(),
2190 };
2191 match self.expected_found_str(pretty_exp_found) {
2192 Some((expected, found)) if expected == found => {
2193 self.expected_found_str(exp_found)
2194 }
2195 ret => ret,
2196 }
2197 }
2198 ValuePairs::PolySigs(exp_found) => {
2199 let exp_found = self.resolve_vars_if_possible(exp_found);
2200 if exp_found.references_error() {
2201 return None;
2202 }
2203 let (fn_def1, fn_def2) = if let ObligationCauseCode::CompareImplItem {
2204 impl_item_def_id,
2205 trait_item_def_id,
2206 ..
2207 } = *cause.code()
2208 {
2209 (Some((trait_item_def_id, None)), Some((impl_item_def_id.to_def_id(), None)))
2210 } else {
2211 (None, None)
2212 };
2213
2214 Some(self.cmp_fn_sig(exp_found.expected, fn_def1, exp_found.found, fn_def2))
2215 }
2216 }
2217 }
2218
2219 fn expected_found_str_term(
2220 &self,
2221 exp_found: ty::error::ExpectedFound<ty::Term<'tcx>>,
2222 long_ty_path: &mut Option<PathBuf>,
2223 ) -> Option<(DiagStyledString, DiagStyledString)> {
2224 let exp_found = self.resolve_vars_if_possible(exp_found);
2225 if exp_found.references_error() {
2226 return None;
2227 }
2228
2229 Some(match (exp_found.expected.kind(), exp_found.found.kind()) {
2230 (ty::TermKind::Ty(expected), ty::TermKind::Ty(found)) => {
2231 let (mut exp, mut fnd) = self.cmp(expected, found);
2232 let len = self.tcx.sess.diagnostic_width() + 40;
2236 let exp_s = exp.content();
2237 let fnd_s = fnd.content();
2238 if exp_s.len() > len {
2239 let exp_s = self.tcx.short_string(expected, long_ty_path);
2240 exp = DiagStyledString::highlighted(exp_s);
2241 }
2242 if fnd_s.len() > len {
2243 let fnd_s = self.tcx.short_string(found, long_ty_path);
2244 fnd = DiagStyledString::highlighted(fnd_s);
2245 }
2246 (exp, fnd)
2247 }
2248 _ => (
2249 DiagStyledString::highlighted(exp_found.expected.to_string()),
2250 DiagStyledString::highlighted(exp_found.found.to_string()),
2251 ),
2252 })
2253 }
2254
2255 fn expected_found_str<T: fmt::Display + TypeFoldable<TyCtxt<'tcx>>>(
2257 &self,
2258 exp_found: ty::error::ExpectedFound<T>,
2259 ) -> Option<(DiagStyledString, DiagStyledString)> {
2260 let exp_found = self.resolve_vars_if_possible(exp_found);
2261 if exp_found.references_error() {
2262 return None;
2263 }
2264
2265 Some((
2266 DiagStyledString::highlighted(exp_found.expected.to_string()),
2267 DiagStyledString::highlighted(exp_found.found.to_string()),
2268 ))
2269 }
2270
2271 pub fn is_try_conversion(&self, span: Span, trait_def_id: DefId) -> bool {
2275 span.is_desugaring(DesugaringKind::QuestionMark)
2276 && self.tcx.is_diagnostic_item(sym::From, trait_def_id)
2277 }
2278
2279 pub fn same_type_modulo_infer<T: relate::Relate<TyCtxt<'tcx>>>(&self, a: T, b: T) -> bool {
2286 let (a, b) = self.resolve_vars_if_possible((a, b));
2287 SameTypeModuloInfer(self).relate(a, b).is_ok()
2288 }
2289}
2290
2291struct SameTypeModuloInfer<'a, 'tcx>(&'a InferCtxt<'tcx>);
2292
2293impl<'tcx> TypeRelation<TyCtxt<'tcx>> for SameTypeModuloInfer<'_, 'tcx> {
2294 fn cx(&self) -> TyCtxt<'tcx> {
2295 self.0.tcx
2296 }
2297
2298 fn relate_ty_args(
2299 &mut self,
2300 a_ty: Ty<'tcx>,
2301 _: Ty<'tcx>,
2302 _: DefId,
2303 a_args: ty::GenericArgsRef<'tcx>,
2304 b_args: ty::GenericArgsRef<'tcx>,
2305 _: impl FnOnce(ty::GenericArgsRef<'tcx>) -> Ty<'tcx>,
2306 ) -> RelateResult<'tcx, Ty<'tcx>> {
2307 relate::relate_args_invariantly(self, a_args, b_args)?;
2308 Ok(a_ty)
2309 }
2310
2311 fn relate_with_variance<T: relate::Relate<TyCtxt<'tcx>>>(
2312 &mut self,
2313 _variance: ty::Variance,
2314 _info: ty::VarianceDiagInfo<TyCtxt<'tcx>>,
2315 a: T,
2316 b: T,
2317 ) -> relate::RelateResult<'tcx, T> {
2318 self.relate(a, b)
2319 }
2320
2321 fn tys(&mut self, a: Ty<'tcx>, b: Ty<'tcx>) -> RelateResult<'tcx, Ty<'tcx>> {
2322 match (a.kind(), b.kind()) {
2323 (ty::Int(_) | ty::Uint(_), ty::Infer(ty::InferTy::IntVar(_)))
2324 | (
2325 ty::Infer(ty::InferTy::IntVar(_)),
2326 ty::Int(_) | ty::Uint(_) | ty::Infer(ty::InferTy::IntVar(_)),
2327 )
2328 | (ty::Float(_), ty::Infer(ty::InferTy::FloatVar(_)))
2329 | (
2330 ty::Infer(ty::InferTy::FloatVar(_)),
2331 ty::Float(_) | ty::Infer(ty::InferTy::FloatVar(_)),
2332 )
2333 | (ty::Infer(ty::InferTy::TyVar(_)), _)
2334 | (_, ty::Infer(ty::InferTy::TyVar(_))) => Ok(a),
2335 (ty::Infer(_), _) | (_, ty::Infer(_)) => Err(TypeError::Mismatch),
2336 _ => relate::structurally_relate_tys(self, a, b),
2337 }
2338 }
2339
2340 fn regions(
2341 &mut self,
2342 a: ty::Region<'tcx>,
2343 b: ty::Region<'tcx>,
2344 ) -> RelateResult<'tcx, ty::Region<'tcx>> {
2345 if (a.is_var() && b.is_free())
2346 || (b.is_var() && a.is_free())
2347 || (a.is_var() && b.is_var())
2348 || a == b
2349 {
2350 Ok(a)
2351 } else {
2352 Err(TypeError::Mismatch)
2353 }
2354 }
2355
2356 fn binders<T>(
2357 &mut self,
2358 a: ty::Binder<'tcx, T>,
2359 b: ty::Binder<'tcx, T>,
2360 ) -> relate::RelateResult<'tcx, ty::Binder<'tcx, T>>
2361 where
2362 T: relate::Relate<TyCtxt<'tcx>>,
2363 {
2364 Ok(a.rebind(self.relate(a.skip_binder(), b.skip_binder())?))
2365 }
2366
2367 fn consts(
2368 &mut self,
2369 a: ty::Const<'tcx>,
2370 _b: ty::Const<'tcx>,
2371 ) -> relate::RelateResult<'tcx, ty::Const<'tcx>> {
2372 Ok(a)
2375 }
2376}
2377
2378pub enum FailureCode {
2379 Error0317,
2380 Error0580,
2381 Error0308,
2382 Error0644,
2383}
2384
2385impl<'tcx> ObligationCauseExt<'tcx> for ObligationCause<'tcx> {
fn as_failure_code(&self, terr: TypeError<'tcx>) -> FailureCode {
match self.code() {
ObligationCauseCode::IfExpressionWithNoElse =>
FailureCode::Error0317,
ObligationCauseCode::MainFunctionType => FailureCode::Error0580,
ObligationCauseCode::CompareImplItem { .. } |
ObligationCauseCode::MatchExpressionArm(_) |
ObligationCauseCode::IfExpression { .. } |
ObligationCauseCode::LetElse |
ObligationCauseCode::LangFunctionType(_) |
ObligationCauseCode::IntrinsicType |
ObligationCauseCode::MethodReceiver => FailureCode::Error0308,
_ =>
match terr {
TypeError::CyclicTy(ty) if
ty.is_closure() || ty.is_coroutine() ||
ty.is_coroutine_closure() => {
FailureCode::Error0644
}
TypeError::IntrinsicCast | TypeError::ForceInlineCast =>
FailureCode::Error0308,
_ => FailureCode::Error0308,
},
}
}
fn as_failure_code_diag(&self, terr: TypeError<'tcx>, span: Span,
subdiags: Vec<TypeErrorAdditionalDiags>)
-> ObligationCauseFailureCode {
match self.code() {
ObligationCauseCode::CompareImplItem {
kind: ty::AssocKind::Fn { .. }, .. } => {
ObligationCauseFailureCode::MethodCompat { span, subdiags }
}
ObligationCauseCode::CompareImplItem {
kind: ty::AssocKind::Type { .. }, .. } => {
ObligationCauseFailureCode::TypeCompat { span, subdiags }
}
ObligationCauseCode::CompareImplItem {
kind: ty::AssocKind::Const { .. }, .. } => {
ObligationCauseFailureCode::ConstCompat { span, subdiags }
}
ObligationCauseCode::BlockTailExpression(..,
hir::MatchSource::TryDesugar(_)) => {
ObligationCauseFailureCode::TryCompat { span, subdiags }
}
ObligationCauseCode::MatchExpressionArm(MatchExpressionArmCause {
source, .. }) => {
match source {
hir::MatchSource::TryDesugar(_) => {
ObligationCauseFailureCode::TryCompat { span, subdiags }
}
_ =>
ObligationCauseFailureCode::MatchCompat { span, subdiags },
}
}
ObligationCauseCode::IfExpression { .. } => {
ObligationCauseFailureCode::IfElseDifferent { span, subdiags }
}
ObligationCauseCode::IfExpressionWithNoElse => {
ObligationCauseFailureCode::NoElse { span }
}
ObligationCauseCode::LetElse => {
ObligationCauseFailureCode::NoDiverge { span, subdiags }
}
ObligationCauseCode::MainFunctionType => {
ObligationCauseFailureCode::FnMainCorrectType { span }
}
&ObligationCauseCode::LangFunctionType(lang_item_name) => {
ObligationCauseFailureCode::FnLangCorrectType {
span,
subdiags,
lang_item_name,
}
}
ObligationCauseCode::IntrinsicType => {
ObligationCauseFailureCode::IntrinsicCorrectType {
span,
subdiags,
}
}
ObligationCauseCode::MethodReceiver => {
ObligationCauseFailureCode::MethodCorrectType {
span,
subdiags,
}
}
_ =>
match terr {
TypeError::CyclicTy(ty) if
ty.is_closure() || ty.is_coroutine() ||
ty.is_coroutine_closure() => {
ObligationCauseFailureCode::ClosureSelfref { span }
}
TypeError::ForceInlineCast => {
ObligationCauseFailureCode::CantCoerceForceInline {
span,
subdiags,
}
}
TypeError::IntrinsicCast => {
ObligationCauseFailureCode::CantCoerceIntrinsic {
span,
subdiags,
}
}
_ => ObligationCauseFailureCode::Generic { span, subdiags },
},
}
}
fn as_requirement_str(&self) -> &'static str {
match self.code() {
ObligationCauseCode::CompareImplItem {
kind: ty::AssocKind::Fn { .. }, .. } => {
"method type is compatible with trait"
}
ObligationCauseCode::CompareImplItem {
kind: ty::AssocKind::Type { .. }, .. } => {
"associated type is compatible with trait"
}
ObligationCauseCode::CompareImplItem {
kind: ty::AssocKind::Const { .. }, .. } => {
"const is compatible with trait"
}
ObligationCauseCode::MainFunctionType =>
"`main` function has the correct type",
ObligationCauseCode::LangFunctionType(_) =>
"lang item function has the correct type",
ObligationCauseCode::IntrinsicType =>
"intrinsic has the correct type",
ObligationCauseCode::MethodReceiver =>
"method receiver has the correct type",
_ => "types are compatible",
}
}
}#[extension(pub trait ObligationCauseExt<'tcx>)]
2386impl<'tcx> ObligationCause<'tcx> {
2387 fn as_failure_code(&self, terr: TypeError<'tcx>) -> FailureCode {
2388 match self.code() {
2389 ObligationCauseCode::IfExpressionWithNoElse => FailureCode::Error0317,
2390 ObligationCauseCode::MainFunctionType => FailureCode::Error0580,
2391 ObligationCauseCode::CompareImplItem { .. }
2392 | ObligationCauseCode::MatchExpressionArm(_)
2393 | ObligationCauseCode::IfExpression { .. }
2394 | ObligationCauseCode::LetElse
2395 | ObligationCauseCode::LangFunctionType(_)
2396 | ObligationCauseCode::IntrinsicType
2397 | ObligationCauseCode::MethodReceiver => FailureCode::Error0308,
2398
2399 _ => match terr {
2403 TypeError::CyclicTy(ty)
2404 if ty.is_closure() || ty.is_coroutine() || ty.is_coroutine_closure() =>
2405 {
2406 FailureCode::Error0644
2407 }
2408 TypeError::IntrinsicCast | TypeError::ForceInlineCast => FailureCode::Error0308,
2409 _ => FailureCode::Error0308,
2410 },
2411 }
2412 }
2413
2414 fn as_failure_code_diag(
2415 &self,
2416 terr: TypeError<'tcx>,
2417 span: Span,
2418 subdiags: Vec<TypeErrorAdditionalDiags>,
2419 ) -> ObligationCauseFailureCode {
2420 match self.code() {
2421 ObligationCauseCode::CompareImplItem { kind: ty::AssocKind::Fn { .. }, .. } => {
2422 ObligationCauseFailureCode::MethodCompat { span, subdiags }
2423 }
2424 ObligationCauseCode::CompareImplItem { kind: ty::AssocKind::Type { .. }, .. } => {
2425 ObligationCauseFailureCode::TypeCompat { span, subdiags }
2426 }
2427 ObligationCauseCode::CompareImplItem { kind: ty::AssocKind::Const { .. }, .. } => {
2428 ObligationCauseFailureCode::ConstCompat { span, subdiags }
2429 }
2430 ObligationCauseCode::BlockTailExpression(.., hir::MatchSource::TryDesugar(_)) => {
2431 ObligationCauseFailureCode::TryCompat { span, subdiags }
2432 }
2433 ObligationCauseCode::MatchExpressionArm(MatchExpressionArmCause { source, .. }) => {
2434 match source {
2435 hir::MatchSource::TryDesugar(_) => {
2436 ObligationCauseFailureCode::TryCompat { span, subdiags }
2437 }
2438 _ => ObligationCauseFailureCode::MatchCompat { span, subdiags },
2439 }
2440 }
2441 ObligationCauseCode::IfExpression { .. } => {
2442 ObligationCauseFailureCode::IfElseDifferent { span, subdiags }
2443 }
2444 ObligationCauseCode::IfExpressionWithNoElse => {
2445 ObligationCauseFailureCode::NoElse { span }
2446 }
2447 ObligationCauseCode::LetElse => {
2448 ObligationCauseFailureCode::NoDiverge { span, subdiags }
2449 }
2450 ObligationCauseCode::MainFunctionType => {
2451 ObligationCauseFailureCode::FnMainCorrectType { span }
2452 }
2453 &ObligationCauseCode::LangFunctionType(lang_item_name) => {
2454 ObligationCauseFailureCode::FnLangCorrectType { span, subdiags, lang_item_name }
2455 }
2456 ObligationCauseCode::IntrinsicType => {
2457 ObligationCauseFailureCode::IntrinsicCorrectType { span, subdiags }
2458 }
2459 ObligationCauseCode::MethodReceiver => {
2460 ObligationCauseFailureCode::MethodCorrectType { span, subdiags }
2461 }
2462
2463 _ => match terr {
2467 TypeError::CyclicTy(ty)
2468 if ty.is_closure() || ty.is_coroutine() || ty.is_coroutine_closure() =>
2469 {
2470 ObligationCauseFailureCode::ClosureSelfref { span }
2471 }
2472 TypeError::ForceInlineCast => {
2473 ObligationCauseFailureCode::CantCoerceForceInline { span, subdiags }
2474 }
2475 TypeError::IntrinsicCast => {
2476 ObligationCauseFailureCode::CantCoerceIntrinsic { span, subdiags }
2477 }
2478 _ => ObligationCauseFailureCode::Generic { span, subdiags },
2479 },
2480 }
2481 }
2482
2483 fn as_requirement_str(&self) -> &'static str {
2484 match self.code() {
2485 ObligationCauseCode::CompareImplItem { kind: ty::AssocKind::Fn { .. }, .. } => {
2486 "method type is compatible with trait"
2487 }
2488 ObligationCauseCode::CompareImplItem { kind: ty::AssocKind::Type { .. }, .. } => {
2489 "associated type is compatible with trait"
2490 }
2491 ObligationCauseCode::CompareImplItem { kind: ty::AssocKind::Const { .. }, .. } => {
2492 "const is compatible with trait"
2493 }
2494 ObligationCauseCode::MainFunctionType => "`main` function has the correct type",
2495 ObligationCauseCode::LangFunctionType(_) => "lang item function has the correct type",
2496 ObligationCauseCode::IntrinsicType => "intrinsic has the correct type",
2497 ObligationCauseCode::MethodReceiver => "method receiver has the correct type",
2498 _ => "types are compatible",
2499 }
2500 }
2501}
2502
2503pub struct ObligationCauseAsDiagArg<'tcx>(pub ObligationCause<'tcx>);
2505
2506impl IntoDiagArg for ObligationCauseAsDiagArg<'_> {
2507 fn into_diag_arg(self, _: &mut Option<std::path::PathBuf>) -> rustc_errors::DiagArgValue {
2508 let kind = match self.0.code() {
2509 ObligationCauseCode::CompareImplItem { kind: ty::AssocKind::Fn { .. }, .. } => {
2510 "method_compat"
2511 }
2512 ObligationCauseCode::CompareImplItem { kind: ty::AssocKind::Type { .. }, .. } => {
2513 "type_compat"
2514 }
2515 ObligationCauseCode::CompareImplItem { kind: ty::AssocKind::Const { .. }, .. } => {
2516 "const_compat"
2517 }
2518 ObligationCauseCode::MainFunctionType => "fn_main_correct_type",
2519 ObligationCauseCode::LangFunctionType(_) => "fn_lang_correct_type",
2520 ObligationCauseCode::IntrinsicType => "intrinsic_correct_type",
2521 ObligationCauseCode::MethodReceiver => "method_correct_type",
2522 _ => "other",
2523 }
2524 .into();
2525 rustc_errors::DiagArgValue::Str(kind)
2526 }
2527}
2528
2529#[derive(#[automatically_derived]
impl ::core::clone::Clone for TyCategory {
#[inline]
fn clone(&self) -> TyCategory {
let _: ::core::clone::AssertParamIsClone<hir::CoroutineKind>;
*self
}
}Clone, #[automatically_derived]
impl ::core::marker::Copy for TyCategory { }Copy, #[automatically_derived]
impl ::core::cmp::PartialEq for TyCategory {
#[inline]
fn eq(&self, other: &TyCategory) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr &&
match (self, other) {
(TyCategory::Coroutine(__self_0),
TyCategory::Coroutine(__arg1_0)) => __self_0 == __arg1_0,
_ => true,
}
}
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for TyCategory {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<hir::CoroutineKind>;
}
}Eq, #[automatically_derived]
impl ::core::hash::Hash for TyCategory {
#[inline]
fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
let __self_discr = ::core::intrinsics::discriminant_value(self);
::core::hash::Hash::hash(&__self_discr, state);
match self {
TyCategory::Coroutine(__self_0) =>
::core::hash::Hash::hash(__self_0, state),
_ => {}
}
}
}Hash)]
2532pub enum TyCategory {
2533 Closure,
2534 Opaque,
2535 OpaqueFuture,
2536 Coroutine(hir::CoroutineKind),
2537 Foreign,
2538}
2539
2540impl fmt::Display for TyCategory {
2541 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2542 match self {
2543 Self::Closure => "closure".fmt(f),
2544 Self::Opaque => "opaque type".fmt(f),
2545 Self::OpaqueFuture => "future".fmt(f),
2546 Self::Coroutine(gk) => gk.fmt(f),
2547 Self::Foreign => "foreign type".fmt(f),
2548 }
2549 }
2550}
2551
2552impl TyCategory {
2553 pub fn from_ty(tcx: TyCtxt<'_>, ty: Ty<'_>) -> Option<(Self, DefId)> {
2554 match *ty.kind() {
2555 ty::Closure(def_id, _) => Some((Self::Closure, def_id)),
2556 ty::Alias(ty::AliasTy { kind: ty::Opaque { def_id }, .. }) => {
2557 let kind =
2558 if tcx.ty_is_opaque_future(ty) { Self::OpaqueFuture } else { Self::Opaque };
2559 Some((kind, def_id))
2560 }
2561 ty::Coroutine(def_id, ..) => {
2562 Some((Self::Coroutine(tcx.coroutine_kind(def_id).unwrap()), def_id))
2563 }
2564 ty::Foreign(def_id) => Some((Self::Foreign, def_id)),
2565 _ => None,
2566 }
2567 }
2568}