use core::fmt::{Debug, Error, Formatter};
use core::iter::FromIterator;
use core::mem::{self, MaybeUninit};
use core::ops::Index;
use core::ops::IndexMut;
use core::ptr;
use core::slice::{from_raw_parts, from_raw_parts_mut};
#[cfg(feature = "std")]
use std::collections::{BTreeMap, HashMap};
use typenum::U64;
use bitmaps::{Bitmap, Bits, Iter as BitmapIter};
use crate::types::ChunkLength;
mod iter;
pub use self::iter::{Drain, Iter, IterMut, OptionDrain, OptionIter, OptionIterMut};
#[cfg(feature = "refpool")]
mod refpool;
pub struct SparseChunk<A, N: Bits + ChunkLength<A> = U64> {
map: Bitmap<N>,
data: MaybeUninit<N::SizedType>,
}
impl<A, N: Bits + ChunkLength<A>> Drop for SparseChunk<A, N> {
fn drop(&mut self) {
if mem::needs_drop::<A>() {
let bits = self.map;
for index in &bits {
unsafe { ptr::drop_in_place(&mut self.values_mut()[index]) }
}
}
}
}
impl<A: Clone, N: Bits + ChunkLength<A>> Clone for SparseChunk<A, N> {
fn clone(&self) -> Self {
let mut out = Self::new();
for index in &self.map {
out.insert(index, self[index].clone());
}
out
}
}
impl<A, N> SparseChunk<A, N>
where
N: Bits + ChunkLength<A>,
{
pub const CAPACITY: usize = N::USIZE;
#[inline]
fn values(&self) -> &[A] {
unsafe { from_raw_parts(&self.data as *const _ as *const A, N::USIZE) }
}
#[inline]
fn values_mut(&mut self) -> &mut [A] {
unsafe { from_raw_parts_mut(&mut self.data as *mut _ as *mut A, N::USIZE) }
}
#[inline]
unsafe fn force_read(index: usize, chunk: &Self) -> A {
ptr::read(&chunk.values()[index as usize])
}
#[inline]
unsafe fn force_write(index: usize, value: A, chunk: &mut Self) {
ptr::write(&mut chunk.values_mut()[index as usize], value)
}
pub fn new() -> Self {
Self {
map: Bitmap::default(),
data: MaybeUninit::uninit(),
}
}
pub fn unit(index: usize, value: A) -> Self {
let mut chunk = Self::new();
chunk.insert(index, value);
chunk
}
pub fn pair(index1: usize, value1: A, index2: usize, value2: A) -> Self {
let mut chunk = Self::new();
chunk.insert(index1, value1);
chunk.insert(index2, value2);
chunk
}
#[inline]
pub fn len(&self) -> usize {
self.map.len()
}
#[inline]
pub fn is_empty(&self) -> bool {
self.map.len() == 0
}
#[inline]
pub fn is_full(&self) -> bool {
self.len() == N::USIZE
}
pub fn insert(&mut self, index: usize, value: A) -> Option<A> {
if index >= N::USIZE {
panic!("SparseChunk::insert: index out of bounds");
}
if self.map.set(index, true) {
Some(mem::replace(&mut self.values_mut()[index], value))
} else {
unsafe { SparseChunk::force_write(index, value, self) };
None
}
}
pub fn remove(&mut self, index: usize) -> Option<A> {
if index >= N::USIZE {
panic!("SparseChunk::remove: index out of bounds");
}
if self.map.set(index, false) {
Some(unsafe { SparseChunk::force_read(index, self) })
} else {
None
}
}
pub fn pop(&mut self) -> Option<A> {
self.first_index().and_then(|index| self.remove(index))
}
pub fn get(&self, index: usize) -> Option<&A> {
if index >= N::USIZE {
return None;
}
if self.map.get(index) {
Some(unsafe { self.get_unchecked(index) })
} else {
None
}
}
pub fn get_mut(&mut self, index: usize) -> Option<&mut A> {
if index >= N::USIZE {
return None;
}
if self.map.get(index) {
Some(unsafe { self.get_unchecked_mut(index) })
} else {
None
}
}
pub unsafe fn get_unchecked(&self, index: usize) -> &A {
self.values().get_unchecked(index)
}
pub unsafe fn get_unchecked_mut(&mut self, index: usize) -> &mut A {
self.values_mut().get_unchecked_mut(index)
}
pub fn indices(&self) -> BitmapIter<'_, N> {
self.map.into_iter()
}
pub fn first_index(&self) -> Option<usize> {
self.map.first_index()
}
pub fn iter(&self) -> Iter<'_, A, N> {
Iter {
indices: self.indices(),
chunk: self,
}
}
pub fn iter_mut(&mut self) -> IterMut<'_, A, N> {
IterMut {
bitmap: self.map,
chunk: self,
}
}
pub fn drain(self) -> Drain<A, N> {
Drain { chunk: self }
}
pub fn entries(&self) -> impl Iterator<Item = (usize, &A)> {
self.indices().zip(self.iter())
}
pub fn option_iter(&self) -> OptionIter<'_, A, N> {
OptionIter {
chunk: self,
index: 0,
}
}
pub fn option_iter_mut(&mut self) -> OptionIterMut<'_, A, N> {
OptionIterMut {
chunk: self,
index: 0,
}
}
pub fn option_drain(self) -> OptionDrain<A, N> {
OptionDrain {
chunk: self,
index: 0,
}
}
}
impl<A, N: Bits + ChunkLength<A>> Default for SparseChunk<A, N> {
fn default() -> Self {
Self::new()
}
}
impl<A, N: Bits + ChunkLength<A>> Index<usize> for SparseChunk<A, N> {
type Output = A;
#[inline]
fn index(&self, index: usize) -> &Self::Output {
self.get(index).unwrap()
}
}
impl<A, N: Bits + ChunkLength<A>> IndexMut<usize> for SparseChunk<A, N> {
#[inline]
fn index_mut(&mut self, index: usize) -> &mut Self::Output {
self.get_mut(index).unwrap()
}
}
impl<A, N: Bits + ChunkLength<A>> IntoIterator for SparseChunk<A, N> {
type Item = A;
type IntoIter = Drain<A, N>;
#[inline]
fn into_iter(self) -> Self::IntoIter {
self.drain()
}
}
impl<A, N: Bits + ChunkLength<A>> FromIterator<Option<A>> for SparseChunk<A, N> {
fn from_iter<I>(iter: I) -> Self
where
I: IntoIterator<Item = Option<A>>,
{
let mut out = Self::new();
for (index, value) in iter.into_iter().enumerate() {
if let Some(value) = value {
out.insert(index, value);
}
}
out
}
}
impl<A, N> PartialEq for SparseChunk<A, N>
where
A: PartialEq,
N: Bits + ChunkLength<A>,
{
fn eq(&self, other: &Self) -> bool {
if self.map != other.map {
return false;
}
for index in self.indices() {
if self.get(index) != other.get(index) {
return false;
}
}
true
}
}
#[cfg(feature = "std")]
impl<A, N> PartialEq<BTreeMap<usize, A>> for SparseChunk<A, N>
where
A: PartialEq,
N: Bits + ChunkLength<A>,
{
fn eq(&self, other: &BTreeMap<usize, A>) -> bool {
if self.len() != other.len() {
return false;
}
for index in self.indices() {
if self.get(index) != other.get(&index) {
return false;
}
}
true
}
}
#[cfg(feature = "std")]
impl<A, N> PartialEq<HashMap<usize, A>> for SparseChunk<A, N>
where
A: PartialEq,
N: Bits + ChunkLength<A>,
{
fn eq(&self, other: &HashMap<usize, A>) -> bool {
if self.len() != other.len() {
return false;
}
for index in self.indices() {
if self.get(index) != other.get(&index) {
return false;
}
}
true
}
}
impl<A, N> Eq for SparseChunk<A, N>
where
A: Eq,
N: Bits + ChunkLength<A>,
{
}
impl<A, N> Debug for SparseChunk<A, N>
where
A: Debug,
N: Bits + ChunkLength<A>,
{
fn fmt(&self, f: &mut Formatter<'_>) -> Result<(), Error> {
f.write_str("SparseChunk")?;
f.debug_map().entries(self.entries()).finish()
}
}
#[cfg(test)]
mod test {
use super::*;
use typenum::U32;
#[test]
fn insert_remove_iterate() {
let mut chunk: SparseChunk<_, U32> = SparseChunk::new();
assert_eq!(None, chunk.insert(5, 5));
assert_eq!(None, chunk.insert(1, 1));
assert_eq!(None, chunk.insert(24, 42));
assert_eq!(None, chunk.insert(22, 22));
assert_eq!(Some(42), chunk.insert(24, 24));
assert_eq!(None, chunk.insert(31, 31));
assert_eq!(Some(24), chunk.remove(24));
assert_eq!(4, chunk.len());
let indices: Vec<_> = chunk.indices().collect();
assert_eq!(vec![1, 5, 22, 31], indices);
let values: Vec<_> = chunk.into_iter().collect();
assert_eq!(vec![1, 5, 22, 31], values);
}
#[test]
fn clone_chunk() {
let mut chunk: SparseChunk<_, U32> = SparseChunk::new();
assert_eq!(None, chunk.insert(5, 5));
assert_eq!(None, chunk.insert(1, 1));
assert_eq!(None, chunk.insert(24, 42));
assert_eq!(None, chunk.insert(22, 22));
let cloned = chunk.clone();
let right_indices: Vec<_> = chunk.indices().collect();
let left_indices: Vec<_> = cloned.indices().collect();
let right: Vec<_> = chunk.into_iter().collect();
let left: Vec<_> = cloned.into_iter().collect();
assert_eq!(left, right);
assert_eq!(left_indices, right_indices);
assert_eq!(vec![1, 5, 22, 24], left_indices);
assert_eq!(vec![1, 5, 22, 24], right_indices);
}
use crate::tests::DropTest;
use std::sync::atomic::{AtomicUsize, Ordering};
#[test]
fn dropping() {
let counter = AtomicUsize::new(0);
{
let mut chunk: SparseChunk<DropTest<'_>> = SparseChunk::new();
for i in 0..40 {
chunk.insert(i, DropTest::new(&counter));
}
assert_eq!(40, counter.load(Ordering::Relaxed));
for i in 0..20 {
chunk.remove(i);
}
assert_eq!(20, counter.load(Ordering::Relaxed));
}
assert_eq!(0, counter.load(Ordering::Relaxed));
}
#[test]
fn equality() {
let mut c1 = SparseChunk::<usize>::new();
for i in 0..32 {
c1.insert(i, i);
}
let mut c2 = c1.clone();
assert_eq!(c1, c2);
for i in 4..8 {
c2.insert(i, 0);
}
assert_ne!(c1, c2);
c2 = c1.clone();
for i in 0..16 {
c2.remove(i);
}
assert_ne!(c1, c2);
}
}