impl<T> Step for T
where
T: crate::Newtype + Clone + PartialOrd + MinMax,
T::Inner: Step,
{
#[inline]
fn steps_between(start: &Self, end: &Self) -> (usize, Option<usize>) {
<T::Inner as Step>::steps_between(start.as_ref(), end.as_ref())
}
#[inline]
fn forward_checked(start: Self, count: usize) -> Option<Self> {
<T::Inner as Step>::forward_checked(start.as_ref().clone(), count).map(T::from)
}
#[inline]
fn backward_checked(start: Self, count: usize) -> Option<Self> {
<T::Inner as Step>::backward_checked(start.as_ref().clone(), count).map(Self::from)
}
}
#[derive(Clone)]
pub struct Iterator<T>
where
T: crate::Newtype,
{
start: T::Inner,
last: T::Inner,
}
impl<T> Iterator<T>
where
T: crate::Newtype,
T::Inner: Step,
{
#[inline]
pub fn is_empty(&self) -> bool {
self.start > self.last
}
#[inline]
pub fn from<R: std::ops::RangeBounds<T>>(range: &R) -> Self {
use crate::iter::Step;
use std::ops::Bound;
let start = match range.start_bound() {
Bound::Included(s) => s.as_ref().clone(),
Bound::Excluded(s) => Step::forward(s.as_ref().clone(), 1),
Bound::Unbounded => T::Inner::MIN,
};
let last = match range.end_bound() {
Bound::Included(e) => e.as_ref().clone(),
Bound::Excluded(e) => Step::backward(e.as_ref().clone(), 1),
Bound::Unbounded => T::Inner::MAX,
};
Self { start, last }
}
}
impl<T> std::iter::Iterator for Iterator<T>
where
T: crate::Newtype + From<T::Inner>,
T::Inner: Step + std::fmt::Debug,
{
type Item = T;
#[inline]
fn next(&mut self) -> Option<Self::Item> {
if Iterator::is_empty(self) {
return None;
}
let next = crate::iter::Step::forward_checked(self.start.clone(), 1)?;
Some(T::from(core::mem::replace(&mut self.start, next)))
}
#[inline]
fn size_hint(&self) -> (usize, Option<usize>) {
if self.is_empty() {
return (0, Some(0));
}
let hint = Step::steps_between(&self.start, &self.last);
(
hint.0.saturating_add(1),
hint.1.and_then(|steps| steps.checked_add(1)),
)
}
#[inline]
fn count(self) -> usize {
if self.is_empty() {
return 0;
}
crate::iter::Step::steps_between(&self.start, &self.last)
.1
.and_then(|steps| steps.checked_add(1))
.expect("count overflowed usize")
}
#[inline]
fn is_sorted(self) -> bool {
true
}
}
impl<T> DoubleEndedIterator for Iterator<T>
where
T: crate::Newtype + From<T::Inner>,
T::Inner: Step + std::fmt::Debug,
{
fn next_back(&mut self) -> Option<Self::Item> {
if Iterator::is_empty(self) {
return None;
}
let next = crate::iter::Step::backward_checked(self.last.clone(), 1)?;
Some(T::from(core::mem::replace(&mut self.last, next)))
}
}
impl<T> ExactSizeIterator for Iterator<T>
where
T: crate::Newtype + From<T::Inner>,
T::Inner: Step + std::fmt::Debug,
{
}
impl<T> std::iter::FusedIterator for Iterator<T>
where
T: crate::Newtype + From<T::Inner>,
T::Inner: Step + std::fmt::Debug,
{
}
pub trait MinMax {
const MAX: Self;
const MIN: Self;
}
macro_rules! impl_min_max {
($($t:ty),*) => {
$(
impl MinMax for $t {
const MAX: Self = <$t>::MAX;
const MIN: Self = <$t>::MIN;
}
)+
};
}
impl_min_max!(
u8, u16, u32, u64, u128, i8, i16, i32, i64, i128, usize, isize
);
pub trait Step: Clone + PartialOrd + MinMax + Sized {
fn steps_between(start: &Self, end: &Self) -> (usize, Option<usize>);
fn forward_checked(start: Self, count: usize) -> Option<Self>;
fn forward(start: Self, count: usize) -> Self {
Step::forward_checked(start, count).expect("overflow in `Step::forward`")
}
fn backward_checked(start: Self, count: usize) -> Option<Self>;
fn backward(start: Self, count: usize) -> Self {
Step::backward_checked(start, count).expect("overflow in `Step::backward`")
}
}
macro_rules! step_identical_methods {
() => {
#[inline]
#[allow(arithmetic_overflow)]
fn forward(start: Self, n: usize) -> Self {
if Self::forward_checked(start, n).is_none() {
let _ = Self::MAX + 1;
}
start.wrapping_add(n as Self)
}
#[inline]
#[allow(arithmetic_overflow)]
fn backward(start: Self, n: usize) -> Self {
if Self::backward_checked(start, n).is_none() {
let _ = Self::MIN - 1;
}
start.wrapping_sub(n as Self)
}
};
}
macro_rules! step_integer_impls {
{
[ $( [ $u_narrower:ident $i_narrower:ident ] ),+ ] <= usize <
[ $( [ $u_wider:ident $i_wider:ident ] ),+ ]
} => {
$(
#[allow(unreachable_patterns)]
impl Step for $u_narrower {
step_identical_methods!();
#[inline]
fn steps_between(start: &Self, end: &Self) -> (usize, Option<usize>) {
if *start <= *end {
let steps = (*end - *start) as usize;
(steps, Some(steps))
} else {
(0, None)
}
}
#[inline]
fn forward_checked(start: Self, n: usize) -> Option<Self> {
match Self::try_from(n) {
Ok(n) => start.checked_add(n),
Err(_) => None, }
}
#[inline]
fn backward_checked(start: Self, n: usize) -> Option<Self> {
match Self::try_from(n) {
Ok(n) => start.checked_sub(n),
Err(_) => None, }
}
}
#[allow(unreachable_patterns)]
impl Step for $i_narrower {
step_identical_methods!();
#[inline]
fn steps_between(start: &Self, end: &Self) -> (usize, Option<usize>) {
if *start <= *end {
let steps = (*end as isize).wrapping_sub(*start as isize) as usize;
(steps, Some(steps))
} else {
(0, None)
}
}
#[inline]
fn forward_checked(start: Self, n: usize) -> Option<Self> {
match $u_narrower::try_from(n) {
Ok(n) => {
let wrapped = start.wrapping_add(n as Self);
if wrapped >= start {
Some(wrapped)
} else {
None }
}
Err(_) => None,
}
}
#[inline]
fn backward_checked(start: Self, n: usize) -> Option<Self> {
match $u_narrower::try_from(n) {
Ok(n) => {
let wrapped = start.wrapping_sub(n as Self);
if wrapped <= start {
Some(wrapped)
} else {
None }
}
Err(_) => None,
}
}
}
)+
$(
#[allow(unreachable_patterns)]
impl Step for $u_wider {
step_identical_methods!();
#[inline]
fn steps_between(start: &Self, end: &Self) -> (usize, Option<usize>) {
if *start <= *end {
if let Ok(steps) = usize::try_from(*end - *start) {
(steps, Some(steps))
} else {
(usize::MAX, None)
}
} else {
(0, None)
}
}
#[inline]
fn forward_checked(start: Self, n: usize) -> Option<Self> {
start.checked_add(n as Self)
}
#[inline]
fn backward_checked(start: Self, n: usize) -> Option<Self> {
start.checked_sub(n as Self)
}
}
#[allow(unreachable_patterns)]
impl Step for $i_wider {
step_identical_methods!();
#[inline]
fn steps_between(start: &Self, end: &Self) -> (usize, Option<usize>) {
if *start <= *end {
match end.checked_sub(*start) {
Some(result) => {
if let Ok(steps) = usize::try_from(result) {
(steps, Some(steps))
} else {
(usize::MAX, None)
}
}
None => (usize::MAX, None),
}
} else {
(0, None)
}
}
#[inline]
fn forward_checked(start: Self, n: usize) -> Option<Self> {
start.checked_add(n as Self)
}
#[inline]
fn backward_checked(start: Self, n: usize) -> Option<Self> {
start.checked_sub(n as Self)
}
}
)+
};
}
#[cfg(target_pointer_width = "64")]
step_integer_impls! {
[ [u8 i8], [u16 i16], [u32 i32], [u64 i64], [usize isize] ] <= usize < [ [u128 i128] ]
}
#[cfg(target_pointer_width = "32")]
step_integer_impls! {
[ [u8 i8], [u16 i16], [u32 i32], [usize isize] ] <= usize < [ [u64 i64], [u128 i128] ]
}
#[cfg(target_pointer_width = "16")]
step_integer_impls! {
[ [u8 i8], [u16 i16], [usize isize] ] <= usize < [ [u32 i32], [u64 i64], [u128 i128] ]
}