1use core::ptr::{self};
2use core::slice::{self};
34use crate::stable::alloc::{Allocator, Global};
56use super::{Drain, Vec};
78/// A splicing iterator for `Vec`.
9///
10/// This struct is created by [`Vec::splice()`].
11/// See its documentation for more.
12///
13/// # Example
14///
15/// ```
16/// let mut v = vec![0, 1, 2];
17/// let new = [7, 8];
18/// let iter: std::vec::Splice<_> = v.splice(1.., new);
19/// ```
20#[derive(Debug)]
21pub struct Splice<'a, I: Iterator + 'a, A: Allocator + 'a = Global> {
22pub(super) drain: Drain<'a, I::Item, A>,
23pub(super) replace_with: I,
24}
2526impl<I: Iterator, A: Allocator> Iterator for Splice<'_, I, A> {
27type Item = I::Item;
2829#[inline(always)]
30fn next(&mut self) -> Option<Self::Item> {
31self.drain.next()
32 }
3334#[inline(always)]
35fn size_hint(&self) -> (usize, Option<usize>) {
36self.drain.size_hint()
37 }
38}
3940impl<I: Iterator, A: Allocator> DoubleEndedIterator for Splice<'_, I, A> {
41#[inline(always)]
42fn next_back(&mut self) -> Option<Self::Item> {
43self.drain.next_back()
44 }
45}
4647impl<I: Iterator, A: Allocator> ExactSizeIterator for Splice<'_, I, A> {}
4849impl<I: Iterator, A: Allocator> Drop for Splice<'_, I, A> {
50#[inline]
51fn drop(&mut self) {
52self.drain.by_ref().for_each(drop);
5354unsafe {
55if self.drain.tail_len == 0 {
56self.drain.vec.as_mut().extend(self.replace_with.by_ref());
57return;
58 }
5960// First fill the range left by drain().
61if !self.drain.fill(&mut self.replace_with) {
62return;
63 }
6465// There may be more elements. Use the lower bound as an estimate.
66 // FIXME: Is the upper bound a better guess? Or something else?
67let (lower_bound, _upper_bound) = self.replace_with.size_hint();
68if lower_bound > 0 {
69self.drain.move_tail(lower_bound);
70if !self.drain.fill(&mut self.replace_with) {
71return;
72 }
73 }
7475// Collect any remaining elements.
76 // This is a zero-length vector which does not allocate if `lower_bound` was exact.
77let mut collected = self
78.replace_with
79 .by_ref()
80 .collect::<Vec<I::Item>>()
81 .into_iter();
82// Now we have an exact count.
83if collected.len() > 0 {
84self.drain.move_tail(collected.len());
85let filled = self.drain.fill(&mut collected);
86debug_assert!(filled);
87debug_assert_eq!(collected.len(), 0);
88 }
89 }
90// Let `Drain::drop` move the tail back if necessary and restore `vec.len`.
91}
92}
9394/// Private helper methods for `Splice::drop`
95impl<T, A: Allocator> Drain<'_, T, A> {
96/// The range from `self.vec.len` to `self.tail_start` contains elements
97 /// that have been moved out.
98 /// Fill that range as much as possible with new elements from the `replace_with` iterator.
99 /// Returns `true` if we filled the entire range. (`replace_with.next()` didn’t return `None`.)
100#[inline(always)]
101unsafe fn fill<I: Iterator<Item = T>>(&mut self, replace_with: &mut I) -> bool {
102let vec = unsafe { self.vec.as_mut() };
103let range_start = vec.len;
104let range_end = self.tail_start;
105let range_slice = unsafe {
106 slice::from_raw_parts_mut(vec.as_mut_ptr().add(range_start), range_end - range_start)
107 };
108109for place in range_slice {
110if let Some(new_item) = replace_with.next() {
111unsafe { ptr::write(place, new_item) };
112 vec.len += 1;
113 } else {
114return false;
115 }
116 }
117true
118}
119120/// Makes room for inserting more elements before the tail.
121#[inline(always)]
122unsafe fn move_tail(&mut self, additional: usize) {
123let vec = unsafe { self.vec.as_mut() };
124let len = self.tail_start + self.tail_len;
125 vec.buf.reserve(len, additional);
126127let new_tail_start = self.tail_start + additional;
128unsafe {
129let src = vec.as_ptr().add(self.tail_start);
130let dst = vec.as_mut_ptr().add(new_tail_start);
131 ptr::copy(src, dst, self.tail_len);
132 }
133self.tail_start = new_tail_start;
134 }
135}