compact 0.2.16

Store objects containing dynamic fields either compactly in consecutive memory or using traditional heap pointers
Documentation
use super::compact::Compact;

/// A wrapper to make an `Option` of a nontrivial `Compact` possible.
/// Unfortunately, we can't blanket-`impl` that, since that overlaps
/// (for the compiler) with the `impl` for trivial `Copy` types...
#[derive(Clone, Default, Debug)]
pub struct CompactOption<T: Compact + Clone>(pub Option<T>);

impl<T: Compact + Clone> ::std::ops::Deref for CompactOption<T> {
    type Target = Option<T>;

    fn deref(&self) -> &Option<T> {
        &self.0
    }
}

impl<T: Compact + Clone> ::std::ops::DerefMut for CompactOption<T> {
    fn deref_mut(&mut self) -> &mut Option<T> {
        &mut self.0
    }
}

impl<T: Clone + Compact> Compact for CompactOption<T> {
    fn is_still_compact(&self) -> bool {
        self.0
            .as_ref()
            .map(|t| t.is_still_compact())
            .unwrap_or(true)
    }

    fn dynamic_size_bytes(&self) -> usize {
        self.0.as_ref().map(|t| t.dynamic_size_bytes()).unwrap_or(0)
    }

    unsafe fn compact(source: *mut Self, dest: *mut Self, new_dynamic_part: *mut u8) {
        if let CompactOption(Some(ref mut s)) = *source {
            ::std::ptr::copy_nonoverlapping(source, dest, 1);
            if let CompactOption(Some(ref mut d)) = *dest {
                Compact::compact(s, d, new_dynamic_part);
            } else {
                unreachable!()
            }
        } else {
            ::std::ptr::write(dest, CompactOption(None));
        }
    }

    unsafe fn decompact(source: *const Self) -> Self {
        if let CompactOption(Some(ref s)) = *source {
            CompactOption(Some(Compact::decompact(s)))
        } else {
            CompactOption(None)
        }
    }
}

#[cfg(feature = "serde-serialization")]
use std::marker::PhantomData;

#[cfg(feature = "serde-serialization")]
impl<T: Compact + ::serde::ser::Serialize> ::serde::ser::Serialize for CompactOption<T> {
    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
    where
        S: ::serde::ser::Serializer,
    {
        match self.0 {
            Some(ref value) => serializer.serialize_some(value),
            None => serializer.serialize_none(),
        }
    }
}

#[cfg(feature = "serde-serialization")]
struct CompactOptionVisitor<T: Compact> {
    marker: PhantomData<fn() -> CompactOption<T>>,
}

#[cfg(feature = "serde-serialization")]
impl<T: Compact> CompactOptionVisitor<T> {
    fn new() -> Self {
        CompactOptionVisitor {
            marker: PhantomData,
        }
    }
}

#[cfg(feature = "serde-serialization")]
impl<'de, T: Compact + ::serde::de::Deserialize<'de>> ::serde::de::Visitor<'de>
    for CompactOptionVisitor<T>
{
    type Value = CompactOption<T>;

    fn expecting(&self, formatter: &mut ::std::fmt::Formatter) -> ::std::fmt::Result {
        formatter.write_str("An option")
    }

    fn visit_some<D>(self, deserializer: D) -> Result<Self::Value, D::Error>
    where
        D: ::serde::de::Deserializer<'de>,
    {
        Ok(CompactOption(Some(T::deserialize(deserializer)?)))
    }

    fn visit_none<E>(self) -> Result<Self::Value, E>
    where
        E: ::serde::de::Error,
    {
        Ok(CompactOption(None))
    }
}

#[cfg(feature = "serde-serialization")]
impl<'de, T: Compact + ::serde::de::Deserialize<'de>> ::serde::de::Deserialize<'de>
    for CompactOption<T>
{
    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
    where
        D: ::serde::de::Deserializer<'de>,
    {
        deserializer.deserialize_option(CompactOptionVisitor::new())
    }
}

#[test]
fn basic_option() {
    use super::compact_vec::CompactVec;
    use super::simple_allocator_trait::{Allocator, DefaultHeap};
    let mut option: CompactOption<CompactVec<u32>> = CompactOption(Some(CompactVec::new()));

    if let Some(ref mut list) = *option {
        list.push(1);
        list.push(2);
        list.push(3);
        assert_eq!(&[1, 2, 3], &**list);
    } else {
        unreachable!()
    }

    let bytes = option.total_size_bytes();
    let storage = DefaultHeap::allocate(bytes);

    unsafe {
        Compact::compact_behind(&mut option, storage as *mut CompactOption<CompactVec<u32>>);
        ::std::mem::forget(option);
        if let Some(ref list) = **(storage as *mut CompactOption<CompactVec<u32>>) {
            assert_eq!(&[1, 2, 3], &**list);
        } else {
            unreachable!()
        }
        println!("before decompact!");
        if let Some(ref list) = *Compact::decompact(storage as *mut CompactOption<CompactVec<u32>>)
        {
            assert_eq!(&[1, 2, 3], &**list);
        } else {
            unreachable!()
        }
        DefaultHeap::deallocate(storage, bytes);
    }
}