#[cfg(not(nightly))]
use sptr::Strict;
use std::ptr::NonNull;
#[repr(C)]
struct DatumBlob {
_data: [u8; 0],
_marker: core::marker::PhantomData<(*mut u8, core::marker::PhantomPinned)>,
}
#[repr(transparent)]
#[derive(Debug, Copy, Clone, PartialEq)]
pub struct Datum(*mut DatumBlob);
impl Datum {
pub fn value(self) -> usize {
#[allow(unstable_name_collisions)]
self.0.addr()
}
pub fn is_null(self) -> bool {
self.0.is_null()
}
pub fn cast_mut_ptr<T>(self) -> *mut T {
#[allow(unstable_name_collisions)]
self.0.cast()
}
}
impl From<usize> for Datum {
fn from(val: usize) -> Datum {
#[allow(unstable_name_collisions)]
Datum(NonNull::<DatumBlob>::dangling().as_ptr().with_addr(val))
}
}
impl From<Datum> for usize {
fn from(val: Datum) -> usize {
#[allow(unstable_name_collisions)]
val.0.addr()
}
}
impl From<isize> for Datum {
fn from(val: isize) -> Datum {
Datum::from(val as usize)
}
}
impl From<u8> for Datum {
fn from(val: u8) -> Datum {
Datum::from(usize::from(val))
}
}
impl From<u16> for Datum {
fn from(val: u16) -> Datum {
Datum::from(usize::from(val))
}
}
impl From<u32> for Datum {
fn from(val: u32) -> Datum {
Datum::from(val as usize)
}
}
impl From<u64> for Datum {
fn from(val: u64) -> Datum {
Datum::from(val as usize)
}
}
impl From<i8> for Datum {
fn from(val: i8) -> Datum {
Datum::from(isize::from(val))
}
}
impl From<i16> for Datum {
fn from(val: i16) -> Datum {
Datum::from(isize::from(val))
}
}
impl From<i32> for Datum {
fn from(val: i32) -> Datum {
Datum::from(val as usize)
}
}
impl From<i64> for Datum {
fn from(val: i64) -> Datum {
Datum::from(val as usize)
}
}
impl From<bool> for Datum {
fn from(val: bool) -> Datum {
Datum::from(val as usize)
}
}
impl<T> From<*mut T> for Datum {
fn from(val: *mut T) -> Datum {
Datum(val.cast())
}
}
impl<T> From<*const T> for Datum {
fn from(val: *const T) -> Datum {
Datum(val as *mut _)
}
}
impl<T> PartialEq<*mut T> for Datum {
fn eq(&self, other: &*mut T) -> bool {
&self.0.cast() == other
}
}
impl<T> PartialEq<Datum> for *mut T {
fn eq(&self, other: &Datum) -> bool {
self == &other.0.cast()
}
}
#[repr(C)]
#[derive(Debug, Copy, Clone)]
pub struct NullableDatum {
pub value: Datum,
pub isnull: bool,
}
impl TryFrom<NullableDatum> for Datum {
type Error = ();
fn try_from(nd: NullableDatum) -> Result<Datum, ()> {
let NullableDatum { value, isnull } = nd;
if isnull {
Err(())
} else {
Ok(value)
}
}
}
impl From<NullableDatum> for Option<Datum> {
fn from(nd: NullableDatum) -> Option<Datum> {
Some(Datum::try_from(nd).ok()?)
}
}
#[cfg(test)]
mod test {
use super::*;
#[test]
fn roundtrip_integers() {
#[cfg(target_pointer_width = "64")]
mod types {
pub type UnsignedInt = u64;
pub type SignedInt = i64;
}
#[cfg(target_pointer_width = "32")]
mod types {
pub type UnsignedInt = u32;
pub type SignedInt = i32;
}
use types::*;
let val: SignedInt = 123456;
let datum = Datum::from(val);
assert_eq!(datum.value() as SignedInt, val);
let val: isize = 123456;
let datum = Datum::from(val);
assert_eq!(datum.value() as isize, val);
let val: SignedInt = -123456;
let datum = Datum::from(val);
assert_eq!(datum.value() as SignedInt, val);
let val: isize = -123456;
let datum = Datum::from(val);
assert_eq!(datum.value() as isize, val);
let val: UnsignedInt = 123456;
let datum = Datum::from(val);
assert_eq!(datum.value() as UnsignedInt, val);
let val: usize = 123456;
let datum = Datum::from(val);
assert_eq!(datum.value() as usize, val);
}
}