diff --git a/src/difference.rs b/src/difference.rs index 16c6b5fb8..5f4587448 100644 --- a/src/difference.rs +++ b/src/difference.rs @@ -309,4 +309,100 @@ mod vector { Self { buffer: Vec::new() } } } +} + +pub use self::array::DiffArray; +mod array { + use std::convert::TryInto; + use std::ops::{AddAssign, Neg, Mul}; + use super::{Semigroup, Monoid}; + + /// Like a [`DiffVector`](super::DiffVector), but the elements are stored + /// in an array rather than a vector. + #[derive(Abomonation, Ord, PartialOrd, Eq, PartialEq, Debug, Clone)] + pub struct DiffArray { + buffer: [R; N], + } + + impl DiffArray { + /// Creates a new `DiffArray` from an array. + #[inline(always)] + pub fn new(arr: [R; N]) -> DiffArray { + DiffArray { buffer: arr } + } + } + + impl IntoIterator for DiffArray { + type Item = R; + type IntoIter = ::std::array::IntoIter; + + fn into_iter(self) -> Self::IntoIter { + ::std::array::IntoIter::new(self.buffer) + } + } + + impl std::ops::Deref for DiffArray { + type Target = [R]; + + fn deref(&self) -> &Self::Target { + &self.buffer[..] + } + } + + impl std::ops::DerefMut for DiffArray { + fn deref_mut(&mut self) -> &mut Self::Target { + &mut self.buffer[..] + } + } + + impl Semigroup for DiffArray { + #[inline] + fn is_zero(&self) -> bool { + self.buffer.iter().all(|x| x.is_zero()) + } + } + + impl<'a, R: AddAssign<&'a R>+Clone, const N: usize> AddAssign<&'a DiffArray> for DiffArray { + #[inline] + fn add_assign(&mut self, rhs: &'a Self) { + for (index, update) in rhs.buffer.iter().enumerate() { + self.buffer[index] += update; + } + } + } + + impl + Clone, const N: usize> Neg for DiffArray { + type Output = DiffArray<::Output, N>; + + #[inline] + fn neg(mut self) -> Self::Output { + for update in self.buffer.iter_mut() { + *update = -update.clone(); + } + self + } + } + + impl, const N: usize> Mul for DiffArray { + type Output = DiffArray<>::Output, N>; + + fn mul(self, other: T) -> Self::Output { + // It would be nice to avoid the intermediate vector here, but + // seems to be impossible to do safely, and doing so unsafely + // is risky without some new MaybeUninit APIs: + // https://github.com/rust-lang/rust/pull/80600. + let buffer: Vec<_> = ::std::array::IntoIter::new(self.buffer) + .map(|x| x * other) + .collect(); + DiffArray { + buffer: buffer.try_into().unwrap_or_else(|_| unreachable!()), + } + } + } + + impl Monoid for DiffArray { + fn zero() -> Self { + Self { buffer: [R::zero(); N] } + } + } } \ No newline at end of file