Auto merge of #92285 - compiler-errors:dyn-proj-bounds, r=nikomatsakis
check ~Projection~ all supertrait bounds when confirming dyn candidate I'm pretty sure Projection is the only other PredicateKind that we care about enforcing here. Fixes #80800
This commit is contained in:
commit
984204814e
@ -468,21 +468,19 @@ fn confirm_object_candidate(
|
|||||||
.predicates
|
.predicates
|
||||||
.into_iter()
|
.into_iter()
|
||||||
{
|
{
|
||||||
if let ty::PredicateKind::Trait(..) = super_trait.kind().skip_binder() {
|
let normalized_super_trait = normalize_with_depth_to(
|
||||||
let normalized_super_trait = normalize_with_depth_to(
|
self,
|
||||||
self,
|
obligation.param_env,
|
||||||
obligation.param_env,
|
obligation.cause.clone(),
|
||||||
obligation.cause.clone(),
|
obligation.recursion_depth + 1,
|
||||||
obligation.recursion_depth + 1,
|
super_trait,
|
||||||
super_trait,
|
&mut nested,
|
||||||
&mut nested,
|
);
|
||||||
);
|
nested.push(Obligation::new(
|
||||||
nested.push(Obligation::new(
|
obligation.cause.clone(),
|
||||||
obligation.cause.clone(),
|
obligation.param_env,
|
||||||
obligation.param_env,
|
normalized_super_trait,
|
||||||
normalized_super_trait,
|
));
|
||||||
));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
let assoc_types: Vec<_> = tcx
|
let assoc_types: Vec<_> = tcx
|
||||||
|
24
src/test/ui/traits/object/enforce-supertrait-projection.rs
Normal file
24
src/test/ui/traits/object/enforce-supertrait-projection.rs
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
trait SuperTrait {
|
||||||
|
type A;
|
||||||
|
type B;
|
||||||
|
}
|
||||||
|
|
||||||
|
trait Trait: SuperTrait<A = <Self as SuperTrait>::B> {}
|
||||||
|
|
||||||
|
fn transmute<A, B>(x: A) -> B {
|
||||||
|
foo::<A, B, dyn Trait<A = A, B = B>>(x)
|
||||||
|
//~^ ERROR type mismatch resolving `<dyn Trait<A = A, B = B> as SuperTrait>::A == B`
|
||||||
|
}
|
||||||
|
|
||||||
|
fn foo<A, B, T: ?Sized>(x: T::A) -> B
|
||||||
|
where
|
||||||
|
T: Trait<B = B>,
|
||||||
|
{
|
||||||
|
x
|
||||||
|
}
|
||||||
|
|
||||||
|
static X: u8 = 0;
|
||||||
|
fn main() {
|
||||||
|
let x = transmute::<&u8, &[u8; 1_000_000]>(&X);
|
||||||
|
println!("{:?}", x[100_000]);
|
||||||
|
}
|
@ -0,0 +1,26 @@
|
|||||||
|
error[E0271]: type mismatch resolving `<dyn Trait<A = A, B = B> as SuperTrait>::A == B`
|
||||||
|
--> $DIR/enforce-supertrait-projection.rs:9:5
|
||||||
|
|
|
||||||
|
LL | fn transmute<A, B>(x: A) -> B {
|
||||||
|
| - - expected type parameter
|
||||||
|
| |
|
||||||
|
| found type parameter
|
||||||
|
LL | foo::<A, B, dyn Trait<A = A, B = B>>(x)
|
||||||
|
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ expected type parameter `B`, found type parameter `A`
|
||||||
|
|
|
||||||
|
= note: expected type parameter `B`
|
||||||
|
found type parameter `A`
|
||||||
|
= note: a type parameter was expected, but a different one was found; you might be missing a type parameter or trait bound
|
||||||
|
= note: for more information, visit https://doc.rust-lang.org/book/ch10-02-traits.html#traits-as-parameters
|
||||||
|
note: required by a bound in `foo`
|
||||||
|
--> $DIR/enforce-supertrait-projection.rs:15:8
|
||||||
|
|
|
||||||
|
LL | fn foo<A, B, T: ?Sized>(x: T::A) -> B
|
||||||
|
| --- required by a bound in this
|
||||||
|
LL | where
|
||||||
|
LL | T: Trait<B = B>,
|
||||||
|
| ^^^^^^^^^^^^ required by this bound in `foo`
|
||||||
|
|
||||||
|
error: aborting due to previous error
|
||||||
|
|
||||||
|
For more information about this error, try `rustc --explain E0271`.
|
@ -0,0 +1,11 @@
|
|||||||
|
error: lifetime may not live long enough
|
||||||
|
--> $DIR/supertrait-lifetime-bound.rs:10:5
|
||||||
|
|
|
||||||
|
LL | fn test2<'a>() {
|
||||||
|
| -- lifetime `'a` defined here
|
||||||
|
...
|
||||||
|
LL | test1::<dyn Bar<&'a u32>, _>();
|
||||||
|
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ requires that `'a` must outlive `'static`
|
||||||
|
|
||||||
|
error: aborting due to previous error
|
||||||
|
|
@ -1,16 +1,14 @@
|
|||||||
// check-pass
|
trait Foo: 'static { }
|
||||||
|
|
||||||
use std::any::Any;
|
trait Bar<T>: Foo { }
|
||||||
|
|
||||||
trait A<T>: Any {
|
fn test1<T: ?Sized + Bar<S>, S>() { }
|
||||||
fn m(&self) {}
|
|
||||||
|
fn test2<'a>() {
|
||||||
|
// Here: the type `dyn Bar<&'a u32>` references `'a`,
|
||||||
|
// and so it does not outlive `'static`.
|
||||||
|
test1::<dyn Bar<&'a u32>, _>();
|
||||||
|
//~^ ERROR the type `(dyn Bar<&'a u32> + 'static)` does not fulfill the required lifetime
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S, T: 'static> A<S> for T {}
|
fn main() { }
|
||||||
|
|
||||||
fn call_obj<'a>() {
|
|
||||||
let obj: &dyn A<&'a ()> = &();
|
|
||||||
obj.m();
|
|
||||||
}
|
|
||||||
|
|
||||||
fn main() {}
|
|
||||||
|
15
src/test/ui/traits/object/supertrait-lifetime-bound.stderr
Normal file
15
src/test/ui/traits/object/supertrait-lifetime-bound.stderr
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
error[E0477]: the type `(dyn Bar<&'a u32> + 'static)` does not fulfill the required lifetime
|
||||||
|
--> $DIR/supertrait-lifetime-bound.rs:10:5
|
||||||
|
|
|
||||||
|
LL | test1::<dyn Bar<&'a u32>, _>();
|
||||||
|
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
|
|
|
||||||
|
note: type must satisfy the static lifetime as required by this binding
|
||||||
|
--> $DIR/supertrait-lifetime-bound.rs:5:22
|
||||||
|
|
|
||||||
|
LL | fn test1<T: ?Sized + Bar<S>, S>() { }
|
||||||
|
| ^^^^^^
|
||||||
|
|
||||||
|
error: aborting due to previous error
|
||||||
|
|
||||||
|
For more information about this error, try `rustc --explain E0477`.
|
Loading…
Reference in New Issue
Block a user