Skip to content

Commit 02d748e

Browse files
committed
Refactor dyn-compatibility error and suggestions
This CL makes a number of small changes to dyn compatibility errors: - "object safety" has been renamed to "dyn-compatibility" throughout - "Convert to enum" suggestions are no longer generated when there exists a type-generic impl of the trait or an impl for `dyn OtherTrait` - Several error messages are reorganized for user readability Additionally, the dyn compatibility error creation code has been split out into functions. cc #132713 cc #133267
1 parent f2abf82 commit 02d748e

File tree

170 files changed

+1365
-1071
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

170 files changed

+1365
-1071
lines changed

compiler/rustc_feature/src/removed.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -159,7 +159,7 @@ declare_features! (
159159
/// then removed. But there was no utility storing it separately, so now
160160
/// it's in this list.
161161
(removed, no_stack_check, "1.0.0", None, None),
162-
/// Allows making `dyn Trait` well-formed even if `Trait` is not dyn-compatible (object safe).
162+
/// Allows making `dyn Trait` well-formed even if `Trait` is not dyn compatible (object safe).
163163
/// Renamed to `dyn_compatible_for_dispatch`.
164164
(removed, object_safe_for_dispatch, "1.83.0", Some(43561),
165165
Some("renamed to `dyn_compatible_for_dispatch`")),

compiler/rustc_feature/src/unstable.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -271,7 +271,7 @@ declare_features! (
271271
(unstable, doc_notable_trait, "1.52.0", Some(45040)),
272272
/// Allows using the `may_dangle` attribute (RFC 1327).
273273
(unstable, dropck_eyepatch, "1.10.0", Some(34761)),
274-
/// Allows making `dyn Trait` well-formed even if `Trait` is not dyn-compatible[^1].
274+
/// Allows making `dyn Trait` well-formed even if `Trait` is not dyn compatible[^1].
275275
/// In that case, `dyn Trait: Trait` does not hold. Moreover, coercions and
276276
/// casts in safe Rust to `dyn Trait` for such a `Trait` is also forbidden.
277277
///

compiler/rustc_hir_analysis/src/coherence/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -184,7 +184,7 @@ fn check_object_overlap<'tcx>(
184184
// check for overlap with the automatic `impl Trait for dyn Trait`
185185
if let ty::Dynamic(data, ..) = trait_ref.self_ty().kind() {
186186
// This is something like `impl Trait1 for Trait2`. Illegal if
187-
// Trait1 is a supertrait of Trait2 or Trait2 is not dyn-compatible.
187+
// Trait1 is a supertrait of Trait2 or Trait2 is not dyn compatible.
188188

189189
let component_def_ids = data.iter().flat_map(|predicate| {
190190
match predicate.skip_binder() {

compiler/rustc_hir_analysis/src/hir_ty_lowering/dyn_compatibility.rs

+6-11
Original file line numberDiff line numberDiff line change
@@ -103,17 +103,12 @@ impl<'tcx> dyn HirTyLowerer<'tcx> + '_ {
103103
// most importantly, that the supertraits don't contain `Self`,
104104
// to avoid ICEs.
105105
for item in &regular_traits {
106-
let violations =
107-
hir_ty_lowering_dyn_compatibility_violations(tcx, item.trait_ref().def_id());
106+
let item_def_id = item.trait_ref().def_id();
107+
let violations = hir_ty_lowering_dyn_compatibility_violations(tcx, item_def_id);
108108
if !violations.is_empty() {
109-
let reported = report_dyn_incompatibility(
110-
tcx,
111-
span,
112-
Some(hir_id),
113-
item.trait_ref().def_id(),
114-
&violations,
115-
)
116-
.emit();
109+
let reported =
110+
report_dyn_incompatibility(tcx, span, Some(hir_id), item_def_id, &violations)
111+
.emit();
117112
return Ty::new_error(tcx, reported);
118113
}
119114
}
@@ -276,7 +271,7 @@ impl<'tcx> dyn HirTyLowerer<'tcx> + '_ {
276271
self.dcx(),
277272
i.bottom().1,
278273
E0038,
279-
"the {} `{}` cannot be made into an object",
274+
"the {} `{}` is not dyn compatible",
280275
tcx.def_descr(def_id),
281276
tcx.item_name(def_id),
282277
)

compiler/rustc_trait_selection/src/error_reporting/traits/mod.rs

+115-78
Original file line numberDiff line numberDiff line change
@@ -432,33 +432,18 @@ pub fn report_dyn_incompatibility<'tcx>(
432432
hir::Node::Item(item) => Some(item.ident.span),
433433
_ => None,
434434
});
435+
435436
let mut err = struct_span_code_err!(
436437
tcx.dcx(),
437438
span,
438439
E0038,
439-
"the trait `{}` cannot be made into an object",
440+
"the trait `{}` is not dyn compatible",
440441
trait_str
441442
);
442-
err.span_label(span, format!("`{trait_str}` cannot be made into an object"));
443-
444-
if let Some(hir_id) = hir_id
445-
&& let hir::Node::Ty(ty) = tcx.hir_node(hir_id)
446-
&& let hir::TyKind::TraitObject([trait_ref, ..], ..) = ty.kind
447-
{
448-
let mut hir_id = hir_id;
449-
while let hir::Node::Ty(ty) = tcx.parent_hir_node(hir_id) {
450-
hir_id = ty.hir_id;
451-
}
452-
if tcx.parent_hir_node(hir_id).fn_sig().is_some() {
453-
// Do not suggest `impl Trait` when dealing with things like super-traits.
454-
err.span_suggestion_verbose(
455-
ty.span.until(trait_ref.span),
456-
"consider using an opaque type instead",
457-
"impl ",
458-
Applicability::MaybeIncorrect,
459-
);
460-
}
461-
}
443+
err.span_label(span, format!("`{trait_str}` is not dyn compatible"));
444+
445+
attempt_dyn_to_impl_suggestion(tcx, hir_id, &mut err);
446+
462447
let mut reported_violations = FxIndexSet::default();
463448
let mut multi_span = vec![];
464449
let mut messages = vec![];
@@ -473,7 +458,7 @@ pub fn report_dyn_incompatibility<'tcx>(
473458
if reported_violations.insert(violation.clone()) {
474459
let spans = violation.spans();
475460
let msg = if trait_span.is_none() || spans.is_empty() {
476-
format!("the trait cannot be made into an object because {}", violation.error_msg())
461+
format!("the trait is not dyn compatible because {}", violation.error_msg())
477462
} else {
478463
format!("...because {}", violation.error_msg())
479464
};
@@ -490,24 +475,20 @@ pub fn report_dyn_incompatibility<'tcx>(
490475
let has_multi_span = !multi_span.is_empty();
491476
let mut note_span = MultiSpan::from_spans(multi_span.clone());
492477
if let (Some(trait_span), true) = (trait_span, has_multi_span) {
493-
note_span.push_span_label(trait_span, "this trait cannot be made into an object...");
478+
note_span.push_span_label(trait_span, "this trait is not dyn compatible...");
494479
}
495480
for (span, msg) in iter::zip(multi_span, messages) {
496481
note_span.push_span_label(span, msg);
497482
}
498483
// FIXME(dyn_compat_renaming): Update the URL.
499484
err.span_note(
500485
note_span,
501-
"for a trait to be \"dyn-compatible\" it needs to allow building a vtable to allow the call \
502-
to be resolvable dynamically; for more information visit \
503-
<https://doc.rust-lang.org/reference/items/traits.html#object-safety>",
486+
"for a trait to be dyn compatible it needs to allow building a vtable\n\
487+
for more information, visit <https://doc.rust-lang.org/reference/items/traits.html#object-safety>",
504488
);
505489

506490
// Only provide the help if its a local trait, otherwise it's not actionable.
507491
if trait_span.is_some() {
508-
let mut reported_violations: Vec<_> = reported_violations.into_iter().collect();
509-
reported_violations.sort();
510-
511492
let mut potential_solutions: Vec<_> =
512493
reported_violations.into_iter().map(|violation| violation.solution()).collect();
513494
potential_solutions.sort();
@@ -518,68 +499,124 @@ pub fn report_dyn_incompatibility<'tcx>(
518499
}
519500
}
520501

502+
attempt_dyn_to_enum_suggestion(tcx, trait_def_id, &*trait_str, &mut err);
503+
504+
err
505+
}
506+
507+
/// Attempt to suggest converting the `dyn Trait` argument to an enumeration
508+
/// over the types that implement `Trait`.
509+
fn attempt_dyn_to_enum_suggestion(
510+
tcx: TyCtxt<'_>,
511+
trait_def_id: DefId,
512+
trait_str: &str,
513+
err: &mut Diag<'_>,
514+
) {
521515
let impls_of = tcx.trait_impls_of(trait_def_id);
522-
let impls = if impls_of.blanket_impls().is_empty() {
523-
impls_of
524-
.non_blanket_impls()
525-
.values()
526-
.flatten()
527-
.filter(|def_id| {
528-
!matches!(tcx.type_of(*def_id).instantiate_identity().kind(), ty::Dynamic(..))
529-
})
530-
.collect::<Vec<_>>()
531-
} else {
532-
vec![]
516+
517+
// Don't suggest converting to an enum if there are any blanket impls.
518+
if !impls_of.blanket_impls().is_empty() {
519+
return;
520+
}
521+
522+
let concrete_impls = {
523+
let mut concrete_impls = Vec::new();
524+
for impl_id in impls_of.non_blanket_impls().values().flatten() {
525+
// Don't suggest converting to enum if there are any non-lifetime generics.
526+
if has_non_lifetime_generics(tcx, *impl_id) {
527+
return;
528+
}
529+
530+
let impl_type = tcx.type_of(*impl_id).instantiate_identity();
531+
532+
// Don't suggest converting to enum if there are any
533+
// `impl Trait for dyn OtherTrait`
534+
if let ty::Dynamic(..) = impl_type.kind() {
535+
return;
536+
}
537+
538+
concrete_impls.push(impl_type);
539+
}
540+
concrete_impls
533541
};
534-
let externally_visible = if !impls.is_empty()
535-
&& let Some(def_id) = trait_def_id.as_local()
542+
if concrete_impls.is_empty() {
543+
return;
544+
}
545+
546+
const MAX_IMPLS_TO_SUGGEST_CONVERTING_TO_ENUM: usize = 9;
547+
if concrete_impls.len() > MAX_IMPLS_TO_SUGGEST_CONVERTING_TO_ENUM {
548+
return;
549+
}
550+
551+
let externally_visible = if let Some(def_id) = trait_def_id.as_local() {
536552
// We may be executing this during typeck, which would result in cycle
537553
// if we used effective_visibilities query, which looks into opaque types
538554
// (and therefore calls typeck).
539-
&& tcx.resolutions(()).effective_visibilities.is_exported(def_id)
540-
{
541-
true
555+
tcx.resolutions(()).effective_visibilities.is_exported(def_id)
542556
} else {
543557
false
544558
};
545-
match &impls[..] {
546-
[] => {}
547-
_ if impls.len() > 9 => {}
548-
[only] if externally_visible => {
549-
err.help(with_no_trimmed_paths!(format!(
550-
"only type `{}` is seen to implement the trait in this crate, consider using it \
551-
directly instead",
552-
tcx.type_of(*only).instantiate_identity(),
553-
)));
554-
}
555-
[only] => {
556-
err.help(with_no_trimmed_paths!(format!(
557-
"only type `{}` implements the trait, consider using it directly instead",
558-
tcx.type_of(*only).instantiate_identity(),
559-
)));
560-
}
561-
impls => {
562-
let types = impls
563-
.iter()
564-
.map(|t| {
565-
with_no_trimmed_paths!(format!(" {}", tcx.type_of(*t).instantiate_identity(),))
566-
})
567-
.collect::<Vec<_>>();
568-
err.help(format!(
569-
"the following types implement the trait, consider defining an enum where each \
570-
variant holds one of these types, implementing `{}` for this new enum and using \
571-
it instead:\n{}",
572-
trait_str,
573-
types.join("\n"),
574-
));
575-
}
559+
560+
if let [only_impl] = &concrete_impls[..] {
561+
let within = if externally_visible { " within this crate" } else { "" };
562+
err.help(with_no_trimmed_paths!(format!(
563+
"only type `{only_impl}` implements `{trait_str}`{within}. \
564+
Consider using it directly instead."
565+
)));
566+
} else {
567+
let types = concrete_impls
568+
.iter()
569+
.map(|t| with_no_trimmed_paths!(format!(" {}", t)))
570+
.collect::<Vec<String>>()
571+
.join("\n");
572+
573+
err.help(format!(
574+
"the following types implement `{trait_str}`:\n\
575+
{types}\n\
576+
Consider defining an enum where each variant holds one of these types,\n\
577+
implementing `{trait_str}` for this new enum and using it instead.",
578+
));
576579
}
580+
577581
if externally_visible {
578582
err.note(format!(
579-
"`{trait_str}` can be implemented in other crates; if you want to support your users \
583+
"`{trait_str}` may be implemented in other crates; if you want to support your users \
580584
passing their own types here, you can't refer to a specific type",
581585
));
582586
}
587+
}
583588

584-
err
589+
fn has_non_lifetime_generics(tcx: TyCtxt<'_>, def_id: DefId) -> bool {
590+
tcx.generics_of(def_id).own_params.iter().any(|param| match param.kind {
591+
ty::GenericParamDefKind::Type { .. } | ty::GenericParamDefKind::Const { .. } => true,
592+
ty::GenericParamDefKind::Lifetime => false,
593+
})
594+
}
595+
596+
/// Attempt to suggest that a `dyn Trait` argument or return type be converted
597+
/// to use `impl Trait`.
598+
fn attempt_dyn_to_impl_suggestion(tcx: TyCtxt<'_>, hir_id: Option<hir::HirId>, err: &mut Diag<'_>) {
599+
let Some(hir_id) = hir_id else { return };
600+
let hir::Node::Ty(ty) = tcx.hir_node(hir_id) else { return };
601+
let hir::TyKind::TraitObject([trait_ref, ..], ..) = ty.kind else { return };
602+
603+
// Only suggest converting `dyn` to `impl` if we're in a function signature.
604+
// This ensures that we don't suggest converting e.g.
605+
// `type Alias = Box<dyn DynIncompatibleTrait>;` to
606+
// `type Alias = Box<impl DynIncompatibleTrait>;`
607+
let Some((_id, first_non_type_parent_node)) =
608+
tcx.hir().parent_iter(hir_id).find(|(_id, node)| !matches!(node, hir::Node::Ty(_)))
609+
else {
610+
return;
611+
};
612+
if first_non_type_parent_node.fn_sig().is_none() {
613+
return;
614+
}
615+
616+
err.span_suggestion_verbose(
617+
ty.span.until(trait_ref.span),
618+
"consider using an opaque type instead",
619+
"impl ",
620+
Applicability::MaybeIncorrect,
621+
);
585622
}

compiler/rustc_trait_selection/src/traits/dyn_compatibility.rs

-1
Original file line numberDiff line numberDiff line change
@@ -52,7 +52,6 @@ fn dyn_compatibility_violations(
5252
) -> &'_ [DynCompatibilityViolation] {
5353
debug_assert!(tcx.generics_of(trait_def_id).has_self);
5454
debug!("dyn_compatibility_violations: {:?}", trait_def_id);
55-
5655
tcx.arena.alloc_from_iter(
5756
tcx.supertrait_def_ids(trait_def_id)
5857
.flat_map(|def_id| dyn_compatibility_violations_for_trait(tcx, def_id)),

tests/rustdoc-ui/invalid_const_in_lifetime_position.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,4 +8,4 @@ fn f<'a>(arg : Box<dyn X<Y<1> = &'a ()>>) {}
88
//~| ERROR associated type takes 0 generic arguments but 1 generic argument
99
//~| ERROR associated type takes 1 lifetime argument but 0 lifetime arguments
1010
//~| ERROR associated type takes 0 generic arguments but 1 generic argument
11-
//~| ERROR trait `X` cannot be made into an object
11+
//~| ERROR trait `X` is not dyn compatible

tests/rustdoc-ui/invalid_const_in_lifetime_position.stderr

+5-4
Original file line numberDiff line numberDiff line change
@@ -92,17 +92,18 @@ LL | type Y<'a>;
9292
| ^
9393
= note: duplicate diagnostic emitted due to `-Z deduplicate-diagnostics=no`
9494

95-
error[E0038]: the trait `X` cannot be made into an object
95+
error[E0038]: the trait `X` is not dyn compatible
9696
--> $DIR/invalid_const_in_lifetime_position.rs:4:20
9797
|
9898
LL | fn f<'a>(arg : Box<dyn X<Y<1> = &'a ()>>) {}
99-
| ^^^^^^^^^^^^^^^^^^^^ `X` cannot be made into an object
99+
| ^^^^^^^^^^^^^^^^^^^^ `X` is not dyn compatible
100100
|
101-
note: for a trait to be "dyn-compatible" it needs to allow building a vtable to allow the call to be resolvable dynamically; for more information visit <https://doc.rust-lang.org/reference/items/traits.html#object-safety>
101+
note: for a trait to be dyn compatible it needs to allow building a vtable
102+
for more information, visit <https://doc.rust-lang.org/reference/items/traits.html#object-safety>
102103
--> $DIR/invalid_const_in_lifetime_position.rs:2:10
103104
|
104105
LL | trait X {
105-
| - this trait cannot be made into an object...
106+
| - this trait is not dyn compatible...
106107
LL | type Y<'a>;
107108
| ^ ...because it contains the generic associated type `Y`
108109
= help: consider moving `Y` to another trait

tests/rustdoc-ui/issues/issue-105742.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -4,8 +4,8 @@ use std::ops::Index;
44
pub fn next<'a, T>(s: &'a mut dyn SVec<Item = T, Output = T>) {
55
//~^ expected 1 lifetime argument
66
//~| expected 1 generic argument
7-
//~| the trait `SVec` cannot be made into an object
8-
//~| `SVec` cannot be made into an object
7+
//~| the trait `SVec` is not dyn compatible
8+
//~| `SVec` is not dyn compatible
99
//~| missing generics for associated type `SVec::Item`
1010
//~| missing generics for associated type `SVec::Item`
1111
let _ = s;

tests/rustdoc-ui/issues/issue-105742.stderr

+5-4
Original file line numberDiff line numberDiff line change
@@ -294,19 +294,20 @@ help: add missing generic argument
294294
LL | Output = <Self as SVec>::Item> as SVec>::Item<T>,
295295
| +++
296296

297-
error[E0038]: the trait `SVec` cannot be made into an object
297+
error[E0038]: the trait `SVec` is not dyn compatible
298298
--> $DIR/issue-105742.rs:4:31
299299
|
300300
LL | pub fn next<'a, T>(s: &'a mut dyn SVec<Item = T, Output = T>) {
301-
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ `SVec` cannot be made into an object
301+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ `SVec` is not dyn compatible
302302
|
303-
note: for a trait to be "dyn-compatible" it needs to allow building a vtable to allow the call to be resolvable dynamically; for more information visit <https://doc.rust-lang.org/reference/items/traits.html#object-safety>
303+
note: for a trait to be dyn compatible it needs to allow building a vtable
304+
for more information, visit <https://doc.rust-lang.org/reference/items/traits.html#object-safety>
304305
--> $DIR/issue-105742.rs:14:17
305306
|
306307
LL | pub trait SVec: Index<
307308
| ____________----__^
308309
| | |
309-
| | this trait cannot be made into an object...
310+
| | this trait is not dyn compatible...
310311
LL | | <Self as SVec>::Item,
311312
LL | |
312313
LL | |

tests/ui/associated-consts/associated-const-in-trait.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -5,9 +5,9 @@ trait Trait {
55
}
66

77
impl dyn Trait {
8-
//~^ ERROR the trait `Trait` cannot be made into an object [E0038]
8+
//~^ ERROR the trait `Trait` is not dyn compatible [E0038]
99
const fn n() -> usize { Self::N }
10-
//~^ ERROR the trait `Trait` cannot be made into an object [E0038]
10+
//~^ ERROR the trait `Trait` is not dyn compatible [E0038]
1111
}
1212

1313
fn main() {}

0 commit comments

Comments
 (0)