1use std::ops::ControlFlow;
8
9use rustc_errors::FatalError;
10use rustc_hir::def::DefKind;
11use rustc_hir::def_id::DefId;
12use rustc_hir::{self as hir, LangItem};
13use rustc_middle::query::Providers;
14use rustc_middle::ty::{
15 self, EarlyBinder, GenericArgs, Ty, TyCtxt, TypeFoldable, TypeFolder, TypeSuperFoldable,
16 TypeSuperVisitable, TypeVisitable, TypeVisitableExt, TypeVisitor, TypingMode, Unnormalized,
17 Upcast, elaborate,
18};
19use rustc_span::{DUMMY_SP, Span};
20use smallvec::SmallVec;
21use tracing::{debug, instrument};
22
23use super::elaborate;
24use crate::infer::TyCtxtInferExt;
25pub use crate::traits::DynCompatibilityViolation;
26use crate::traits::query::evaluate_obligation::InferCtxtExt;
27use crate::traits::{
28 AssocConstViolation, MethodViolation, Obligation, ObligationCause,
29 normalize_param_env_or_error, util,
30};
31
32x;#[instrument(level = "debug", skip(tcx), ret)]
38pub fn hir_ty_lowering_dyn_compatibility_violations(
39 tcx: TyCtxt<'_>,
40 trait_def_id: DefId,
41) -> Vec<DynCompatibilityViolation> {
42 debug_assert!(tcx.generics_of(trait_def_id).has_self);
43 elaborate::supertrait_def_ids(tcx, trait_def_id)
44 .map(|def_id| predicates_reference_self(tcx, def_id, true))
45 .filter(|spans| !spans.is_empty())
46 .map(DynCompatibilityViolation::SupertraitSelf)
47 .collect()
48}
49
50fn dyn_compatibility_violations(
51 tcx: TyCtxt<'_>,
52 trait_def_id: DefId,
53) -> &'_ [DynCompatibilityViolation] {
54 if true {
if !tcx.generics_of(trait_def_id).has_self {
::core::panicking::panic("assertion failed: tcx.generics_of(trait_def_id).has_self")
};
};debug_assert!(tcx.generics_of(trait_def_id).has_self);
55 {
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/traits/dyn_compatibility.rs:55",
"rustc_trait_selection::traits::dyn_compatibility",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/traits/dyn_compatibility.rs"),
::tracing_core::__macro_support::Option::Some(55u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::traits::dyn_compatibility"),
::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!("dyn_compatibility_violations: {0:?}",
trait_def_id) as &dyn Value))])
});
} else { ; }
};debug!("dyn_compatibility_violations: {:?}", trait_def_id);
56 tcx.arena.alloc_from_iter(
57 elaborate::supertrait_def_ids(tcx, trait_def_id)
58 .flat_map(|def_id| dyn_compatibility_violations_for_trait(tcx, def_id)),
59 )
60}
61
62fn is_dyn_compatible(tcx: TyCtxt<'_>, trait_def_id: DefId) -> bool {
63 tcx.dyn_compatibility_violations(trait_def_id).is_empty()
64}
65
66pub fn is_vtable_safe_method(tcx: TyCtxt<'_>, trait_def_id: DefId, method: ty::AssocItem) -> bool {
71 if true {
if !tcx.generics_of(trait_def_id).has_self {
::core::panicking::panic("assertion failed: tcx.generics_of(trait_def_id).has_self")
};
};debug_assert!(tcx.generics_of(trait_def_id).has_self);
72 {
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/traits/dyn_compatibility.rs:72",
"rustc_trait_selection::traits::dyn_compatibility",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/traits/dyn_compatibility.rs"),
::tracing_core::__macro_support::Option::Some(72u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::traits::dyn_compatibility"),
::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!("is_vtable_safe_method({0:?}, {1:?})",
trait_def_id, method) as &dyn Value))])
});
} else { ; }
};debug!("is_vtable_safe_method({:?}, {:?})", trait_def_id, method);
73 if tcx.generics_require_sized_self(method.def_id) {
75 return false;
76 }
77
78 virtual_call_violations_for_method(tcx, trait_def_id, method).is_empty()
79}
80
81x;#[instrument(level = "debug", skip(tcx), ret)]
82fn dyn_compatibility_violations_for_trait(
83 tcx: TyCtxt<'_>,
84 trait_def_id: DefId,
85) -> Vec<DynCompatibilityViolation> {
86 let mut violations: Vec<_> = tcx
88 .associated_items(trait_def_id)
89 .in_definition_order()
90 .flat_map(|&item| dyn_compatibility_violations_for_assoc_item(tcx, trait_def_id, item))
91 .collect();
92
93 if trait_has_sized_self(tcx, trait_def_id) {
95 let spans = get_sized_bounds(tcx, trait_def_id);
97 violations.push(DynCompatibilityViolation::SizedSelf(spans));
98 } else if let Some(span) = tcx.trait_def(trait_def_id).force_dyn_incompatible {
99 violations.push(DynCompatibilityViolation::ExplicitlyDynIncompatible([span].into()));
100 }
101
102 let spans = predicates_reference_self(tcx, trait_def_id, false);
103 if !spans.is_empty() {
104 violations.push(DynCompatibilityViolation::SupertraitSelf(spans));
105 }
106 let spans = bounds_reference_self(tcx, trait_def_id);
107 if !spans.is_empty() {
108 violations.push(DynCompatibilityViolation::SupertraitSelf(spans));
109 }
110 let spans = super_predicates_have_non_lifetime_binders(tcx, trait_def_id);
111 if !spans.is_empty() {
112 violations.push(DynCompatibilityViolation::SupertraitNonLifetimeBinder(spans));
113 }
114 let spans = super_predicates_are_unconditionally_const(tcx, trait_def_id);
115 if !spans.is_empty() {
116 violations.push(DynCompatibilityViolation::SupertraitConst(spans));
117 }
118
119 violations
120}
121
122fn sized_trait_bound_spans<'tcx>(
123 tcx: TyCtxt<'tcx>,
124 bounds: hir::GenericBounds<'tcx>,
125) -> impl 'tcx + Iterator<Item = Span> {
126 bounds.iter().filter_map(move |b| match b {
127 hir::GenericBound::Trait(trait_ref)
128 if trait_has_sized_self(
129 tcx,
130 trait_ref.trait_ref.trait_def_id().unwrap_or_else(|| FatalError.raise()),
131 ) =>
132 {
133 Some(trait_ref.span)
135 }
136 _ => None,
137 })
138}
139
140fn get_sized_bounds(tcx: TyCtxt<'_>, trait_def_id: DefId) -> SmallVec<[Span; 1]> {
141 tcx.hir_get_if_local(trait_def_id)
142 .and_then(|node| match node {
143 hir::Node::Item(hir::Item {
144 kind: hir::ItemKind::Trait { generics, bounds, .. },
145 ..
146 }) => Some(
147 generics
148 .predicates
149 .iter()
150 .filter_map(|pred| {
151 match pred.kind {
152 hir::WherePredicateKind::BoundPredicate(pred)
153 if pred.bounded_ty.hir_id.owner.to_def_id() == trait_def_id =>
154 {
155 Some(sized_trait_bound_spans(tcx, pred.bounds))
158 }
159 _ => None,
160 }
161 })
162 .flatten()
163 .chain(sized_trait_bound_spans(tcx, bounds))
165 .collect::<SmallVec<[Span; 1]>>(),
166 ),
167 _ => None,
168 })
169 .unwrap_or_else(SmallVec::new)
170}
171
172fn predicates_reference_self(
173 tcx: TyCtxt<'_>,
174 trait_def_id: DefId,
175 supertraits_only: bool,
176) -> SmallVec<[Span; 1]> {
177 let trait_ref = ty::Binder::dummy(ty::TraitRef::identity(tcx, trait_def_id));
178 let predicates = if supertraits_only {
179 tcx.explicit_super_predicates_of(trait_def_id).skip_binder()
180 } else {
181 tcx.predicates_of(trait_def_id).predicates
182 };
183 predicates
184 .iter()
185 .map(|&(predicate, sp)| (predicate.instantiate_supertrait(tcx, trait_ref), sp))
186 .filter_map(|(clause, sp)| {
187 predicate_references_self(tcx, trait_def_id, clause, sp, AllowSelfProjections::No)
192 })
193 .collect()
194}
195
196fn bounds_reference_self(tcx: TyCtxt<'_>, trait_def_id: DefId) -> SmallVec<[Span; 1]> {
197 tcx.associated_items(trait_def_id)
198 .in_definition_order()
199 .filter(|item| item.is_type())
201 .filter(|item| !tcx.generics_require_sized_self(item.def_id))
203 .flat_map(|item| {
204 tcx.explicit_item_bounds(item.def_id)
205 .iter_identity_copied()
206 .map(Unnormalized::skip_norm_wip)
207 })
208 .filter_map(|(clause, sp)| {
209 predicate_references_self(tcx, trait_def_id, clause, sp, AllowSelfProjections::Yes)
212 })
213 .collect()
214}
215
216fn predicate_references_self<'tcx>(
217 tcx: TyCtxt<'tcx>,
218 trait_def_id: DefId,
219 predicate: ty::Clause<'tcx>,
220 sp: Span,
221 allow_self_projections: AllowSelfProjections,
222) -> Option<Span> {
223 match predicate.kind().skip_binder() {
224 ty::ClauseKind::Trait(ref data) => {
225 data.trait_ref.args[1..]
227 .iter()
228 .any(|&arg| {
229 contains_illegal_self_type_reference(
230 tcx,
231 trait_def_id,
232 arg,
233 allow_self_projections,
234 )
235 })
236 .then_some(sp)
237 }
238 ty::ClauseKind::Projection(ref data) => {
239 data.projection_term.args[1..]
255 .iter()
256 .any(|&arg| {
257 contains_illegal_self_type_reference(
258 tcx,
259 trait_def_id,
260 arg,
261 allow_self_projections,
262 )
263 })
264 .then_some(sp)
265 }
266 ty::ClauseKind::ConstArgHasType(_ct, ty) => {
267 contains_illegal_self_type_reference(tcx, trait_def_id, ty, allow_self_projections)
268 .then_some(sp)
269 }
270
271 ty::ClauseKind::WellFormed(..)
272 | ty::ClauseKind::TypeOutlives(..)
273 | ty::ClauseKind::RegionOutlives(..)
274 | ty::ClauseKind::HostEffect(..)
275 | ty::ClauseKind::UnstableFeature(_) => None,
276
277 ty::ClauseKind::ConstEvaluatable(..) => None,
279 }
280}
281
282fn super_predicates_have_non_lifetime_binders(
283 tcx: TyCtxt<'_>,
284 trait_def_id: DefId,
285) -> SmallVec<[Span; 1]> {
286 tcx.explicit_super_predicates_of(trait_def_id)
287 .iter_identity_copied()
288 .map(Unnormalized::skip_norm_wip)
289 .filter_map(|(pred, span)| pred.has_non_region_bound_vars().then_some(span))
290 .collect()
291}
292
293fn super_predicates_are_unconditionally_const(
297 tcx: TyCtxt<'_>,
298 trait_def_id: DefId,
299) -> SmallVec<[Span; 1]> {
300 tcx.explicit_super_predicates_of(trait_def_id)
301 .iter_identity_copied()
302 .map(Unnormalized::skip_norm_wip)
303 .filter_map(|(pred, span)| {
304 if let ty::ClauseKind::HostEffect(_) = pred.kind().skip_binder() {
305 Some(span)
306 } else {
307 None
308 }
309 })
310 .collect()
311}
312
313fn trait_has_sized_self(tcx: TyCtxt<'_>, trait_def_id: DefId) -> bool {
314 tcx.generics_require_sized_self(trait_def_id)
315}
316
317fn generics_require_sized_self(tcx: TyCtxt<'_>, def_id: DefId) -> bool {
318 let Some(sized_def_id) = tcx.lang_items().sized_trait() else {
319 return false; };
321
322 let predicates: Vec<_> = tcx
324 .predicates_of(def_id)
325 .instantiate_identity(tcx)
326 .predicates
327 .into_iter()
328 .map(Unnormalized::skip_norm_wip)
329 .collect();
330 elaborate(tcx, predicates).any(|pred| match pred.kind().skip_binder() {
331 ty::ClauseKind::Trait(ref trait_pred) => {
332 trait_pred.def_id() == sized_def_id && trait_pred.self_ty().is_param(0)
333 }
334 ty::ClauseKind::RegionOutlives(_)
335 | ty::ClauseKind::TypeOutlives(_)
336 | ty::ClauseKind::Projection(_)
337 | ty::ClauseKind::ConstArgHasType(_, _)
338 | ty::ClauseKind::WellFormed(_)
339 | ty::ClauseKind::ConstEvaluatable(_)
340 | ty::ClauseKind::UnstableFeature(_)
341 | ty::ClauseKind::HostEffect(..) => false,
342 })
343}
344
345x;#[instrument(level = "debug", skip(tcx), ret)]
346pub fn dyn_compatibility_violations_for_assoc_item(
347 tcx: TyCtxt<'_>,
348 trait_def_id: DefId,
349 item: ty::AssocItem,
350) -> Vec<DynCompatibilityViolation> {
351 if tcx.defaultness(item.def_id).is_final() {
353 return Vec::new();
354 }
355
356 if tcx.generics_require_sized_self(item.def_id) {
358 return Vec::new();
359 }
360
361 let span = || item.ident(tcx).span;
362
363 match item.kind {
364 ty::AssocKind::Const { name, is_type_const } => {
365 let mut errors = Vec::new();
370
371 if tcx.features().min_generic_const_args() {
372 if !tcx.generics_of(item.def_id).is_own_empty() {
373 errors.push(AssocConstViolation::Generic);
374 } else if !is_type_const {
375 errors.push(AssocConstViolation::NonType);
376 }
377
378 let ty = ty::Binder::dummy(
379 tcx.type_of(item.def_id).instantiate_identity().skip_norm_wip(),
380 );
381 if contains_illegal_self_type_reference(
382 tcx,
383 trait_def_id,
384 ty,
385 AllowSelfProjections::Yes,
386 ) {
387 errors.push(AssocConstViolation::TypeReferencesSelf);
388 }
389 } else {
390 errors.push(AssocConstViolation::FeatureNotEnabled);
391 }
392
393 errors
394 .into_iter()
395 .map(|error| DynCompatibilityViolation::AssocConst(name, error, span()))
396 .collect()
397 }
398 ty::AssocKind::Fn { name, .. } => {
399 virtual_call_violations_for_method(tcx, trait_def_id, item)
400 .into_iter()
401 .map(|v| {
402 let node = tcx.hir_get_if_local(item.def_id);
403 let span = match (&v, node) {
405 (MethodViolation::ReferencesSelfInput(Some(span)), _) => *span,
406 (MethodViolation::UndispatchableReceiver(Some(span)), _) => *span,
407 (MethodViolation::ReferencesImplTraitInTrait(span), _) => *span,
408 (MethodViolation::ReferencesSelfOutput, Some(node)) => {
409 node.fn_decl().map_or(item.ident(tcx).span, |decl| decl.output.span())
410 }
411 _ => span(),
412 };
413
414 DynCompatibilityViolation::Method(name, v, span)
415 })
416 .collect()
417 }
418 ty::AssocKind::Type { data } => {
419 if !tcx.generics_of(item.def_id).is_own_empty()
420 && let ty::AssocTypeData::Normal(name) = data
421 {
422 vec![DynCompatibilityViolation::GenericAssocTy(name, span())]
423 } else {
424 Vec::new()
428 }
429 }
430 }
431}
432
433fn virtual_call_violations_for_method<'tcx>(
438 tcx: TyCtxt<'tcx>,
439 trait_def_id: DefId,
440 method: ty::AssocItem,
441) -> Vec<MethodViolation> {
442 let sig = tcx.fn_sig(method.def_id).instantiate_identity().skip_norm_wip();
443
444 if !method.is_method() {
446 let sugg = if let Some(hir::Node::TraitItem(hir::TraitItem {
447 generics,
448 kind: hir::TraitItemKind::Fn(sig, _),
449 ..
450 })) = tcx.hir_get_if_local(method.def_id).as_ref()
451 {
452 let sm = tcx.sess.source_map();
453 Some((
454 (
455 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("&self{0}",
if sig.decl.inputs.is_empty() { "" } else { ", " }))
})format!("&self{}", if sig.decl.inputs.is_empty() { "" } else { ", " }),
456 sm.span_through_char(sig.span, '(').shrink_to_hi(),
457 ),
458 (
459 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} Self: Sized",
generics.add_where_or_trailing_comma()))
})format!("{} Self: Sized", generics.add_where_or_trailing_comma()),
460 generics.tail_span_for_predicate_suggestion(),
461 ),
462 ))
463 } else {
464 None
465 };
466
467 return ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[MethodViolation::StaticMethod(sugg)]))vec![MethodViolation::StaticMethod(sugg)];
470 }
471
472 let mut errors = Vec::new();
473
474 for (i, &input_ty) in sig.skip_binder().inputs().iter().enumerate().skip(1) {
475 if contains_illegal_self_type_reference(
476 tcx,
477 trait_def_id,
478 sig.rebind(input_ty),
479 AllowSelfProjections::Yes,
480 ) {
481 let span = if let Some(hir::Node::TraitItem(hir::TraitItem {
482 kind: hir::TraitItemKind::Fn(sig, _),
483 ..
484 })) = tcx.hir_get_if_local(method.def_id).as_ref()
485 {
486 Some(sig.decl.inputs[i].span)
487 } else {
488 None
489 };
490 errors.push(MethodViolation::ReferencesSelfInput(span));
491 }
492 }
493 if contains_illegal_self_type_reference(
494 tcx,
495 trait_def_id,
496 sig.output(),
497 AllowSelfProjections::Yes,
498 ) {
499 errors.push(MethodViolation::ReferencesSelfOutput);
500 }
501 if let Some(error) = contains_illegal_impl_trait_in_trait(tcx, method.def_id, sig.output()) {
502 errors.push(error);
503 }
504 if sig.skip_binder().c_variadic() {
505 errors.push(MethodViolation::CVariadic);
506 }
507
508 let own_counts = tcx.generics_of(method.def_id).own_counts();
510 if own_counts.types > 0 || own_counts.consts > 0 {
511 errors.push(MethodViolation::Generic);
512 }
513
514 let receiver_ty = tcx.liberate_late_bound_regions(method.def_id, sig.input(0));
515
516 if receiver_ty != tcx.types.self_param {
521 if !receiver_is_dispatchable(tcx, method, receiver_ty) {
522 let span = if let Some(hir::Node::TraitItem(hir::TraitItem {
523 kind: hir::TraitItemKind::Fn(sig, _),
524 ..
525 })) = tcx.hir_get_if_local(method.def_id).as_ref()
526 {
527 Some(sig.decl.inputs[0].span)
528 } else {
529 None
530 };
531 errors.push(MethodViolation::UndispatchableReceiver(span));
532 } else {
533 }
536 }
537
538 if tcx.predicates_of(method.def_id).predicates.iter().any(|&(pred, _span)| {
541 if pred.as_type_outlives_clause().is_some() {
551 return false;
552 }
553
554 if let ty::ClauseKind::Trait(ty::TraitPredicate {
568 trait_ref: pred_trait_ref,
569 polarity: ty::PredicatePolarity::Positive,
570 }) = pred.kind().skip_binder()
571 && pred_trait_ref.self_ty() == tcx.types.self_param
572 && tcx.trait_is_auto(pred_trait_ref.def_id)
573 {
574 if pred_trait_ref.args.len() != 1 {
579 if !tcx.dcx().has_errors().is_some() {
{
::core::panicking::panic_fmt(format_args!("auto traits cannot have generic parameters"));
}
};assert!(
580 tcx.dcx().has_errors().is_some(),
581 "auto traits cannot have generic parameters"
582 );
583 }
584 return false;
585 }
586
587 contains_illegal_self_type_reference(tcx, trait_def_id, pred, AllowSelfProjections::Yes)
588 }) {
589 errors.push(MethodViolation::WhereClauseReferencesSelf);
590 }
591
592 errors
593}
594
595fn receiver_for_self_ty<'tcx>(
598 tcx: TyCtxt<'tcx>,
599 receiver_ty: Ty<'tcx>,
600 self_ty: Ty<'tcx>,
601 method_def_id: DefId,
602) -> Ty<'tcx> {
603 {
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/traits/dyn_compatibility.rs:603",
"rustc_trait_selection::traits::dyn_compatibility",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/traits/dyn_compatibility.rs"),
::tracing_core::__macro_support::Option::Some(603u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::traits::dyn_compatibility"),
::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!("receiver_for_self_ty({0:?}, {1:?}, {2:?})",
receiver_ty, self_ty, method_def_id) as &dyn Value))])
});
} else { ; }
};debug!("receiver_for_self_ty({:?}, {:?}, {:?})", receiver_ty, self_ty, method_def_id);
604 let args = GenericArgs::for_item(tcx, method_def_id, |param, _| {
605 if param.index == 0 { self_ty.into() } else { tcx.mk_param_from_def(param) }
606 });
607
608 let result = EarlyBinder::bind(receiver_ty).instantiate(tcx, args).skip_norm_wip();
609 {
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/traits/dyn_compatibility.rs:609",
"rustc_trait_selection::traits::dyn_compatibility",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/traits/dyn_compatibility.rs"),
::tracing_core::__macro_support::Option::Some(609u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::traits::dyn_compatibility"),
::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!("receiver_for_self_ty({0:?}, {1:?}, {2:?}) = {3:?}",
receiver_ty, self_ty, method_def_id, result) as
&dyn Value))])
});
} else { ; }
};debug!(
610 "receiver_for_self_ty({:?}, {:?}, {:?}) = {:?}",
611 receiver_ty, self_ty, method_def_id, result
612 );
613 result
614}
615
616fn receiver_is_dispatchable<'tcx>(
666 tcx: TyCtxt<'tcx>,
667 method: ty::AssocItem,
668 receiver_ty: Ty<'tcx>,
669) -> bool {
670 {
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/traits/dyn_compatibility.rs:670",
"rustc_trait_selection::traits::dyn_compatibility",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/traits/dyn_compatibility.rs"),
::tracing_core::__macro_support::Option::Some(670u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::traits::dyn_compatibility"),
::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!("receiver_is_dispatchable: method = {0:?}, receiver_ty = {1:?}",
method, receiver_ty) as &dyn Value))])
});
} else { ; }
};debug!("receiver_is_dispatchable: method = {:?}, receiver_ty = {:?}", method, receiver_ty);
671
672 if receiver_ty.references_error() {
673 return false;
674 }
675
676 let (Some(unsize_did), Some(dispatch_from_dyn_did)) =
677 (tcx.lang_items().unsize_trait(), tcx.lang_items().dispatch_from_dyn_trait())
678 else {
679 {
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/traits/dyn_compatibility.rs:679",
"rustc_trait_selection::traits::dyn_compatibility",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/traits/dyn_compatibility.rs"),
::tracing_core::__macro_support::Option::Some(679u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::traits::dyn_compatibility"),
::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!("receiver_is_dispatchable: Missing `Unsize` or `DispatchFromDyn` traits")
as &dyn Value))])
});
} else { ; }
};debug!("receiver_is_dispatchable: Missing `Unsize` or `DispatchFromDyn` traits");
680 return false;
681 };
682
683 let unsized_self_ty: Ty<'tcx> =
686 Ty::new_param(tcx, u32::MAX, rustc_span::sym::RustaceansAreAwesome);
687
688 let unsized_receiver_ty =
690 receiver_for_self_ty(tcx, receiver_ty, unsized_self_ty, method.def_id);
691
692 let param_env = {
695 let mut predicates: Vec<_> = tcx
708 .predicates_of(method.def_id)
709 .instantiate_identity(tcx)
710 .predicates
711 .into_iter()
712 .map(Unnormalized::skip_norm_wip)
713 .collect();
714
715 let unsize_predicate =
717 ty::TraitRef::new(tcx, unsize_did, [tcx.types.self_param, unsized_self_ty]);
718 predicates.push(unsize_predicate.upcast(tcx));
719
720 let trait_def_id = method.trait_container(tcx).unwrap();
722 let args = GenericArgs::for_item(tcx, trait_def_id, |param, _| {
723 if param.index == 0 { unsized_self_ty.into() } else { tcx.mk_param_from_def(param) }
724 });
725 let trait_predicate = ty::TraitRef::new_from_args(tcx, trait_def_id, args);
726 predicates.push(trait_predicate.upcast(tcx));
727
728 let meta_sized_predicate = {
729 let meta_sized_did = tcx.require_lang_item(LangItem::MetaSized, DUMMY_SP);
730 ty::TraitRef::new(tcx, meta_sized_did, [unsized_self_ty]).upcast(tcx)
731 };
732 predicates.push(meta_sized_predicate);
733
734 normalize_param_env_or_error(
735 tcx,
736 ty::ParamEnv::new(tcx.mk_clauses(&predicates)),
737 ObligationCause::dummy_with_span(tcx.def_span(method.def_id)),
738 )
739 };
740
741 let obligation = {
743 let predicate =
744 ty::TraitRef::new(tcx, dispatch_from_dyn_did, [receiver_ty, unsized_receiver_ty]);
745
746 Obligation::new(tcx, ObligationCause::dummy(), param_env, predicate)
747 };
748
749 let infcx = tcx.infer_ctxt().build(TypingMode::non_body_analysis());
750 infcx.predicate_must_hold_modulo_regions(&obligation)
752}
753
754#[derive(#[automatically_derived]
impl ::core::marker::Copy for AllowSelfProjections { }Copy, #[automatically_derived]
impl ::core::clone::Clone for AllowSelfProjections {
#[inline]
fn clone(&self) -> AllowSelfProjections { *self }
}Clone)]
755enum AllowSelfProjections {
756 Yes,
757 No,
758}
759
760fn contains_illegal_self_type_reference<'tcx, T: TypeVisitable<TyCtxt<'tcx>>>(
802 tcx: TyCtxt<'tcx>,
803 trait_def_id: DefId,
804 value: T,
805 allow_self_projections: AllowSelfProjections,
806) -> bool {
807 value
808 .visit_with(&mut IllegalSelfTypeVisitor {
809 tcx,
810 trait_def_id,
811 supertraits: None,
812 allow_self_projections,
813 })
814 .is_break()
815}
816
817struct IllegalSelfTypeVisitor<'tcx> {
818 tcx: TyCtxt<'tcx>,
819 trait_def_id: DefId,
820 supertraits: Option<Vec<ty::TraitRef<'tcx>>>,
821 allow_self_projections: AllowSelfProjections,
822}
823
824impl<'tcx> IllegalSelfTypeVisitor<'tcx> {
825 fn is_supertrait_of_current_trait(&mut self, trait_ref: ty::TraitRef<'tcx>) -> bool {
826 let supertraits = self.supertraits.get_or_insert_with(|| {
828 util::supertraits(
829 self.tcx,
830 ty::Binder::dummy(ty::TraitRef::identity(self.tcx, self.trait_def_id)),
831 )
832 .map(|trait_ref| {
833 self.tcx.erase_and_anonymize_regions(
834 self.tcx.instantiate_bound_regions_with_erased(trait_ref),
835 )
836 })
837 .collect()
838 });
839
840 let trait_ref = trait_ref
846 .fold_with(&mut EraseEscapingBoundRegions { tcx: self.tcx, binder: ty::INNERMOST });
847 supertraits.contains(&trait_ref)
848 }
849}
850
851impl<'tcx> TypeVisitor<TyCtxt<'tcx>> for IllegalSelfTypeVisitor<'tcx> {
852 type Result = ControlFlow<()>;
853
854 fn visit_ty(&mut self, ty: Ty<'tcx>) -> Self::Result {
855 match ty.kind() {
856 ty::Param(_) => {
857 if ty == self.tcx.types.self_param {
858 ControlFlow::Break(())
859 } else {
860 ControlFlow::Continue(())
861 }
862 }
863 ty::Alias(ty::AliasTy { kind: ty::Projection { def_id }, .. })
864 if self.tcx.is_impl_trait_in_trait(*def_id) =>
865 {
866 ControlFlow::Continue(())
868 }
869 ty::Alias(proj @ ty::AliasTy { kind: ty::Projection { .. }, .. }) => {
870 match self.allow_self_projections {
871 AllowSelfProjections::Yes => {
872 if self.is_supertrait_of_current_trait(proj.trait_ref(self.tcx)) {
874 ControlFlow::Continue(())
875 } else {
876 ty.super_visit_with(self)
877 }
878 }
879 AllowSelfProjections::No => ty.super_visit_with(self),
880 }
881 }
882 _ => ty.super_visit_with(self),
883 }
884 }
885
886 fn visit_const(&mut self, ct: ty::Const<'tcx>) -> Self::Result {
887 let ct = self.tcx.expand_abstract_consts(ct);
888
889 match ct.kind() {
890 ty::ConstKind::Unevaluated(proj) if self.tcx.features().min_generic_const_args() => {
891 match self.allow_self_projections {
892 AllowSelfProjections::Yes
893 if #[allow(non_exhaustive_omitted_patterns)] match self.tcx.def_kind(proj.def) {
DefKind::AssocConst { .. } => true,
_ => false,
}matches!(self.tcx.def_kind(proj.def), DefKind::AssocConst { .. })
894 && let trait_def_id = self.tcx.parent(proj.def)
895 && self.tcx.def_kind(trait_def_id) == DefKind::Trait =>
896 {
897 let trait_ref = ty::TraitRef::from_assoc(self.tcx, trait_def_id, proj.args);
898
899 if self.is_supertrait_of_current_trait(trait_ref) {
901 ControlFlow::Continue(())
902 } else {
903 ct.super_visit_with(self)
904 }
905 }
906 _ => ct.super_visit_with(self),
907 }
908 }
909 _ => ct.super_visit_with(self),
910 }
911 }
912}
913
914struct EraseEscapingBoundRegions<'tcx> {
915 tcx: TyCtxt<'tcx>,
916 binder: ty::DebruijnIndex,
917}
918
919impl<'tcx> TypeFolder<TyCtxt<'tcx>> for EraseEscapingBoundRegions<'tcx> {
920 fn cx(&self) -> TyCtxt<'tcx> {
921 self.tcx
922 }
923
924 fn fold_binder<T>(&mut self, t: ty::Binder<'tcx, T>) -> ty::Binder<'tcx, T>
925 where
926 T: TypeFoldable<TyCtxt<'tcx>>,
927 {
928 self.binder.shift_in(1);
929 let result = t.super_fold_with(self);
930 self.binder.shift_out(1);
931 result
932 }
933
934 fn fold_region(&mut self, r: ty::Region<'tcx>) -> ty::Region<'tcx> {
935 if let ty::ReBound(ty::BoundVarIndexKind::Bound(debruijn), _) = r.kind()
936 && debruijn < self.binder
937 {
938 r
939 } else {
940 self.tcx.lifetimes.re_erased
941 }
942 }
943}
944
945fn contains_illegal_impl_trait_in_trait<'tcx>(
946 tcx: TyCtxt<'tcx>,
947 fn_def_id: DefId,
948 ty: ty::Binder<'tcx, Ty<'tcx>>,
949) -> Option<MethodViolation> {
950 let ty = tcx.liberate_late_bound_regions(fn_def_id, ty);
951
952 if tcx.asyncness(fn_def_id).is_async() {
953 Some(MethodViolation::AsyncFn)
955 } else {
956 ty.visit_with(&mut IllegalRpititVisitor { tcx, allowed: None }).break_value()
957 }
958}
959
960struct IllegalRpititVisitor<'tcx> {
961 tcx: TyCtxt<'tcx>,
962 allowed: Option<ty::AliasTy<'tcx>>,
963}
964
965impl<'tcx> TypeVisitor<TyCtxt<'tcx>> for IllegalRpititVisitor<'tcx> {
966 type Result = ControlFlow<MethodViolation>;
967
968 fn visit_ty(&mut self, ty: Ty<'tcx>) -> Self::Result {
969 if let ty::Alias(proj @ ty::AliasTy { kind: ty::Projection { def_id }, .. }) = *ty.kind()
970 && Some(proj) != self.allowed
971 && self.tcx.is_impl_trait_in_trait(def_id)
972 {
973 ControlFlow::Break(MethodViolation::ReferencesImplTraitInTrait(
974 self.tcx.def_span(def_id),
975 ))
976 } else {
977 ty.super_visit_with(self)
978 }
979 }
980}
981
982pub(crate) fn provide(providers: &mut Providers) {
983 *providers = Providers {
984 dyn_compatibility_violations,
985 is_dyn_compatible,
986 generics_require_sized_self,
987 ..*providers
988 };
989}