use alloc::collections::TryReserveError;
use alloc::vec::Vec;
use core::fmt;
use core::iter::{Enumerate, FusedIterator};
use core::marker::PhantomData;
use core::mem::{ManuallyDrop, MaybeUninit};
use core::ops::{Index, IndexMut};
use crate::util::{Never, PanicOnDrop, UnwrapNever};
use crate::{DefaultKey, Key, KeyData};
union SlotUnion<T> {
value: ManuallyDrop<T>,
next_free: u32,
}
struct Slot<T> {
u: SlotUnion<T>,
version: u32, }
enum SlotContent<'a, T: 'a> {
Occupied(&'a T),
Vacant(&'a u32),
}
enum SlotContentMut<'a, T: 'a> {
OccupiedMut(&'a mut T),
VacantMut(&'a mut u32),
}
use self::SlotContent::{Occupied, Vacant};
use self::SlotContentMut::{OccupiedMut, VacantMut};
impl<T> Slot<T> {
#[inline(always)]
pub fn occupied(&self) -> bool {
self.version % 2 > 0
}
pub fn get(&self) -> SlotContent<'_, T> {
unsafe {
if self.occupied() {
Occupied(&*self.u.value)
} else {
Vacant(&self.u.next_free)
}
}
}
pub fn get_mut(&mut self) -> SlotContentMut<'_, T> {
unsafe {
if self.occupied() {
OccupiedMut(&mut *self.u.value)
} else {
VacantMut(&mut self.u.next_free)
}
}
}
}
impl<T> Drop for Slot<T> {
fn drop(&mut self) {
if core::mem::needs_drop::<T>() && self.occupied() {
unsafe {
ManuallyDrop::drop(&mut self.u.value);
}
}
}
}
impl<T: Clone> Clone for Slot<T> {
fn clone(&self) -> Self {
Self {
u: match self.get() {
Occupied(value) => SlotUnion {
value: ManuallyDrop::new(value.clone()),
},
Vacant(&next_free) => SlotUnion { next_free },
},
version: self.version,
}
}
fn clone_from(&mut self, source: &Self) {
match (self.get_mut(), source.get()) {
(OccupiedMut(self_val), Occupied(source_val)) => self_val.clone_from(source_val),
(OccupiedMut(_), Vacant(&next_free)) => unsafe {
ManuallyDrop::drop(&mut self.u.value);
self.u = SlotUnion { next_free };
},
(VacantMut(self_next_free), Vacant(&source_next_free)) => {
*self_next_free = source_next_free
},
(VacantMut(_), Occupied(value)) => {
self.u = SlotUnion {
value: ManuallyDrop::new(value.clone()),
}
},
}
self.version = source.version;
}
}
impl<T: fmt::Debug> fmt::Debug for Slot<T> {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
let mut builder = fmt.debug_struct("Slot");
builder.field("version", &self.version);
match self.get() {
Occupied(value) => builder.field("value", value).finish(),
Vacant(next_free) => builder.field("next_free", next_free).finish(),
}
}
}
#[derive(Debug)]
pub struct SlotMap<K: Key, V> {
slots: Vec<Slot<V>>,
free_head: u32,
num_elems: u32,
_k: PhantomData<fn(K) -> K>,
}
impl<V> SlotMap<DefaultKey, V> {
pub fn new() -> Self {
Self::with_capacity_and_key(0)
}
pub fn with_capacity(capacity: usize) -> Self {
Self::with_capacity_and_key(capacity)
}
}
impl<K: Key, V> SlotMap<K, V> {
pub fn with_key() -> Self {
Self::with_capacity_and_key(0)
}
pub fn with_capacity_and_key(capacity: usize) -> Self {
let mut slots = Vec::with_capacity(capacity + 1);
slots.push(Slot {
u: SlotUnion { next_free: 0 },
version: 0,
});
Self {
slots,
free_head: 1,
num_elems: 0,
_k: PhantomData,
}
}
pub fn len(&self) -> usize {
self.num_elems as usize
}
pub fn is_empty(&self) -> bool {
self.num_elems == 0
}
pub fn capacity(&self) -> usize {
self.slots.capacity() - 1
}
pub fn reserve(&mut self, additional: usize) {
let needed = (self.len() + additional).saturating_sub(self.slots.len() - 1);
self.slots.reserve(needed);
}
pub fn try_reserve(&mut self, additional: usize) -> Result<(), TryReserveError> {
let needed = (self.len() + additional).saturating_sub(self.slots.len() - 1);
self.slots.try_reserve(needed)
}
pub fn contains_key(&self, key: K) -> bool {
let kd = key.data();
self.slots
.get(kd.idx as usize)
.map_or(false, |slot| slot.version == kd.version.get())
}
#[inline(always)]
pub fn insert(&mut self, value: V) -> K {
self.try_insert_with_key::<_, Never>(move |_| Ok(value))
.unwrap_never()
}
#[inline(always)]
pub fn insert_with_key<F>(&mut self, f: F) -> K
where
F: FnOnce(K) -> V,
{
self.try_insert_with_key::<_, Never>(move |k| Ok(f(k)))
.unwrap_never()
}
pub fn try_insert_with_key<F, E>(&mut self, f: F) -> Result<K, E>
where
F: FnOnce(K) -> Result<V, E>,
{
if let Some(slot) = self.slots.get_mut(self.free_head as usize) {
let occupied_version = slot.version | 1;
let kd = KeyData::new(self.free_head, occupied_version);
let value = f(kd.into())?;
unsafe {
self.free_head = slot.u.next_free;
slot.u.value = ManuallyDrop::new(value);
slot.version = occupied_version;
}
self.num_elems += 1;
return Ok(kd.into());
}
if self.slots.len() >= u32::MAX as usize {
panic!("SlotMap is full");
}
let version = 1;
let kd = KeyData::new(self.slots.len() as u32, version);
self.slots.push(Slot {
u: SlotUnion {
value: ManuallyDrop::new(f(kd.into())?),
},
version,
});
self.free_head = kd.idx + 1;
self.num_elems += 1;
Ok(kd.into())
}
#[inline(always)]
unsafe fn remove_from_slot(&mut self, idx: usize) -> V {
let slot = self.slots.get_unchecked_mut(idx);
let value = ManuallyDrop::take(&mut slot.u.value);
slot.u.next_free = self.free_head;
self.free_head = idx as u32;
self.num_elems -= 1;
slot.version = slot.version.wrapping_add(1);
value
}
pub fn remove(&mut self, key: K) -> Option<V> {
let kd = key.data();
if self.contains_key(key) {
Some(unsafe { self.remove_from_slot(kd.idx as usize) })
} else {
None
}
}
pub fn detach(&mut self, key: K) -> Option<V> {
let kd = key.data();
if self.contains_key(key) {
unsafe {
let slot = self.slots.get_unchecked_mut(kd.idx as usize);
let value = ManuallyDrop::take(&mut slot.u.value);
slot.u.next_free = u32::MAX;
slot.version = slot.version.wrapping_add(1);
self.num_elems -= 1;
Some(value)
}
} else {
None
}
}
pub fn reattach(&mut self, detached_key: K, value: V) {
let kd = detached_key.data();
let slot = self
.slots
.get_mut(kd.idx as usize)
.filter(|slot| slot.version == kd.version.get().wrapping_add(1))
.filter(|slot| unsafe {
slot.u.next_free == u32::MAX
})
.expect("key is not detached");
slot.u.value = ManuallyDrop::new(value);
slot.version = slot.version.wrapping_sub(1);
self.num_elems += 1;
}
pub fn retain<F>(&mut self, mut f: F)
where
F: FnMut(K, &mut V) -> bool,
{
for i in 1..self.slots.len() {
let slot = unsafe { self.slots.get_unchecked_mut(i) };
let version = slot.version;
let should_remove = if let OccupiedMut(value) = slot.get_mut() {
let key = KeyData::new(i as u32, version).into();
!f(key, value)
} else {
false
};
if should_remove {
unsafe { self.remove_from_slot(i) };
}
}
}
pub fn clear(&mut self) {
self.drain();
}
pub fn drain(&mut self) -> Drain<'_, K, V> {
Drain { cur: 1, sm: self }
}
pub fn get(&self, key: K) -> Option<&V> {
let kd = key.data();
self.slots
.get(kd.idx as usize)
.filter(|slot| slot.version == kd.version.get())
.map(|slot| unsafe { &*slot.u.value })
}
pub unsafe fn get_unchecked(&self, key: K) -> &V {
debug_assert!(self.contains_key(key));
&self.slots.get_unchecked(key.data().idx as usize).u.value
}
pub fn get_mut(&mut self, key: K) -> Option<&mut V> {
let kd = key.data();
self.slots
.get_mut(kd.idx as usize)
.filter(|slot| slot.version == kd.version.get())
.map(|slot| unsafe { &mut *slot.u.value })
}
pub unsafe fn get_unchecked_mut(&mut self, key: K) -> &mut V {
debug_assert!(self.contains_key(key));
&mut self
.slots
.get_unchecked_mut(key.data().idx as usize)
.u
.value
}
pub fn get_disjoint_mut<const N: usize>(&mut self, keys: [K; N]) -> Option<[&mut V; N]> {
let mut ptrs: [MaybeUninit<*mut V>; N] = [(); N].map(|_| MaybeUninit::uninit());
let slots_ptr = self.slots.as_mut_ptr();
let mut i = 0;
while i < N {
let kd = keys[i].data();
if !self.contains_key(kd.into()) {
break;
}
unsafe {
let slot = &mut *slots_ptr.add(kd.idx as usize);
slot.version ^= 1;
ptrs[i] = MaybeUninit::new(&mut *slot.u.value);
}
i += 1;
}
for k in &keys[..i] {
let idx = k.data().idx as usize;
unsafe {
(*slots_ptr.add(idx)).version ^= 1;
}
}
if i == N {
Some(ptrs.map(|p| unsafe { &mut *p.assume_init() }))
} else {
None
}
}
pub unsafe fn get_disjoint_unchecked_mut<const N: usize>(
&mut self,
keys: [K; N],
) -> [&mut V; N] {
let slots_ptr = self.slots.as_mut_ptr();
keys.map(|k| unsafe {
let slot = &mut *slots_ptr.add(k.data().idx as usize);
&mut *slot.u.value
})
}
pub fn iter(&self) -> Iter<'_, K, V> {
let mut it = self.slots.iter().enumerate();
it.next(); Iter {
slots: it,
num_left: self.len(),
_k: PhantomData,
}
}
pub fn iter_mut(&mut self) -> IterMut<'_, K, V> {
let len = self.len();
let mut it = self.slots.iter_mut().enumerate();
it.next(); IterMut {
num_left: len,
slots: it,
_k: PhantomData,
}
}
pub fn keys(&self) -> Keys<'_, K, V> {
Keys { inner: self.iter() }
}
pub fn values(&self) -> Values<'_, K, V> {
Values { inner: self.iter() }
}
pub fn values_mut(&mut self) -> ValuesMut<'_, K, V> {
ValuesMut {
inner: self.iter_mut(),
}
}
}
impl<K: Key, V> Clone for SlotMap<K, V>
where
V: Clone,
{
fn clone(&self) -> Self {
Self {
slots: self.slots.clone(),
..*self
}
}
fn clone_from(&mut self, source: &Self) {
let guard = PanicOnDrop("abort - a panic during SlotMap::clone_from is unrecoverable");
self.slots.clone_from(&source.slots);
self.free_head = source.free_head;
self.num_elems = source.num_elems;
core::mem::forget(guard);
}
}
impl<K: Key, V> Default for SlotMap<K, V> {
fn default() -> Self {
Self::with_key()
}
}
impl<K: Key, V> Index<K> for SlotMap<K, V> {
type Output = V;
fn index(&self, key: K) -> &V {
match self.get(key) {
Some(r) => r,
None => panic!("invalid SlotMap key used"),
}
}
}
impl<K: Key, V> IndexMut<K> for SlotMap<K, V> {
fn index_mut(&mut self, key: K) -> &mut V {
match self.get_mut(key) {
Some(r) => r,
None => panic!("invalid SlotMap key used"),
}
}
}
#[derive(Debug)]
pub struct Drain<'a, K: 'a + Key, V: 'a> {
sm: &'a mut SlotMap<K, V>,
cur: usize,
}
#[derive(Debug, Clone)]
pub struct IntoIter<K: Key, V> {
num_left: usize,
slots: Enumerate<alloc::vec::IntoIter<Slot<V>>>,
_k: PhantomData<fn(K) -> K>,
}
#[derive(Debug)]
pub struct Iter<'a, K: 'a + Key, V: 'a> {
num_left: usize,
slots: Enumerate<core::slice::Iter<'a, Slot<V>>>,
_k: PhantomData<fn(K) -> K>,
}
impl<'a, K: 'a + Key, V: 'a> Clone for Iter<'a, K, V> {
fn clone(&self) -> Self {
Iter {
num_left: self.num_left,
slots: self.slots.clone(),
_k: self._k,
}
}
}
#[derive(Debug)]
pub struct IterMut<'a, K: 'a + Key, V: 'a> {
num_left: usize,
slots: Enumerate<core::slice::IterMut<'a, Slot<V>>>,
_k: PhantomData<fn(K) -> K>,
}
#[derive(Debug)]
pub struct Keys<'a, K: 'a + Key, V: 'a> {
inner: Iter<'a, K, V>,
}
impl<'a, K: 'a + Key, V: 'a> Clone for Keys<'a, K, V> {
fn clone(&self) -> Self {
Keys {
inner: self.inner.clone(),
}
}
}
#[derive(Debug)]
pub struct Values<'a, K: 'a + Key, V: 'a> {
inner: Iter<'a, K, V>,
}
impl<'a, K: 'a + Key, V: 'a> Clone for Values<'a, K, V> {
fn clone(&self) -> Self {
Values {
inner: self.inner.clone(),
}
}
}
#[derive(Debug)]
pub struct ValuesMut<'a, K: 'a + Key, V: 'a> {
inner: IterMut<'a, K, V>,
}
impl<'a, K: Key, V> Iterator for Drain<'a, K, V> {
type Item = (K, V);
fn next(&mut self) -> Option<(K, V)> {
let len = self.sm.slots.len();
while self.cur < len {
let idx = self.cur;
self.cur += 1;
unsafe {
let slot = self.sm.slots.get_unchecked(idx);
if slot.occupied() {
let kd = KeyData::new(idx as u32, slot.version);
return Some((kd.into(), self.sm.remove_from_slot(idx)));
}
}
}
None
}
fn size_hint(&self) -> (usize, Option<usize>) {
(self.sm.len(), Some(self.sm.len()))
}
}
impl<'a, K: Key, V> Drop for Drain<'a, K, V> {
fn drop(&mut self) {
self.for_each(|_drop| {});
}
}
impl<K: Key, V> Iterator for IntoIter<K, V> {
type Item = (K, V);
fn next(&mut self) -> Option<(K, V)> {
while let Some((idx, mut slot)) = self.slots.next() {
if slot.occupied() {
let kd = KeyData::new(idx as u32, slot.version);
slot.version = 0;
let value = unsafe { ManuallyDrop::take(&mut slot.u.value) };
self.num_left -= 1;
return Some((kd.into(), value));
}
}
None
}
fn size_hint(&self) -> (usize, Option<usize>) {
(self.num_left, Some(self.num_left))
}
}
impl<'a, K: Key, V> Iterator for Iter<'a, K, V> {
type Item = (K, &'a V);
fn next(&mut self) -> Option<(K, &'a V)> {
while let Some((idx, slot)) = self.slots.next() {
if let Occupied(value) = slot.get() {
let kd = KeyData::new(idx as u32, slot.version);
self.num_left -= 1;
return Some((kd.into(), value));
}
}
None
}
fn size_hint(&self) -> (usize, Option<usize>) {
(self.num_left, Some(self.num_left))
}
}
impl<'a, K: Key, V> Iterator for IterMut<'a, K, V> {
type Item = (K, &'a mut V);
fn next(&mut self) -> Option<(K, &'a mut V)> {
for (idx, slot) in self.slots.by_ref() {
let version = slot.version;
if let OccupiedMut(value) = slot.get_mut() {
let kd = KeyData::new(idx as u32, version);
self.num_left -= 1;
return Some((kd.into(), value));
}
}
None
}
fn size_hint(&self) -> (usize, Option<usize>) {
(self.num_left, Some(self.num_left))
}
}
impl<'a, K: Key, V> Iterator for Keys<'a, K, V> {
type Item = K;
fn next(&mut self) -> Option<K> {
self.inner.next().map(|(key, _)| key)
}
fn size_hint(&self) -> (usize, Option<usize>) {
self.inner.size_hint()
}
}
impl<'a, K: Key, V> Iterator for Values<'a, K, V> {
type Item = &'a V;
fn next(&mut self) -> Option<&'a V> {
self.inner.next().map(|(_, value)| value)
}
fn size_hint(&self) -> (usize, Option<usize>) {
self.inner.size_hint()
}
}
impl<'a, K: Key, V> Iterator for ValuesMut<'a, K, V> {
type Item = &'a mut V;
fn next(&mut self) -> Option<&'a mut V> {
self.inner.next().map(|(_, value)| value)
}
fn size_hint(&self) -> (usize, Option<usize>) {
self.inner.size_hint()
}
}
impl<'a, K: Key, V> IntoIterator for &'a SlotMap<K, V> {
type Item = (K, &'a V);
type IntoIter = Iter<'a, K, V>;
fn into_iter(self) -> Self::IntoIter {
self.iter()
}
}
impl<'a, K: Key, V> IntoIterator for &'a mut SlotMap<K, V> {
type Item = (K, &'a mut V);
type IntoIter = IterMut<'a, K, V>;
fn into_iter(self) -> Self::IntoIter {
self.iter_mut()
}
}
impl<K: Key, V> IntoIterator for SlotMap<K, V> {
type Item = (K, V);
type IntoIter = IntoIter<K, V>;
fn into_iter(self) -> Self::IntoIter {
let len = self.len();
let mut it = self.slots.into_iter().enumerate();
it.next(); IntoIter {
num_left: len,
slots: it,
_k: PhantomData,
}
}
}
impl<'a, K: Key, V> FusedIterator for Iter<'a, K, V> {}
impl<'a, K: Key, V> FusedIterator for IterMut<'a, K, V> {}
impl<'a, K: Key, V> FusedIterator for Keys<'a, K, V> {}
impl<'a, K: Key, V> FusedIterator for Values<'a, K, V> {}
impl<'a, K: Key, V> FusedIterator for ValuesMut<'a, K, V> {}
impl<'a, K: Key, V> FusedIterator for Drain<'a, K, V> {}
impl<K: Key, V> FusedIterator for IntoIter<K, V> {}
impl<'a, K: Key, V> ExactSizeIterator for Iter<'a, K, V> {}
impl<'a, K: Key, V> ExactSizeIterator for IterMut<'a, K, V> {}
impl<'a, K: Key, V> ExactSizeIterator for Keys<'a, K, V> {}
impl<'a, K: Key, V> ExactSizeIterator for Values<'a, K, V> {}
impl<'a, K: Key, V> ExactSizeIterator for ValuesMut<'a, K, V> {}
impl<'a, K: Key, V> ExactSizeIterator for Drain<'a, K, V> {}
impl<K: Key, V> ExactSizeIterator for IntoIter<K, V> {}
#[cfg(feature = "serde")]
mod serialize {
use serde::{de, Deserialize, Deserializer, Serialize, Serializer};
use super::*;
#[derive(Serialize, Deserialize)]
struct SerdeSlot<T> {
value: Option<T>,
version: u32,
}
impl<T: Serialize> Serialize for Slot<T> {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
let serde_slot = SerdeSlot {
version: self.version,
value: match self.get() {
Occupied(value) => Some(value),
Vacant(_) => None,
},
};
serde_slot.serialize(serializer)
}
}
impl<'de, T> Deserialize<'de> for Slot<T>
where
T: Deserialize<'de>,
{
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: Deserializer<'de>,
{
let serde_slot: SerdeSlot<T> = Deserialize::deserialize(deserializer)?;
let occupied = serde_slot.version % 2 == 1;
if occupied ^ serde_slot.value.is_some() {
return Err(de::Error::custom("inconsistent occupation in Slot"));
}
Ok(Self {
u: match serde_slot.value {
Some(value) => SlotUnion {
value: ManuallyDrop::new(value),
},
None => SlotUnion { next_free: 0 },
},
version: serde_slot.version,
})
}
}
impl<K: Key, V: Serialize> Serialize for SlotMap<K, V> {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
self.slots.serialize(serializer)
}
}
impl<'de, K, V> Deserialize<'de> for SlotMap<K, V>
where
K: Key,
V: Deserialize<'de>,
{
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: Deserializer<'de>,
{
let mut slots: Vec<Slot<V>> = Deserialize::deserialize(deserializer)?;
if slots.len() >= u32::MAX as usize {
return Err(de::Error::custom("too many slots"));
}
if slots.first().map_or(true, |slot| slot.version % 2 == 1) {
return Err(de::Error::custom("first slot not empty"));
}
slots[0].version = 0;
slots[0].u.next_free = 0;
let mut num_elems = 0;
let mut next_free = slots.len();
for (i, slot) in slots[1..].iter_mut().enumerate() {
if slot.occupied() {
num_elems += 1;
} else {
slot.u.next_free = next_free as u32;
next_free = i + 1;
}
}
Ok(Self {
num_elems,
slots,
free_head: next_free as u32,
_k: PhantomData,
})
}
}
}
#[cfg(test)]
mod tests {
use std::collections::{HashMap, HashSet};
use quickcheck::quickcheck;
use super::*;
#[derive(Clone)]
struct CountDrop<'a>(&'a std::cell::RefCell<usize>);
impl<'a> Drop for CountDrop<'a> {
fn drop(&mut self) {
*self.0.borrow_mut() += 1;
}
}
#[test]
fn check_drops() {
let drops = std::cell::RefCell::new(0usize);
{
let mut clone = {
let mut sm = SlotMap::new();
let mut sm_keys = Vec::new();
for _ in 0..1000 {
sm_keys.push(sm.insert(CountDrop(&drops)));
}
for i in (0..1000).filter(|i| i % 2 == 0) {
sm.remove(sm_keys[i]);
}
assert_eq!(*drops.borrow(), 500);
sm.clone()
};
assert_eq!(*drops.borrow(), 1000);
for _ in 0..250 {
clone.insert(CountDrop(&drops));
}
}
assert_eq!(*drops.borrow(), 1750);
}
#[test]
fn disjoint() {
let mut sm = SlotMap::new();
for i in 0..20usize {
sm.insert(i);
}
sm.retain(|_, i| *i % 2 == 0);
let keys: Vec<_> = sm.keys().collect();
for i in 0..keys.len() {
for j in 0..keys.len() {
if let Some([r0, r1]) = sm.get_disjoint_mut([keys[i], keys[j]]) {
*r0 ^= *r1;
*r1 = r1.wrapping_add(*r0);
} else {
assert!(i == j);
}
}
}
for i in 0..keys.len() {
for j in 0..keys.len() {
for k in 0..keys.len() {
if let Some([r0, r1, r2]) = sm.get_disjoint_mut([keys[i], keys[j], keys[k]]) {
*r0 ^= *r1;
*r0 = r0.wrapping_add(*r2);
*r1 ^= *r0;
*r1 = r1.wrapping_add(*r2);
*r2 ^= *r0;
*r2 = r2.wrapping_add(*r1);
} else {
assert!(i == j || j == k || i == k);
}
}
}
}
}
quickcheck! {
fn qc_slotmap_equiv_hashmap(operations: Vec<(u8, u32)>) -> bool {
let mut hm = HashMap::new();
let mut hm_keys = Vec::new();
let mut unique_key = 0u32;
let mut sm = SlotMap::new();
let mut sm_keys = Vec::new();
#[cfg(not(feature = "serde"))]
let num_ops = 3;
#[cfg(feature = "serde")]
let num_ops = 4;
for (op, val) in operations {
match op % num_ops {
0 => {
hm.insert(unique_key, val);
hm_keys.push(unique_key);
unique_key += 1;
sm_keys.push(sm.insert(val));
}
1 => {
if val % 10 == 0 {
let hmvals: HashSet<_> = hm.drain().map(|(_, v)| v).collect();
let smvals: HashSet<_> = sm.drain().map(|(_, v)| v).collect();
if hmvals != smvals {
return false;
}
}
if hm_keys.is_empty() { continue; }
let idx = val as usize % hm_keys.len();
if hm.remove(&hm_keys[idx]) != sm.remove(sm_keys[idx]) {
return false;
}
}
2 => {
if hm_keys.is_empty() { continue; }
let idx = val as usize % hm_keys.len();
let (hm_key, sm_key) = (&hm_keys[idx], sm_keys[idx]);
if hm.contains_key(hm_key) != sm.contains_key(sm_key) ||
hm.get(hm_key) != sm.get(sm_key) {
return false;
}
}
#[cfg(feature = "serde")]
3 => {
let ser = serde_json::to_string(&sm).unwrap();
sm = serde_json::from_str(&ser).unwrap();
}
_ => unreachable!(),
}
}
let mut smv: Vec<_> = sm.values().collect();
let mut hmv: Vec<_> = hm.values().collect();
smv.sort();
hmv.sort();
smv == hmv
}
}
#[cfg(feature = "serde")]
#[test]
fn slotmap_serde() {
let mut sm = SlotMap::new();
let first = sm.insert_with_key(|k| (k, 23i32));
let second = sm.insert((first, 42));
let empties = vec![sm.insert((first, 0)), sm.insert((first, 0))];
empties.iter().for_each(|k| {
sm.remove(*k);
});
let third = sm.insert((second, 0));
sm[first].0 = third;
let ser = serde_json::to_string(&sm).unwrap();
let de: SlotMap<DefaultKey, (DefaultKey, i32)> = serde_json::from_str(&ser).unwrap();
assert_eq!(de.len(), sm.len());
let mut smkv: Vec<_> = sm.iter().collect();
let mut dekv: Vec<_> = de.iter().collect();
smkv.sort();
dekv.sort();
assert_eq!(smkv, dekv);
}
#[cfg(feature = "serde")]
#[test]
fn slotmap_serde_freelist() {
let mut sm = SlotMap::new();
let k = sm.insert(5i32);
sm.remove(k);
let ser = serde_json::to_string(&sm).unwrap();
let mut de: SlotMap<DefaultKey, i32> = serde_json::from_str(&ser).unwrap();
de.insert(0);
de.insert(1);
de.insert(2);
assert_eq!(de.len(), 3);
}
}