Skip to content

Commit

Permalink
TryPush trait to enable non-reallocating pushes
Browse files Browse the repository at this point in the history
Signed-off-by: Moritz Hoffmann <[email protected]>
  • Loading branch information
antiguru committed Jul 3, 2024
1 parent ebec264 commit cd20c2c
Show file tree
Hide file tree
Showing 4 changed files with 170 additions and 4 deletions.
46 changes: 43 additions & 3 deletions src/impls/offsets.rs
Original file line number Diff line number Diff line change
Expand Up @@ -248,7 +248,11 @@ where
/// Reserve space for `additional` elements.
#[inline]
pub fn reserve(&mut self, additional: usize) {
self.smol.reserve(additional);
if self.chonk.is_empty() {
self.smol.reserve(additional);
} else {
self.chonk.reserve(additional);
}
}

/// Remove all elements.
Expand Down Expand Up @@ -299,6 +303,15 @@ where
fn is_empty(&self) -> bool {
self.is_empty()
}

#[inline]
fn capacity(&self) -> usize {
if self.chonk.is_empty() {
self.smol.capacity()
} else {
self.chonk.capacity()
}
}
}

impl<S, L> OffsetContainer<usize> for OffsetList<S, L>
Expand Down Expand Up @@ -407,6 +420,16 @@ where
fn heap_size<F: FnMut(usize, usize)>(&self, callback: F) {
self.spilled.heap_size(callback);
}

#[inline]
fn capacity(&self) -> usize {
if self.spilled.is_empty() {
// TODO: What else could we do here? `usize::MAX`?
1
} else {
self.spilled.capacity()
}
}
}

impl<S, L> OffsetContainer<usize> for OffsetOptimized<S, L>
Expand Down Expand Up @@ -439,8 +462,25 @@ where
where
I::IntoIter: ExactSizeIterator,
{
for item in iter {
self.push(item);
let mut iter = iter.into_iter();
if !self.spilled.is_empty() {
// We certainly push into `spilled`, so reserve enough space.
let (lower, upper) = iter.size_hint();
self.spilled.reserve(upper.unwrap_or(lower));
}
while let Some(item) = iter.next() {
if self.spilled.is_empty() {
let inserted = self.strided.push(item);
if !inserted {
// This is a transition point from pushing into `strided` to pushing into
// `spilled`.
let (lower, upper) = iter.size_hint();
self.spilled.reserve(upper.unwrap_or(lower));
self.spilled.push(item);
}
} else {
self.spilled.push(item);
}
}
}

Expand Down
86 changes: 85 additions & 1 deletion src/impls/slice_owned.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ use std::marker::PhantomData;
use serde::{Deserialize, Serialize};

use crate::impls::storage::{PushStorage, Storage};
use crate::{Push, PushIter, Region, ReserveItems};
use crate::{Push, PushIter, Region, ReserveItems, TryPush};

/// A container for owned types.
///
Expand Down Expand Up @@ -128,6 +128,30 @@ where
}
}

impl<T, S, const N: usize> TryPush<[T; N]> for OwnedRegion<T, S>
where
[T]: ToOwned,
S: Storage<T>
+ for<'a> PushStorage<PushIter<[T; N]>>
+ std::ops::Index<std::ops::Range<usize>, Output = [T]>,
{
#[inline]
fn try_push(&mut self, item: [T; N]) -> Result<<OwnedRegion<T> as Region>::Index, [T; N]> {
if self.can_push(&item) {
let start = self.slices.len();
self.slices.push_storage(PushIter(item));
Ok((start, self.slices.len()))
} else {
Err(item)
}
}

#[inline]
fn can_push(&self, item: &[T; N]) -> bool {
self.slices.capacity() - self.slices.len() >= item.len()
}
}

impl<T, S, const N: usize> Push<&[T; N]> for OwnedRegion<T, S>
where
T: Clone,
Expand All @@ -141,6 +165,27 @@ where
}
}

impl<T, S, const N: usize> TryPush<&[T; N]> for OwnedRegion<T, S>
where
T: Clone,
S: Storage<T>
+ for<'a> PushStorage<&'a [T]>
+ std::ops::Index<std::ops::Range<usize>, Output = [T]>,
{
#[inline]
fn try_push<'a>(
&mut self,
item: &'a [T; N],
) -> Result<<OwnedRegion<T> as Region>::Index, &'a [T; N]> {
self.try_push(item.as_slice()).map_err(|_| item)
}

#[inline]
fn can_push(&self, item: &&[T; N]) -> bool {
self.can_push(&item.as_slice())
}
}

impl<T, S, const N: usize> Push<&&[T; N]> for OwnedRegion<T, S>
where
T: Clone,
Expand Down Expand Up @@ -183,6 +228,33 @@ where
}
}

impl<T, S> TryPush<&[T]> for OwnedRegion<T, S>
where
T: Clone,
S: Storage<T>
+ for<'a> PushStorage<&'a [T]>
+ std::ops::Index<std::ops::Range<usize>, Output = [T]>,
{
#[inline]
fn try_push<'a>(
&mut self,
item: &'a [T],
) -> Result<<OwnedRegion<T, S> as Region>::Index, &'a [T]> {
if self.can_push(&item) {
let start = self.slices.len();
self.slices.push_storage(item);
Ok((start, self.slices.len()))
} else {
Err(item)
}
}

#[inline]
fn can_push(&self, item: &&[T]) -> bool {
self.slices.capacity() - self.slices.len() >= item.len()
}
}

impl<T: Clone, S: Storage<T>> Push<&&[T]> for OwnedRegion<T, S>
where
for<'a> Self: Push<&'a [T]>,
Expand Down Expand Up @@ -322,4 +394,16 @@ mod tests {
let index = r.push(PushIter(iter));
assert_eq!([1, 1, 1, 1], r.index(index));
}

#[test]
fn try_push() {
let mut r = <OwnedRegion<u8>>::default();
assert!(!r.can_push(&[1; 4]));
assert_eq!(r.try_push(&[1; 4]), Err(&[1; 4]));
r.reserve_items(std::iter::once(&[1; 4]));
assert!(r.can_push(&[1; 4]));
let index = r.try_push(&[1; 4]);
assert_eq!(index, Ok((0, 4)));
assert_eq!([1, 1, 1, 1], r.index(index.unwrap()));
}
}
9 changes: 9 additions & 0 deletions src/impls/storage.rs
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,10 @@ pub trait Storage<T>: Default {
/// Returns `true` if empty, i.e., it doesn't contain any elements.
#[must_use]
fn is_empty(&self) -> bool;

/// Returns the capacity of the storage.
#[must_use]
fn capacity(&self) -> usize;
}

impl<T> Storage<T> for Vec<T> {
Expand Down Expand Up @@ -83,6 +87,11 @@ impl<T> Storage<T> for Vec<T> {
fn is_empty(&self) -> bool {
self.is_empty()
}

#[inline]
fn capacity(&self) -> usize {
self.capacity()
}
}

/// Push an item into storage.
Expand Down
33 changes: 33 additions & 0 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -112,6 +112,18 @@ pub trait Push<T>: Region {
fn push(&mut self, item: T) -> Self::Index;
}

/// Push an item `T` into a region.
pub trait TryPush<T>: Region {
/// Push `item` into self, returning an index that allows to look up the
/// corresponding read item.
#[must_use]
fn try_push(&mut self, item: T) -> Result<Self::Index, T>;

Check failure on line 120 in src/lib.rs

View workflow job for this annotation

GitHub Actions / cargo test on ubuntu, rust stable

this function has an empty `#[must_use]` attribute, but returns a type already marked as `#[must_use]`

Check failure on line 120 in src/lib.rs

View workflow job for this annotation

GitHub Actions / cargo test on ubuntu, rust 1.78

this function has an empty `#[must_use]` attribute, but returns a type already marked as `#[must_use]`

Check failure on line 120 in src/lib.rs

View workflow job for this annotation

GitHub Actions / cargo test on macos, rust stable

this function has an empty `#[must_use]` attribute, but returns a type already marked as `#[must_use]`

Check failure on line 120 in src/lib.rs

View workflow job for this annotation

GitHub Actions / cargo test on macos, rust 1.78

this function has an empty `#[must_use]` attribute, but returns a type already marked as `#[must_use]`

/// Test if an item can be pushed into the region without reallocation.
#[must_use]
fn can_push(&self, item: &T) -> bool;
}

/// Reserve space in the receiving region.
///
/// Closely related to [`Push`], but separate because target type is likely different.
Expand Down Expand Up @@ -226,6 +238,27 @@ impl<R: Region, S: OffsetContainer<<R as Region>::Index>> FlatStack<R, S> {
self.indices.push(index);
}

/// Appends the element to the back of the stack, if there is sufficient capacity
#[inline]
pub fn try_push<T>(&mut self, item: T) -> Result<(), T>
where
R: TryPush<T>,
{
let index = self.region.try_push(item)?;
self.indices.push(index);
Ok(())
}

/// Appends the element to the back of the stack, if there is sufficient capacity
#[inline]
pub fn can_push<T>(&mut self, item: &T) -> bool
where
R: TryPush<T>,
{
// TODO: Include `indices` in the check.
self.region.can_push(item)
}

/// Returns the element at the `offset` position.
#[inline]
#[must_use]
Expand Down

0 comments on commit cd20c2c

Please sign in to comment.