use core::{marker::PhantomData, mem};
pub struct CastToken<T>(PhantomData<T>);
impl<T> CastToken<T> {
pub fn of_val(_value: &T) -> Self {
Self(PhantomData)
}
}
pub trait TryCastMut<'a, T: 'static> {
#[inline(always)]
fn try_cast<U: 'static>(&self, value: &'a mut T) -> Result<&'a mut U, &'a mut T> {
if type_eq::<T, U>() {
Ok(unsafe { &mut *(value as *mut T as *mut U) })
} else {
Err(value)
}
}
}
impl<'a, T: 'static> TryCastMut<'a, T> for &&CastToken<&'a mut T> {}
pub trait TryCastRef<'a, T: 'static> {
#[inline(always)]
fn try_cast<U: 'static>(&self, value: &'a T) -> Result<&'a U, &'a T> {
if type_eq::<T, U>() {
Ok(unsafe { &*(value as *const T as *const U) })
} else {
Err(value)
}
}
}
impl<'a, T: 'static> TryCastRef<'a, T> for &CastToken<&'a T> {}
pub trait TryCastOwned<T: 'static> {
#[inline(always)]
fn try_cast<U: 'static>(&self, value: T) -> Result<U, T> {
if type_eq::<T, U>() {
Ok(unsafe { mem::transmute_copy::<T, U>(&mem::ManuallyDrop::new(value)) })
} else {
Err(value)
}
}
}
impl<T: 'static> TryCastOwned<T> for CastToken<T> {}
#[inline(always)]
fn type_eq<T: 'static, U: 'static>() -> bool {
fn type_id_of<T>() -> usize {
type_id_of::<T> as usize
}
mem::size_of::<T>() == mem::size_of::<U>()
&& mem::align_of::<T>() == mem::align_of::<U>()
&& type_id_of::<T>() == type_id_of::<U>()
}