use core::marker::Unsize;
use core::ptr::{self, DynMetadata, NonNull, Pointee};
#[const_trait]
pub trait DynDynCastTarget: private::Sealed {
type Root: ?Sized;
fn ptr_into_parts(ptr: NonNull<Self>) -> (NonNull<()>, DynMetadata<Self::Root>);
fn ptr_from_parts(data: NonNull<()>, meta: DynMetadata<Self::Root>) -> NonNull<Self>;
fn meta_for_ty<U: Unsize<Self>>() -> DynMetadata<Self::Root>;
}
impl<M: ?Sized, T: Pointee<Metadata = DynMetadata<M>> + ?Sized> const DynDynCastTarget for T {
type Root = M;
fn ptr_into_parts(ptr: NonNull<Self>) -> (NonNull<()>, DynMetadata<M>) {
(ptr.cast(), ptr::metadata(ptr.as_ptr()))
}
fn ptr_from_parts(data: NonNull<()>, meta: DynMetadata<M>) -> NonNull<Self> {
unsafe { NonNull::new_unchecked(ptr::from_raw_parts_mut(data.as_ptr(), meta)) }
}
fn meta_for_ty<U: Unsize<Self>>() -> DynMetadata<M> {
ptr::metadata(ptr::null::<U>() as *const Self)
}
}
mod private {
use core::ptr::{DynMetadata, Pointee};
pub trait Sealed {}
impl<M: ?Sized, T: Pointee<Metadata = DynMetadata<M>> + ?Sized> Sealed for T {}
}