2019-05-17 02:20:14 +01:00
|
|
|
// Check that we can manually implement an object-unsafe trait for its trait object.
|
|
|
|
|
2019-01-08 22:14:04 +01:00
|
|
|
// run-pass
|
|
|
|
|
|
|
|
#![feature(object_safe_for_dispatch)]
|
|
|
|
|
|
|
|
trait Bad {
|
|
|
|
fn stat() -> char {
|
|
|
|
'A'
|
|
|
|
}
|
|
|
|
fn virt(&self) -> char {
|
|
|
|
'B'
|
|
|
|
}
|
|
|
|
fn indirect(&self) -> char {
|
|
|
|
Self::stat()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
trait Good {
|
|
|
|
fn good_virt(&self) -> char {
|
|
|
|
panic!()
|
|
|
|
}
|
|
|
|
fn good_indirect(&self) -> char {
|
|
|
|
panic!()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Bad for dyn Bad + 'a {
|
|
|
|
fn stat() -> char {
|
|
|
|
'C'
|
|
|
|
}
|
|
|
|
fn virt(&self) -> char {
|
|
|
|
'D'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct Struct {}
|
|
|
|
|
|
|
|
impl Bad for Struct {}
|
|
|
|
|
|
|
|
impl Good for Struct {}
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
let s = Struct {};
|
|
|
|
|
|
|
|
let mut res = String::new();
|
|
|
|
|
2019-05-17 02:20:14 +01:00
|
|
|
// Directly call static.
|
2019-01-08 22:14:04 +01:00
|
|
|
res.push(Struct::stat()); // "A"
|
|
|
|
res.push(<dyn Bad>::stat()); // "AC"
|
|
|
|
|
|
|
|
let good: &dyn Good = &s;
|
|
|
|
|
|
|
|
// These look similar enough...
|
|
|
|
let bad = unsafe { std::mem::transmute::<&dyn Good, &dyn Bad>(good) };
|
|
|
|
|
2019-05-17 02:20:14 +01:00
|
|
|
// Call virtual.
|
2019-01-08 22:14:04 +01:00
|
|
|
res.push(s.virt()); // "ACB"
|
|
|
|
res.push(bad.virt()); // "ACBD"
|
|
|
|
|
2019-05-17 02:20:14 +01:00
|
|
|
// Indirectly call static.
|
2019-01-08 22:14:04 +01:00
|
|
|
res.push(s.indirect()); // "ACBDA"
|
|
|
|
res.push(bad.indirect()); // "ACBDAC"
|
|
|
|
|
2019-05-17 02:20:14 +01:00
|
|
|
assert_eq!(&res, "ACBDAC");
|
2019-01-08 22:14:04 +01:00
|
|
|
}
|