Skip to content

Commit 07fc644

Browse files
Do not require associated types with Self: Sized to uphold bounds when confirming object candidate
1 parent 7a6b52b commit 07fc644

File tree

7 files changed

+54
-28
lines changed

7 files changed

+54
-28
lines changed

compiler/rustc_trait_selection/src/solve/assembly/structural_traits.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -354,9 +354,9 @@ pub(in crate::solve) fn predicates_for_object_candidate<'tcx>(
354354
// FIXME(associated_const_equality): Also add associated consts to
355355
// the requirements here.
356356
if item.kind == ty::AssocKind::Type {
357-
// RPITITs are not checked here, since they are not (currently) object-safe
358-
// and cannot be named from a non-`Self: Sized` method.
359-
if item.is_impl_trait_in_trait() {
357+
// associated types that require `Self: Sized` do not show up in the built-in
358+
// implementation of `Trait for dyn Trait`, and can be dropped here.
359+
if tcx.generics_require_sized_self(item.def_id) {
360360
continue;
361361
}
362362

compiler/rustc_trait_selection/src/traits/select/confirmation.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -535,9 +535,9 @@ impl<'cx, 'tcx> SelectionContext<'cx, 'tcx> {
535535
let assoc_types: Vec<_> = tcx
536536
.associated_items(trait_predicate.def_id())
537537
.in_definition_order()
538-
// RPITITs are not checked here, since they are not (currently) object-safe
539-
// and cannot be named from a non-`Self: Sized` method.
540-
.filter(|item| !item.is_impl_trait_in_trait())
538+
// Associated types that require `Self: Sized` do not show up in the built-in
539+
// implementation of `Trait for dyn Trait`, and can be dropped here.
540+
.filter(|item| !tcx.generics_require_sized_self(item.def_id))
541541
.filter_map(
542542
|item| if item.kind == ty::AssocKind::Type { Some(item.def_id) } else { None },
543543
)

tests/ui/impl-trait/in-trait/object-safety.rs

+1
Original file line numberDiff line numberDiff line change
@@ -19,4 +19,5 @@ fn main() {
1919
//~| ERROR the trait `Foo` cannot be made into an object
2020
let s = i.baz();
2121
//~^ ERROR the trait `Foo` cannot be made into an object
22+
//~| ERROR the trait `Foo` cannot be made into an object
2223
}

tests/ui/impl-trait/in-trait/object-safety.stderr

+16-1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,21 @@ LL | fn baz(&self) -> impl Debug;
1313
| ^^^^^^^^^^ ...because method `baz` references an `impl Trait` type in its return type
1414
= help: consider moving `baz` to another trait
1515

16+
error[E0038]: the trait `Foo` cannot be made into an object
17+
--> $DIR/object-safety.rs:20:15
18+
|
19+
LL | let s = i.baz();
20+
| ^^^ `Foo` cannot be made into an object
21+
|
22+
note: for a trait to be "object safe" 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>
23+
--> $DIR/object-safety.rs:7:22
24+
|
25+
LL | trait Foo {
26+
| --- this trait cannot be made into an object...
27+
LL | fn baz(&self) -> impl Debug;
28+
| ^^^^^^^^^^ ...because method `baz` references an `impl Trait` type in its return type
29+
= help: consider moving `baz` to another trait
30+
1631
error[E0038]: the trait `Foo` cannot be made into an object
1732
--> $DIR/object-safety.rs:20:13
1833
|
@@ -44,6 +59,6 @@ LL | fn baz(&self) -> impl Debug;
4459
= help: consider moving `baz` to another trait
4560
= note: required for the cast from `Box<u32>` to `Box<dyn Foo>`
4661

47-
error: aborting due to 3 previous errors
62+
error: aborting due to 4 previous errors
4863

4964
For more information about this error, try `rustc --explain E0038`.

tests/ui/object-safety/assoc_type_bounds_sized_used.rs

+2-4
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
1-
//! This test checks that even if some associated types have
2-
//! `where Self: Sized` bounds, those without still need to be
3-
//! mentioned in trait objects.
1+
//! This test checks that associated types with `Self: Sized` cannot be projected
2+
//! from a `dyn Trait`.
43
54
trait Bop {
65
type Bar: Default
@@ -16,5 +15,4 @@ fn bop<T: Bop + ?Sized>() {
1615

1716
fn main() {
1817
bop::<dyn Bop>();
19-
//~^ ERROR: the size for values of type `dyn Bop` cannot be known at compilation time
2018
}
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
error[E0599]: the function or associated item `default` exists for associated type `<T as Bop>::Bar`, but its trait bounds were not satisfied
2-
--> $DIR/assoc_type_bounds_sized_used.rs:12:30
2+
--> $DIR/assoc_type_bounds_sized_used.rs:11:30
33
|
44
LL | let _ = <T as Bop>::Bar::default();
55
| ^^^^^^^ function or associated item cannot be called on `<T as Bop>::Bar` due to unsatisfied trait bounds
@@ -13,15 +13,15 @@ LL | fn bop<T: Bop + ?Sized>() where T: Sized {
1313
| ++++++++++++++
1414

1515
error[E0277]: the size for values of type `T` cannot be known at compilation time
16-
--> $DIR/assoc_type_bounds_sized_used.rs:12:14
16+
--> $DIR/assoc_type_bounds_sized_used.rs:11:14
1717
|
1818
LL | fn bop<T: Bop + ?Sized>() {
1919
| - this type parameter needs to be `Sized`
2020
LL | let _ = <T as Bop>::Bar::default();
2121
| ^ doesn't have a size known at compile-time
2222
|
2323
note: required by a bound in `Bop::Bar`
24-
--> $DIR/assoc_type_bounds_sized_used.rs:8:15
24+
--> $DIR/assoc_type_bounds_sized_used.rs:7:15
2525
|
2626
LL | type Bar: Default
2727
| --- required by a bound in this associated type
@@ -34,20 +34,7 @@ LL - fn bop<T: Bop + ?Sized>() {
3434
LL + fn bop<T: Bop>() {
3535
|
3636

37-
error[E0277]: the size for values of type `dyn Bop` cannot be known at compilation time
38-
--> $DIR/assoc_type_bounds_sized_used.rs:18:11
39-
|
40-
LL | bop::<dyn Bop>();
41-
| ^^^^^^^ doesn't have a size known at compile-time
42-
|
43-
= help: the trait `Sized` is not implemented for `dyn Bop`
44-
note: required by a bound in `bop`
45-
--> $DIR/assoc_type_bounds_sized_used.rs:11:11
46-
|
47-
LL | fn bop<T: Bop + ?Sized>() {
48-
| ^^^ required by this bound in `bop`
49-
50-
error: aborting due to 3 previous errors
37+
error: aborting due to 2 previous errors
5138

5239
Some errors have detailed explanations: E0277, E0599.
5340
For more information about an error, try `rustc --explain E0277`.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
// check-pass
2+
// revisions: current next
3+
//[next] compile-flags: -Ztrait-solver=next
4+
5+
trait Foo {
6+
type Bar<'a>
7+
where
8+
Self: Sized;
9+
10+
fn test(&self);
11+
}
12+
13+
impl Foo for () {
14+
type Bar<'a> = () where Self: Sized;
15+
16+
fn test(&self) {}
17+
}
18+
19+
fn test(x: &dyn Foo) {
20+
x.test();
21+
}
22+
23+
fn main() {
24+
test(&());
25+
}

0 commit comments

Comments
 (0)