diff --git a/library/alloc/src/collections/vec_deque/mod.rs b/library/alloc/src/collections/vec_deque/mod.rs index f8844e2d3a5cb..bd15a6a96f52c 100644 --- a/library/alloc/src/collections/vec_deque/mod.rs +++ b/library/alloc/src/collections/vec_deque/mod.rs @@ -9,7 +9,7 @@ use core::cmp::{self, Ordering}; use core::hash::{Hash, Hasher}; -use core::iter::{ByRefSized, repeat_n, repeat_with}; +use core::iter::{repeat_n, repeat_with}; // This is used in a bunch of intra-doc links. // FIXME: For some reason, `#[cfg(doc)]` wasn't sufficient, resulting in // failures in linkchecker even though rustdoc built the docs just fine. @@ -477,11 +477,7 @@ impl VecDeque { unsafe { guard.deque.write_iter(dst, iter, &mut guard.written) }; } else { unsafe { - guard.deque.write_iter( - dst, - ByRefSized(&mut iter).take(head_room), - &mut guard.written, - ); + guard.deque.write_iter(dst, (&mut iter).take(head_room), &mut guard.written); guard.deque.write_iter(0, iter, &mut guard.written) }; } diff --git a/library/core/src/array/iter.rs b/library/core/src/array/iter.rs index 1edade41597f7..3b4574c4e9544 100644 --- a/library/core/src/array/iter.rs +++ b/library/core/src/array/iter.rs @@ -260,7 +260,7 @@ impl Iterator for IntoIter { Fold: FnMut(Acc, Self::Item) -> Acc, { let data = &mut self.data; - iter::ByRefSized(&mut self.alive).fold(init, |acc, idx| { + (&mut self.alive).fold(init, |acc, idx| { // SAFETY: idx is obtained by folding over the `alive` range, which implies the // value is currently considered alive but as the range is being consumed each value // we read here will only be read once and then considered dead. @@ -323,7 +323,7 @@ impl DoubleEndedIterator for IntoIter { Fold: FnMut(Acc, Self::Item) -> Acc, { let data = &mut self.data; - iter::ByRefSized(&mut self.alive).rfold(init, |acc, idx| { + (&mut self.alive).rfold(init, |acc, idx| { // SAFETY: idx is obtained by folding over the `alive` range, which implies the // value is currently considered alive but as the range is being consumed each value // we read here will only be read once and then considered dead. diff --git a/library/core/src/iter/adapters/array_chunks.rs b/library/core/src/iter/adapters/array_chunks.rs index 8f1744fc5fbb7..ca1a0a67ff8e5 100644 --- a/library/core/src/iter/adapters/array_chunks.rs +++ b/library/core/src/iter/adapters/array_chunks.rs @@ -1,8 +1,6 @@ use crate::array; use crate::iter::adapters::SourceIter; -use crate::iter::{ - ByRefSized, FusedIterator, InPlaceIterable, TrustedFused, TrustedRandomAccessNoCoerce, -}; +use crate::iter::{FusedIterator, InPlaceIterable, TrustedFused, TrustedRandomAccessNoCoerce}; use crate::num::NonZero; use crate::ops::{ControlFlow, NeverShortCircuit, Try}; @@ -128,7 +126,7 @@ where self.next_back_remainder(); let mut acc = init; - let mut iter = ByRefSized(&mut self.iter).rev(); + let mut iter = (&mut self.iter).rev(); // NB remainder is handled by `next_back_remainder`, so // `next_chunk` can't return `Err` with non-empty remainder diff --git a/library/core/src/iter/adapters/by_ref.rs b/library/core/src/iter/adapters/by_ref.rs new file mode 100644 index 0000000000000..d7b105e7dac0b --- /dev/null +++ b/library/core/src/iter/adapters/by_ref.rs @@ -0,0 +1,554 @@ +use crate::array; +use crate::iter::{FusedIterator, TrustedLen}; +use crate::num::NonZero; +use crate::ops::{ChangeOutputType, NeverShortCircuit, Residual, Try}; + +/// Implements `Iterator` for mutable references to iterators, such as those produced by [`Iterator::by_ref`]. +/// +/// This implementation passes all method calls on to the original iterator. +#[stable(feature = "rust1", since = "1.0.0")] +impl Iterator for &mut I { + type Item = I::Item; + + #[inline] + fn next(&mut self) -> Option { + ByRef::(self).next() + } + + fn next_chunk( + &mut self, + ) -> Result<[Self::Item; N], array::IntoIter> { + ByRef::(self).next_chunk() + } + + fn size_hint(&self) -> (usize, Option) { + I::size_hint(self) + } + + fn advance_by(&mut self, n: usize) -> Result<(), NonZero> { + ByRef::(self).advance_by(n) + } + + fn nth(&mut self, n: usize) -> Option { + ByRef::(self).nth(n) + } + + fn for_each(self, f: F) + where + F: FnMut(Self::Item), + { + ByRef::(self).for_each(f) + } + + fn try_fold(&mut self, init: B, f: F) -> R + where + F: FnMut(B, Self::Item) -> R, + R: Try, + { + ByRef::(self).try_fold(init, f) + } + + fn try_for_each(&mut self, f: F) -> R + where + F: FnMut(Self::Item) -> R, + R: Try, + { + ByRef::(self).try_for_each(f) + } + + fn fold(self, init: B, f: F) -> B + where + F: FnMut(B, Self::Item) -> B, + { + ByRef::(self).fold(init, f) + } + + fn reduce(self, f: F) -> Option + where + F: FnMut(Self::Item, Self::Item) -> Self::Item, + { + ByRef::(self).reduce(f) + } + + fn try_reduce( + &mut self, + f: impl FnMut(Self::Item, Self::Item) -> R, + ) -> ChangeOutputType> + where + R: Try>>, + { + ByRef::(self).try_reduce(f) + } + + fn all(&mut self, f: F) -> bool + where + F: FnMut(Self::Item) -> bool, + { + ByRef::(self).all(f) + } + + fn any(&mut self, f: F) -> bool + where + F: FnMut(Self::Item) -> bool, + { + ByRef::(self).any(f) + } + + fn find

(&mut self, predicate: P) -> Option + where + P: FnMut(&Self::Item) -> bool, + { + ByRef::(self).find(predicate) + } + + fn find_map(&mut self, f: F) -> Option + where + F: FnMut(Self::Item) -> Option, + { + ByRef::(self).find_map(f) + } + + fn try_find( + &mut self, + f: impl FnMut(&Self::Item) -> R, + ) -> ChangeOutputType> + where + R: Try>>, + { + ByRef::(self).try_find(f) + } + + fn position

(&mut self, predicate: P) -> Option + where + P: FnMut(Self::Item) -> bool, + { + ByRef::(self).position(predicate) + } + + // FIXME: also specialize rposition + // + // Unfortunately, it's not possible to infer + // + // `I: DoubleEndedIterator + ExactSizeIterator` + // + // from + // + // `&mut I: DoubleEndedIterator + ExactSizeIterator` + // + // which makes it impossible to call the inner `rposition`, even with + // specialization. +} + +#[stable(feature = "rust1", since = "1.0.0")] +impl DoubleEndedIterator for &mut I { + fn next_back(&mut self) -> Option { + ByRef::(self).next_back() + } + + fn advance_back_by(&mut self, n: usize) -> Result<(), NonZero> { + ByRef::(self).advance_back_by(n) + } + + fn nth_back(&mut self, n: usize) -> Option { + ByRef::(self).nth_back(n) + } + + fn try_rfold(&mut self, init: B, f: F) -> R + where + F: FnMut(B, Self::Item) -> R, + R: Try, + { + ByRef::(self).try_rfold(init, f) + } + + fn rfold(self, init: B, f: F) -> B + where + F: FnMut(B, Self::Item) -> B, + { + ByRef::(self).rfold(init, f) + } + + fn rfind

(&mut self, predicate: P) -> Option + where + P: FnMut(&Self::Item) -> bool, + { + ByRef::(self).rfind(predicate) + } +} + +#[stable(feature = "rust1", since = "1.0.0")] +impl ExactSizeIterator for &mut I { + fn len(&self) -> usize { + I::len(self) + } + + fn is_empty(&self) -> bool { + I::is_empty(self) + } +} + +#[stable(feature = "fused", since = "1.26.0")] +impl FusedIterator for &mut I {} + +#[unstable(feature = "trusted_len", issue = "37572")] +unsafe impl TrustedLen for &mut I {} + +// The following implementations use UFCS-style, rather than trusting autoderef, +// to avoid accidentally calling the `&mut Iterator` implementations. + +/// A helper struct that implements all overridden `Iterator` methods by +/// forwarding them to either [`ByRefDefault`] or the underlying iterator. +/// +/// It's `std`'s policy to not expose specialization publicly, hence we do not +/// specialize the public `Iterator for &mut I` implementation but only the +/// `Iterator` implementation of this private helper struct. +struct ByRef<'a, I: ?Sized>(&'a mut I); + +// Make sure to add all iterator methods that can be forwarded in any way to +// a `Sized` underlying iterator here. Sometimes, e.g. in the case of `fold` +// that can involve going through the fallible version of a functions, since +// they mostly take a reference instead of consuming the iterator. +impl Iterator for ByRef<'_, I> { + type Item = I::Item; + + #[inline] + fn next(&mut self) -> Option { + ByRefDefault::(self.0).next() + } + + default fn next_chunk( + &mut self, + ) -> Result<[Self::Item; N], array::IntoIter> { + ByRefDefault::(self.0).next_chunk() + } + + fn size_hint(&self) -> (usize, Option) { + I::size_hint(self.0) + } + + fn advance_by(&mut self, n: usize) -> Result<(), NonZero> { + ByRefDefault::(self.0).advance_by(n) + } + + fn nth(&mut self, n: usize) -> Option { + ByRefDefault::(self.0).nth(n) + } + + default fn for_each(self, f: F) + where + F: FnMut(Self::Item), + { + ByRefDefault::(self.0).for_each(f); + } + + default fn try_fold(&mut self, init: B, f: F) -> R + where + F: FnMut(B, Self::Item) -> R, + R: Try, + { + ByRefDefault::(self.0).try_fold(init, f) + } + + default fn try_for_each(&mut self, f: F) -> R + where + F: FnMut(Self::Item) -> R, + R: Try, + { + ByRefDefault::(self.0).try_for_each(f) + } + + default fn fold(self, init: B, f: F) -> B + where + F: FnMut(B, Self::Item) -> B, + { + ByRefDefault::(self.0).fold(init, f) + } + + default fn reduce(self, f: F) -> Option + where + F: FnMut(Self::Item, Self::Item) -> Self::Item, + { + ByRefDefault::(self.0).reduce(f) + } + + default fn try_reduce( + &mut self, + f: impl FnMut(Self::Item, Self::Item) -> R, + ) -> ChangeOutputType> + where + R: Try>>, + { + ByRefDefault::(self.0).try_reduce(f) + } + + default fn all(&mut self, f: F) -> bool + where + F: FnMut(Self::Item) -> bool, + { + ByRefDefault::(self.0).all(f) + } + + default fn any(&mut self, f: F) -> bool + where + F: FnMut(Self::Item) -> bool, + { + ByRefDefault::(self.0).any(f) + } + + default fn find

(&mut self, predicate: P) -> Option + where + P: FnMut(&Self::Item) -> bool, + { + ByRefDefault::(self.0).find(predicate) + } + + default fn find_map(&mut self, f: F) -> Option + where + F: FnMut(Self::Item) -> Option, + { + ByRefDefault::(self.0).find_map(f) + } + + default fn try_find( + &mut self, + f: impl FnMut(&Self::Item) -> R, + ) -> ChangeOutputType> + where + R: Try>>, + { + ByRefDefault::(self.0).try_find(f) + } + + default fn position

(&mut self, predicate: P) -> Option + where + P: FnMut(Self::Item) -> bool, + { + ByRefDefault::(self.0).position(predicate) + } +} + +impl Iterator for ByRef<'_, I> { + fn next_chunk( + &mut self, + ) -> Result<[Self::Item; N], array::IntoIter> { + I::next_chunk(self.0) + } + + fn for_each(self, f: F) + where + F: FnMut(Self::Item), + { + I::try_for_each(self.0, NeverShortCircuit::wrap_mut_1(f)).0 + } + + fn try_fold(&mut self, init: B, f: F) -> R + where + F: FnMut(B, Self::Item) -> R, + R: Try, + { + I::try_fold(self.0, init, f) + } + + fn try_for_each(&mut self, f: F) -> R + where + F: FnMut(Self::Item) -> R, + R: Try, + { + I::try_for_each(self.0, f) + } + + fn fold(self, init: B, f: F) -> B + where + F: FnMut(B, Self::Item) -> B, + { + I::try_fold(self.0, init, NeverShortCircuit::wrap_mut_2(f)).0 + } + + fn reduce(self, f: F) -> Option + where + F: FnMut(Self::Item, Self::Item) -> Self::Item, + { + I::try_reduce(self.0, NeverShortCircuit::wrap_mut_2(f)).0 + } + + fn try_reduce( + &mut self, + f: impl FnMut(Self::Item, Self::Item) -> R, + ) -> ChangeOutputType> + where + R: Try>>, + { + I::try_reduce(self.0, f) + } + + fn all(&mut self, f: F) -> bool + where + F: FnMut(Self::Item) -> bool, + { + I::all(self.0, f) + } + + fn any(&mut self, f: F) -> bool + where + F: FnMut(Self::Item) -> bool, + { + I::any(self.0, f) + } + + fn find

(&mut self, predicate: P) -> Option + where + P: FnMut(&Self::Item) -> bool, + { + I::find(self.0, predicate) + } + + fn find_map(&mut self, f: F) -> Option + where + F: FnMut(Self::Item) -> Option, + { + I::find_map(self.0, f) + } + + fn try_find( + &mut self, + f: impl FnMut(&Self::Item) -> R, + ) -> ChangeOutputType> + where + R: Try>>, + { + I::try_find(self.0, f) + } + + fn position

(&mut self, predicate: P) -> Option + where + P: FnMut(Self::Item) -> bool, + { + I::position(self.0, predicate) + } +} + +impl DoubleEndedIterator for ByRef<'_, I> { + fn next_back(&mut self) -> Option { + ByRefDefault::(self.0).next_back() + } + + fn advance_back_by(&mut self, n: usize) -> Result<(), NonZero> { + ByRefDefault::(self.0).advance_back_by(n) + } + + fn nth_back(&mut self, n: usize) -> Option { + ByRefDefault::(self.0).nth_back(n) + } + + default fn try_rfold(&mut self, init: B, f: F) -> R + where + F: FnMut(B, Self::Item) -> R, + R: Try, + { + ByRefDefault::(self.0).try_rfold(init, f) + } + + default fn rfold(self, init: B, f: F) -> B + where + F: FnMut(B, Self::Item) -> B, + { + ByRefDefault::(self.0).rfold(init, f) + } + + default fn rfind

(&mut self, predicate: P) -> Option + where + P: FnMut(&Self::Item) -> bool, + { + ByRefDefault::(self.0).rfind(predicate) + } +} + +impl DoubleEndedIterator for ByRef<'_, I> { + fn try_rfold(&mut self, init: B, f: F) -> R + where + F: FnMut(B, Self::Item) -> R, + R: Try, + { + I::try_rfold(self.0, init, f) + } + + fn rfold(self, init: B, f: F) -> B + where + F: FnMut(B, Self::Item) -> B, + { + I::try_rfold(self.0, init, NeverShortCircuit::wrap_mut_2(f)).0 + } + + fn rfind

(&mut self, predicate: P) -> Option + where + P: FnMut(&Self::Item) -> bool, + { + I::rfind(self.0, predicate) + } +} + +impl ExactSizeIterator for ByRef<'_, I> { + fn len(&self) -> usize { + I::len(self.0) + } + + fn is_empty(&self) -> bool { + I::is_empty(self.0) + } +} + +impl FusedIterator for ByRef<'_, I> {} +unsafe impl TrustedLen for ByRef<'_, I> {} + +/// This struct implements `Iterator` the "naive" way without specialization, +/// which is used to access the default implementations of the iterator methods. +struct ByRefDefault<'a, I: ?Sized>(&'a mut I); + +// Make sure to add all iterator methods that can be directly forwarded here. +impl Iterator for ByRefDefault<'_, I> { + type Item = I::Item; + + #[inline] + fn next(&mut self) -> Option { + I::next(self.0) + } + + fn size_hint(&self) -> (usize, Option) { + I::size_hint(self.0) + } + + fn advance_by(&mut self, n: usize) -> Result<(), NonZero> { + I::advance_by(self.0, n) + } + + fn nth(&mut self, n: usize) -> Option { + I::nth(self.0, n) + } +} + +impl DoubleEndedIterator for ByRefDefault<'_, I> { + fn next_back(&mut self) -> Option { + I::next_back(self.0) + } + + fn advance_back_by(&mut self, n: usize) -> Result<(), NonZero> { + I::advance_back_by(self.0, n) + } + + fn nth_back(&mut self, n: usize) -> Option { + I::nth_back(self.0, n) + } +} + +impl ExactSizeIterator for ByRefDefault<'_, I> { + fn len(&self) -> usize { + I::len(self.0) + } + + fn is_empty(&self) -> bool { + I::is_empty(self.0) + } +} + +impl FusedIterator for ByRefDefault<'_, I> {} +unsafe impl TrustedLen for ByRefDefault<'_, I> {} diff --git a/library/core/src/iter/adapters/by_ref_sized.rs b/library/core/src/iter/adapters/by_ref_sized.rs deleted file mode 100644 index d084bede1eba6..0000000000000 --- a/library/core/src/iter/adapters/by_ref_sized.rs +++ /dev/null @@ -1,92 +0,0 @@ -use crate::num::NonZero; -use crate::ops::{NeverShortCircuit, Try}; - -/// Like `Iterator::by_ref`, but requiring `Sized` so it can forward generics. -/// -/// Ideally this will no longer be required, eventually, but as can be seen in -/// the benchmarks (as of Feb 2022 at least) `by_ref` can have performance cost. -#[unstable(feature = "std_internals", issue = "none")] -#[derive(Debug)] -pub struct ByRefSized<'a, I>(pub &'a mut I); - -// The following implementations use UFCS-style, rather than trusting autoderef, -// to avoid accidentally calling the `&mut Iterator` implementations. - -#[unstable(feature = "std_internals", issue = "none")] -impl Iterator for ByRefSized<'_, I> { - type Item = I::Item; - - #[inline] - fn next(&mut self) -> Option { - I::next(self.0) - } - - #[inline] - fn size_hint(&self) -> (usize, Option) { - I::size_hint(self.0) - } - - #[inline] - fn advance_by(&mut self, n: usize) -> Result<(), NonZero> { - I::advance_by(self.0, n) - } - - #[inline] - fn nth(&mut self, n: usize) -> Option { - I::nth(self.0, n) - } - - #[inline] - fn fold(self, init: B, f: F) -> B - where - F: FnMut(B, Self::Item) -> B, - { - // `fold` needs ownership, so this can't forward directly. - I::try_fold(self.0, init, NeverShortCircuit::wrap_mut_2(f)).0 - } - - #[inline] - fn try_fold(&mut self, init: B, f: F) -> R - where - F: FnMut(B, Self::Item) -> R, - R: Try, - { - I::try_fold(self.0, init, f) - } -} - -#[unstable(feature = "std_internals", issue = "none")] -impl DoubleEndedIterator for ByRefSized<'_, I> { - #[inline] - fn next_back(&mut self) -> Option { - I::next_back(self.0) - } - - #[inline] - fn advance_back_by(&mut self, n: usize) -> Result<(), NonZero> { - I::advance_back_by(self.0, n) - } - - #[inline] - fn nth_back(&mut self, n: usize) -> Option { - I::nth_back(self.0, n) - } - - #[inline] - fn rfold(self, init: B, f: F) -> B - where - F: FnMut(B, Self::Item) -> B, - { - // `rfold` needs ownership, so this can't forward directly. - I::try_rfold(self.0, init, NeverShortCircuit::wrap_mut_2(f)).0 - } - - #[inline] - fn try_rfold(&mut self, init: B, f: F) -> R - where - F: FnMut(B, Self::Item) -> R, - R: Try, - { - I::try_rfold(self.0, init, f) - } -} diff --git a/library/core/src/iter/adapters/mod.rs b/library/core/src/iter/adapters/mod.rs index 2a0ef0189d165..5c58620e7dbab 100644 --- a/library/core/src/iter/adapters/mod.rs +++ b/library/core/src/iter/adapters/mod.rs @@ -3,7 +3,7 @@ use crate::num::NonZero; use crate::ops::{ChangeOutputType, ControlFlow, FromResidual, Residual, Try}; mod array_chunks; -mod by_ref_sized; +mod by_ref; mod chain; mod cloned; mod copied; @@ -30,8 +30,6 @@ mod zip; #[unstable(feature = "iter_array_chunks", reason = "recently added", issue = "100450")] pub use self::array_chunks::ArrayChunks; -#[unstable(feature = "std_internals", issue = "none")] -pub use self::by_ref_sized::ByRefSized; #[unstable(feature = "iter_chain", reason = "recently added", issue = "125964")] pub use self::chain::chain; #[stable(feature = "iter_cloned", since = "1.1.0")] diff --git a/library/core/src/iter/mod.rs b/library/core/src/iter/mod.rs index d62a445d704ae..a12927d397f8a 100644 --- a/library/core/src/iter/mod.rs +++ b/library/core/src/iter/mod.rs @@ -382,8 +382,6 @@ macro_rules! impl_fold_via_try_fold { #[unstable(feature = "iter_array_chunks", reason = "recently added", issue = "100450")] pub use self::adapters::ArrayChunks; -#[unstable(feature = "std_internals", issue = "none")] -pub use self::adapters::ByRefSized; #[stable(feature = "iter_cloned", since = "1.1.0")] pub use self::adapters::Cloned; #[stable(feature = "iter_copied", since = "1.36.0")] diff --git a/library/core/src/iter/traits/double_ended.rs b/library/core/src/iter/traits/double_ended.rs index 7dabaece95561..9dca92b20b9ec 100644 --- a/library/core/src/iter/traits/double_ended.rs +++ b/library/core/src/iter/traits/double_ended.rs @@ -368,78 +368,3 @@ pub trait DoubleEndedIterator: Iterator { self.try_rfold((), check(predicate)).break_value() } } - -#[stable(feature = "rust1", since = "1.0.0")] -impl<'a, I: DoubleEndedIterator + ?Sized> DoubleEndedIterator for &'a mut I { - fn next_back(&mut self) -> Option { - (**self).next_back() - } - fn advance_back_by(&mut self, n: usize) -> Result<(), NonZero> { - (**self).advance_back_by(n) - } - fn nth_back(&mut self, n: usize) -> Option { - (**self).nth_back(n) - } - fn rfold(self, init: B, f: F) -> B - where - F: FnMut(B, Self::Item) -> B, - { - self.spec_rfold(init, f) - } - fn try_rfold(&mut self, init: B, f: F) -> R - where - F: FnMut(B, Self::Item) -> R, - R: Try, - { - self.spec_try_rfold(init, f) - } -} - -/// Helper trait to specialize `rfold` and `rtry_fold` for `&mut I where I: Sized` -trait DoubleEndedIteratorRefSpec: DoubleEndedIterator { - fn spec_rfold(self, init: B, f: F) -> B - where - F: FnMut(B, Self::Item) -> B; - - fn spec_try_rfold(&mut self, init: B, f: F) -> R - where - F: FnMut(B, Self::Item) -> R, - R: Try; -} - -impl DoubleEndedIteratorRefSpec for &mut I { - default fn spec_rfold(self, init: B, mut f: F) -> B - where - F: FnMut(B, Self::Item) -> B, - { - let mut accum = init; - while let Some(x) = self.next_back() { - accum = f(accum, x); - } - accum - } - - default fn spec_try_rfold(&mut self, init: B, mut f: F) -> R - where - F: FnMut(B, Self::Item) -> R, - R: Try, - { - let mut accum = init; - while let Some(x) = self.next_back() { - accum = f(accum, x)?; - } - try { accum } - } -} - -impl DoubleEndedIteratorRefSpec for &mut I { - impl_fold_via_try_fold! { spec_rfold -> spec_try_rfold } - - fn spec_try_rfold(&mut self, init: B, f: F) -> R - where - F: FnMut(B, Self::Item) -> R, - R: Try, - { - (**self).try_rfold(init, f) - } -} diff --git a/library/core/src/iter/traits/exact_size.rs b/library/core/src/iter/traits/exact_size.rs index 908830d8a9514..6e5df8d2379c4 100644 --- a/library/core/src/iter/traits/exact_size.rs +++ b/library/core/src/iter/traits/exact_size.rs @@ -149,13 +149,3 @@ pub trait ExactSizeIterator: Iterator { self.len() == 0 } } - -#[stable(feature = "rust1", since = "1.0.0")] -impl ExactSizeIterator for &mut I { - fn len(&self) -> usize { - (**self).len() - } - fn is_empty(&self) -> bool { - (**self).is_empty() - } -} diff --git a/library/core/src/iter/traits/iterator.rs b/library/core/src/iter/traits/iterator.rs index 3bbb52fdbcb5f..6d6b9f498feb2 100644 --- a/library/core/src/iter/traits/iterator.rs +++ b/library/core/src/iter/traits/iterator.rs @@ -1,8 +1,8 @@ use super::super::{ - ArrayChunks, ByRefSized, Chain, Cloned, Copied, Cycle, Enumerate, Filter, FilterMap, FlatMap, - Flatten, Fuse, Inspect, Intersperse, IntersperseWith, Map, MapWhile, MapWindows, Peekable, - Product, Rev, Scan, Skip, SkipWhile, StepBy, Sum, Take, TakeWhile, TrustedRandomAccessNoCoerce, - Zip, try_process, + ArrayChunks, Chain, Cloned, Copied, Cycle, Enumerate, Filter, FilterMap, FlatMap, Flatten, + Fuse, Inspect, Intersperse, IntersperseWith, Map, MapWhile, MapWindows, Peekable, Product, Rev, + Scan, Skip, SkipWhile, StepBy, Sum, Take, TakeWhile, TrustedRandomAccessNoCoerce, Zip, + try_process, }; use crate::array; use crate::cmp::{self, Ordering}; @@ -2062,7 +2062,7 @@ pub trait Iterator { Self::Item: Try>, B: FromIterator<::Output>, { - try_process(ByRefSized(self), |i| i.collect()) + try_process(self, |i| i.collect()) } /// Collects all the items from an iterator into a collection. @@ -4037,86 +4037,3 @@ where ControlFlow::Break(x) => x, } } - -/// Implements `Iterator` for mutable references to iterators, such as those produced by [`Iterator::by_ref`]. -/// -/// This implementation passes all method calls on to the original iterator. -#[stable(feature = "rust1", since = "1.0.0")] -impl Iterator for &mut I { - type Item = I::Item; - #[inline] - fn next(&mut self) -> Option { - (**self).next() - } - fn size_hint(&self) -> (usize, Option) { - (**self).size_hint() - } - fn advance_by(&mut self, n: usize) -> Result<(), NonZero> { - (**self).advance_by(n) - } - fn nth(&mut self, n: usize) -> Option { - (**self).nth(n) - } - fn fold(self, init: B, f: F) -> B - where - F: FnMut(B, Self::Item) -> B, - { - self.spec_fold(init, f) - } - fn try_fold(&mut self, init: B, f: F) -> R - where - F: FnMut(B, Self::Item) -> R, - R: Try, - { - self.spec_try_fold(init, f) - } -} - -/// Helper trait to specialize `fold` and `try_fold` for `&mut I where I: Sized` -trait IteratorRefSpec: Iterator { - fn spec_fold(self, init: B, f: F) -> B - where - F: FnMut(B, Self::Item) -> B; - - fn spec_try_fold(&mut self, init: B, f: F) -> R - where - F: FnMut(B, Self::Item) -> R, - R: Try; -} - -impl IteratorRefSpec for &mut I { - default fn spec_fold(self, init: B, mut f: F) -> B - where - F: FnMut(B, Self::Item) -> B, - { - let mut accum = init; - while let Some(x) = self.next() { - accum = f(accum, x); - } - accum - } - - default fn spec_try_fold(&mut self, init: B, mut f: F) -> R - where - F: FnMut(B, Self::Item) -> R, - R: Try, - { - let mut accum = init; - while let Some(x) = self.next() { - accum = f(accum, x)?; - } - try { accum } - } -} - -impl IteratorRefSpec for &mut I { - impl_fold_via_try_fold! { spec_fold -> spec_try_fold } - - fn spec_try_fold(&mut self, init: B, f: F) -> R - where - F: FnMut(B, Self::Item) -> R, - R: Try, - { - (**self).try_fold(init, f) - } -} diff --git a/library/core/src/iter/traits/marker.rs b/library/core/src/iter/traits/marker.rs index 2e756a6dd67c4..7042b48a91486 100644 --- a/library/core/src/iter/traits/marker.rs +++ b/library/core/src/iter/traits/marker.rs @@ -31,9 +31,6 @@ pub unsafe trait TrustedFused {} #[lang = "fused_iterator"] pub trait FusedIterator: Iterator {} -#[stable(feature = "fused", since = "1.26.0")] -impl FusedIterator for &mut I {} - /// An iterator that reports an accurate length using size_hint. /// /// The iterator reports a size hint where it is either exact @@ -65,9 +62,6 @@ impl FusedIterator for &mut I {} #[rustc_unsafe_specialization_marker] pub unsafe trait TrustedLen: Iterator {} -#[unstable(feature = "trusted_len", issue = "37572")] -unsafe impl TrustedLen for &mut I {} - /// An iterator that when yielding an item will have taken at least one element /// from its underlying [`SourceIter`]. /// diff --git a/library/coretests/tests/iter/adapters/by_ref_sized.rs b/library/coretests/tests/iter/adapters/by_ref.rs similarity index 57% rename from library/coretests/tests/iter/adapters/by_ref_sized.rs rename to library/coretests/tests/iter/adapters/by_ref.rs index a9c066f0e8cb1..92558906d512f 100644 --- a/library/coretests/tests/iter/adapters/by_ref_sized.rs +++ b/library/coretests/tests/iter/adapters/by_ref.rs @@ -1,20 +1,18 @@ -use core::iter::*; - #[test] fn test_iterator_by_ref_sized() { let a = ['a', 'b', 'c', 'd']; let mut s = String::from("Z"); let mut it = a.iter().copied(); - ByRefSized(&mut it).take(2).for_each(|x| s.push(x)); + (&mut it).take(2).for_each(|x| s.push(x)); assert_eq!(s, "Zab"); - ByRefSized(&mut it).fold((), |(), x| s.push(x)); + (&mut it).fold((), |(), x| s.push(x)); assert_eq!(s, "Zabcd"); let mut s = String::from("Z"); let mut it = a.iter().copied(); - ByRefSized(&mut it).rev().take(2).for_each(|x| s.push(x)); + (&mut it).rev().take(2).for_each(|x| s.push(x)); assert_eq!(s, "Zdc"); - ByRefSized(&mut it).rfold((), |(), x| s.push(x)); + (&mut it).rfold((), |(), x| s.push(x)); assert_eq!(s, "Zdcba"); } diff --git a/library/coretests/tests/iter/adapters/mod.rs b/library/coretests/tests/iter/adapters/mod.rs index dedb4c0a9dd5f..a89787ab85c97 100644 --- a/library/coretests/tests/iter/adapters/mod.rs +++ b/library/coretests/tests/iter/adapters/mod.rs @@ -1,5 +1,5 @@ mod array_chunks; -mod by_ref_sized; +mod by_ref; mod chain; mod cloned; mod copied;