use crate::alloc::Box;
#[cfg(no_maybe_uninit)]
use core::marker::PhantomData;
use core::mem;
#[cfg(not(no_maybe_uninit))]
use core::mem::MaybeUninit;
use core::ptr;
#[cfg(feature = "unstable-debug")]
use core::any;
pub struct Any {
value: Value,
drop: unsafe fn(&mut Value),
fingerprint: Fingerprint,
#[cfg(feature = "unstable-debug")]
type_name: &'static str,
}
union Value {
ptr: *mut (),
inline: [MaybeUninit<usize>; 2],
}
fn is_small<T>() -> bool {
cfg!(not(no_maybe_uninit))
&& mem::size_of::<T>() <= mem::size_of::<Value>()
&& mem::align_of::<T>() <= mem::align_of::<Value>()
}
impl Any {
pub(crate) unsafe fn new<T>(t: T) -> Self {
let value: Value;
let drop: unsafe fn(&mut Value);
let fingerprint = Fingerprint::of::<T>();
if is_small::<T>() {
let mut inline = [MaybeUninit::uninit(); 2];
unsafe { ptr::write(inline.as_mut_ptr().cast::<T>(), t) };
value = Value { inline };
unsafe fn inline_drop<T>(value: &mut Value) {
unsafe { ptr::drop_in_place(value.inline.as_mut_ptr().cast::<T>()) }
}
drop = inline_drop::<T>;
} else {
let ptr = Box::into_raw(Box::new(t)).cast::<()>();
value = Value { ptr };
unsafe fn ptr_drop<T>(value: &mut Value) {
mem::drop(unsafe { Box::from_raw(value.ptr.cast::<T>()) });
}
drop = ptr_drop::<T>;
};
Any {
value,
drop,
fingerprint,
#[cfg(feature = "unstable-debug")]
type_name: any::type_name::<T>(),
}
}
pub(crate) unsafe fn view<T>(&mut self) -> &mut T {
if cfg!(not(miri)) && self.fingerprint != Fingerprint::of::<T>() {
self.invalid_cast_to::<T>();
}
let ptr = if is_small::<T>() {
unsafe { self.value.inline.as_mut_ptr().cast::<T>() }
} else {
unsafe { self.value.ptr.cast::<T>() }
};
unsafe { &mut *ptr }
}
pub(crate) unsafe fn take<T>(mut self) -> T {
if cfg!(not(miri)) && self.fingerprint != Fingerprint::of::<T>() {
self.invalid_cast_to::<T>();
}
if is_small::<T>() {
let ptr = unsafe { self.value.inline.as_mut_ptr().cast::<T>() };
let value = unsafe { ptr::read(ptr) };
mem::forget(self);
value
} else {
let ptr = unsafe { self.value.ptr.cast::<T>() };
let box_t = unsafe { Box::from_raw(ptr) };
mem::forget(self);
*box_t
}
}
#[cfg(not(feature = "unstable-debug"))]
fn invalid_cast_to<T>(&self) -> ! {
panic!("invalid cast; enable `unstable-debug` feature to debug");
}
#[cfg(feature = "unstable-debug")]
fn invalid_cast_to<T>(&self) -> ! {
let from = self.type_name;
let to = any::type_name::<T>();
panic!("invalid cast: {} to {}", from, to);
}
}
impl Drop for Any {
fn drop(&mut self) {
unsafe { (self.drop)(&mut self.value) }
}
}
#[cfg(no_maybe_uninit)]
#[derive(Copy, Clone)]
struct MaybeUninit<T>(PhantomData<T>);
#[cfg(no_maybe_uninit)]
impl<T> MaybeUninit<T> {
fn uninit() -> Self {
MaybeUninit(PhantomData)
}
}
#[derive(Debug, Eq, PartialEq)]
struct Fingerprint {
size: usize,
align: usize,
#[cfg(include_fnptr_in_fingerprint)]
id: usize,
}
impl Fingerprint {
fn of<T>() -> Fingerprint {
Fingerprint {
size: mem::size_of::<T>(),
align: mem::align_of::<T>(),
#[cfg(include_fnptr_in_fingerprint)]
id: Fingerprint::of::<T> as usize,
}
}
}
#[test]
fn test_fingerprint() {
assert_eq!(Fingerprint::of::<usize>(), Fingerprint::of::<usize>());
assert_eq!(Fingerprint::of::<&str>(), Fingerprint::of::<&'static str>());
assert_ne!(Fingerprint::of::<u32>(), Fingerprint::of::<[u8; 4]>());
assert_ne!(Fingerprint::of::<u32>(), Fingerprint::of::<[u32; 2]>());
if cfg!(all(include_fnptr_in_fingerprint, not(miri))) {
assert_ne!(Fingerprint::of::<usize>(), Fingerprint::of::<isize>());
assert_ne!(Fingerprint::of::<usize>(), Fingerprint::of::<&usize>());
assert_ne!(Fingerprint::of::<&usize>(), Fingerprint::of::<&&usize>());
assert_ne!(Fingerprint::of::<&usize>(), Fingerprint::of::<&mut usize>());
struct A;
struct B;
assert_ne!(Fingerprint::of::<A>(), Fingerprint::of::<B>());
}
}