use std::iter::FusedIterator;
use std::marker::PhantomData;
use std::ops::{Add, AddAssign, Sub, SubAssign};
use typenum::Unsigned;
pub(crate) struct RawIndex<N: Unsigned>(usize, PhantomData<N>);
impl<N: Unsigned> Clone for RawIndex<N> {
#[inline]
#[must_use]
fn clone(&self) -> Self {
self.0.into()
}
}
impl<N> Copy for RawIndex<N> where N: Unsigned {}
impl<N: Unsigned> RawIndex<N> {
#[inline]
#[must_use]
pub(crate) fn to_usize(self) -> usize {
self.0
}
#[inline]
#[must_use]
pub(crate) fn inc(&mut self) -> Self {
let old = *self;
self.0 = if self.0 == N::USIZE - 1 {
0
} else {
self.0 + 1
};
old
}
#[inline]
#[must_use]
pub(crate) fn dec(&mut self) -> Self {
self.0 = if self.0 == 0 {
N::USIZE - 1
} else {
self.0 - 1
};
*self
}
}
impl<N: Unsigned> From<usize> for RawIndex<N> {
#[inline]
#[must_use]
fn from(index: usize) -> Self {
debug_assert!(index < N::USIZE);
RawIndex(index, PhantomData)
}
}
impl<N: Unsigned> PartialEq for RawIndex<N> {
#[inline]
#[must_use]
fn eq(&self, other: &Self) -> bool {
self.0 == other.0
}
}
impl<N: Unsigned> Eq for RawIndex<N> {}
impl<N: Unsigned> Add for RawIndex<N> {
type Output = RawIndex<N>;
#[inline]
#[must_use]
fn add(self, other: Self) -> Self::Output {
self + other.0
}
}
impl<N: Unsigned> Add<usize> for RawIndex<N> {
type Output = RawIndex<N>;
#[inline]
#[must_use]
fn add(self, other: usize) -> Self::Output {
let mut result = self.0 + other;
while result >= N::USIZE {
result -= N::USIZE;
}
result.into()
}
}
impl<N: Unsigned> AddAssign<usize> for RawIndex<N> {
#[inline]
fn add_assign(&mut self, other: usize) {
self.0 += other;
while self.0 >= N::USIZE {
self.0 -= N::USIZE;
}
}
}
impl<N: Unsigned> Sub for RawIndex<N> {
type Output = RawIndex<N>;
#[inline]
#[must_use]
fn sub(self, other: Self) -> Self::Output {
self - other.0
}
}
impl<N: Unsigned> Sub<usize> for RawIndex<N> {
type Output = RawIndex<N>;
#[inline]
#[must_use]
fn sub(self, other: usize) -> Self::Output {
let mut start = self.0;
while other > start {
start += N::USIZE;
}
(start - other).into()
}
}
impl<N: Unsigned> SubAssign<usize> for RawIndex<N> {
#[inline]
fn sub_assign(&mut self, other: usize) {
while other > self.0 {
self.0 += N::USIZE;
}
self.0 -= other;
}
}
pub(crate) struct IndexIter<N: Unsigned> {
pub(crate) remaining: usize,
pub(crate) left_index: RawIndex<N>,
pub(crate) right_index: RawIndex<N>,
}
impl<N: Unsigned> Iterator for IndexIter<N> {
type Item = RawIndex<N>;
#[inline]
fn next(&mut self) -> Option<Self::Item> {
if self.remaining > 0 {
self.remaining -= 1;
Some(self.left_index.inc())
} else {
None
}
}
#[inline]
#[must_use]
fn size_hint(&self) -> (usize, Option<usize>) {
(self.remaining, Some(self.remaining))
}
}
impl<N: Unsigned> DoubleEndedIterator for IndexIter<N> {
#[inline]
fn next_back(&mut self) -> Option<Self::Item> {
if self.remaining > 0 {
self.remaining -= 1;
Some(self.right_index.dec())
} else {
None
}
}
}
impl<N: Unsigned> ExactSizeIterator for IndexIter<N> {}
impl<N: Unsigned> FusedIterator for IndexIter<N> {}