In the previous section, we showed how a #[dyn(identity)]
function must return "something that can be converted into a dynx struct", and we showed that a case of returning a Pin<Box<impl Future, A>>
type. But what are the general rules for constructing a dynx
struct? You're asking the right question, but that's a part of the design we haven't bottomed out yet.
In short, there are two "basic" approaches we could take. One of them is more conservative, in that it doesn't change much about Rust today, but it's also much more complex, because dyn
dealing with all the "corner cases" of dyn
is kind of complicated. The other is more radical, but may result in an overall smoother, more coherent design.
Apart from that tantalizing tidbit, we are intentionally not providing the details here, because this document is long enough as it is! The next document dives into this question, along with a related question, which is how dynx
and sealed traits interact.
This is actually a complex question with (at least) two possible answers.
The pointer type P
must implement IntoRawPointer
(along with various other criteria) and its referent must implement Bounds
.
// Not pseudocode, will be added to the stdlib and implemented
// by various types, including `Box` and `Pin<Box>`.
unsafe trait IntoRawPointer: Deref {
/// Convert this pointer into a raw pointer to `Self::Target`.
///
/// This raw pointer must be valid to dereference until `drop_raw` (below) is invoked;
/// this trait is unsafe because the impl must ensure that to be true.
fn into_raw(self) -> *mut Self::Target;
/// Drops the smart pointer itself as well as the contents of the pointer.
/// For example, when `Self = Box<T>`, this will free the box as well as the
/// `T` value.
unsafe fn drop_raw(this: *mut Self::Target);
}
- Must be
IntoRawPointer
which ensures:Deref
andDerefMut
are stable, side-effect free and all that- they deref to the same memory as
into_raw
- If
Bounds
includes a&mut self
method,P
must beDerefMut
- If
Bounds
includes a&self
method,P
must beDeref
- If
Bounds
includesPin<&mut Self>
,P
must beUnpin
... and ... something somethingDerefMut
? how do you get fromPin<P>
toPin<&mut P::Target>
? - If
Bounds
includesPin<&Self>
,P
must beUnpin
... and ... something somethingDerefMut
? how do you get fromPin<P>
toPin<&mut P::Target>
? - If
Bounds
includes an auto traitAutoTrait
,P
must implementAutoTrait
- and:
dynx Bounds
implements the auto traitAutoTrait
(in general,dynx Bounds
implements all ofBounds
)
- and:
Bounds
must be "dyn safe"
Alternatively, we could declare that the pointer type P must implement Bounds
. This is much simpler to express, but it has some issues. For example, if you have
trait Foo {
}
then we could not construct a dynx Foo
from a Box<dyn Foo>
because there is no impl Foo for Box<dyn Foo>
. It would be nice if those impls could be added automatically or at least more easily.
// Not pseudocode, will be added to the stdlib and implemented
// by various types, including `Box` and `Pin<Box>`.
unsafe trait IntoRawPointer: Deref {
/// Convert this pointer into a raw pointer to `Self::Target`.
///
/// This raw pointer must be valid to dereference until `drop_raw` (below) is invoked;
/// this trait is unsafe because the impl must ensure that to be true.
fn into_raw(self) -> *mut Self::Target;
/// These methods would be used by compiler to convert back so we can invoke the original
/// impls.
unsafe fn from_ref(this: &*mut Self::Target) -> &Self;
unsafe fn from_mut_ref(this: &mut *mut Self::Target) -> &mut Self;
...
/// Drops the smart pointer itself as well as the contents of the pointer.
/// For example, when `Self = Box<T>`, this will free the box as well as the
/// `T` value.
unsafe fn drop_raw(this: *mut Self::Target);
}