par_iter/
vec.rs

1//! Parallel iterator types for [vectors][std::vec] (`Vec<T>`)
2//!
3//! You will rarely need to interact with this module directly unless you need
4//! to name one of the iterator types.
5//!
6//! [std::vec]: https://doc.rust-lang.org/stable/std/vec/
7
8use std::{
9    iter, mem,
10    ops::{Range, RangeBounds},
11    ptr, slice,
12};
13
14use crate::{
15    iter::{plumbing::*, *},
16    math::simplify_range,
17    slice::{Iter, IterMut},
18};
19
20impl<'data, T: Sync + 'data> IntoParallelIterator for &'data Vec<T> {
21    type Item = &'data T;
22    type Iter = Iter<'data, T>;
23
24    fn into_par_iter(self) -> Self::Iter {
25        <&[T]>::into_par_iter(self)
26    }
27}
28
29impl<'data, T: Send + 'data> IntoParallelIterator for &'data mut Vec<T> {
30    type Item = &'data mut T;
31    type Iter = IterMut<'data, T>;
32
33    fn into_par_iter(self) -> Self::Iter {
34        <&mut [T]>::into_par_iter(self)
35    }
36}
37
38/// Parallel iterator that moves out of a vector.
39#[derive(Debug, Clone)]
40pub struct IntoIter<T: Send> {
41    vec: Vec<T>,
42}
43
44impl<T: Send> IntoParallelIterator for Vec<T> {
45    type Item = T;
46    type Iter = IntoIter<T>;
47
48    fn into_par_iter(self) -> Self::Iter {
49        IntoIter { vec: self }
50    }
51}
52
53impl<T: Send> ParallelIterator for IntoIter<T> {
54    type Item = T;
55
56    fn drive_unindexed<C>(self, consumer: C) -> C::Result
57    where
58        C: UnindexedConsumer<Self::Item>,
59    {
60        bridge(self, consumer)
61    }
62
63    fn opt_len(&self) -> Option<usize> {
64        Some(self.len())
65    }
66}
67
68impl<T: Send> IndexedParallelIterator for IntoIter<T> {
69    fn drive<C>(self, consumer: C) -> C::Result
70    where
71        C: Consumer<Self::Item>,
72    {
73        bridge(self, consumer)
74    }
75
76    fn len(&self) -> usize {
77        self.vec.len()
78    }
79
80    fn with_producer<CB>(mut self, callback: CB) -> CB::Output
81    where
82        CB: ProducerCallback<Self::Item>,
83    {
84        // Drain every item, and then the vector only needs to free its buffer.
85        self.vec.par_drain(..).with_producer(callback)
86    }
87}
88
89impl<'data, T: Send> ParallelDrainRange<usize> for &'data mut Vec<T> {
90    type Item = T;
91    type Iter = Drain<'data, T>;
92
93    fn par_drain<R: RangeBounds<usize>>(self, range: R) -> Self::Iter {
94        Drain {
95            orig_len: self.len(),
96            range: simplify_range(range, self.len()),
97            vec: self,
98        }
99    }
100}
101
102/// Draining parallel iterator that moves a range out of a vector, but keeps the
103/// total capacity.
104#[derive(Debug)]
105pub struct Drain<'data, T: Send> {
106    vec: &'data mut Vec<T>,
107    range: Range<usize>,
108    orig_len: usize,
109}
110
111impl<'data, T: Send> ParallelIterator for Drain<'data, T> {
112    type Item = T;
113
114    fn drive_unindexed<C>(self, consumer: C) -> C::Result
115    where
116        C: UnindexedConsumer<Self::Item>,
117    {
118        bridge(self, consumer)
119    }
120
121    fn opt_len(&self) -> Option<usize> {
122        Some(self.len())
123    }
124}
125
126impl<'data, T: Send> IndexedParallelIterator for Drain<'data, T> {
127    fn drive<C>(self, consumer: C) -> C::Result
128    where
129        C: Consumer<Self::Item>,
130    {
131        bridge(self, consumer)
132    }
133
134    fn len(&self) -> usize {
135        self.range.len()
136    }
137
138    fn with_producer<CB>(self, callback: CB) -> CB::Output
139    where
140        CB: ProducerCallback<Self::Item>,
141    {
142        unsafe {
143            // Make the vector forget about the drained items, and temporarily the tail too.
144            self.vec.set_len(self.range.start);
145
146            // Create the producer as the exclusive "owner" of the slice.
147            let producer = DrainProducer::from_vec(self.vec, self.range.len());
148
149            // The producer will move or drop each item from the drained range.
150            callback.callback(producer)
151        }
152    }
153}
154
155impl<'data, T: Send> Drop for Drain<'data, T> {
156    fn drop(&mut self) {
157        let Range { start, end } = self.range;
158        if self.vec.len() == self.orig_len {
159            // We must not have produced, so just call a normal drain to remove the items.
160            self.vec.drain(start..end);
161        } else if start == end {
162            // Empty range, so just restore the length to its original state
163            unsafe {
164                self.vec.set_len(self.orig_len);
165            }
166        } else if end < self.orig_len {
167            // The producer was responsible for consuming the drained items.
168            // Move the tail items to their new place, then set the length to include them.
169            unsafe {
170                let ptr = self.vec.as_mut_ptr().add(start);
171                let tail_ptr = self.vec.as_ptr().add(end);
172                let tail_len = self.orig_len - end;
173                ptr::copy(tail_ptr, ptr, tail_len);
174                self.vec.set_len(start + tail_len);
175            }
176        }
177    }
178}
179
180/// ////////////////////////////////////////////////////////////////////////
181
182pub(crate) struct DrainProducer<'data, T: Send> {
183    slice: &'data mut [T],
184}
185
186impl<T: Send> DrainProducer<'_, T> {
187    /// Creates a draining producer, which *moves* items from the slice.
188    ///
189    /// Unsafe because `!Copy` data must not be read after the borrow is
190    /// released.
191    pub(crate) unsafe fn new(slice: &mut [T]) -> DrainProducer<'_, T> {
192        DrainProducer { slice }
193    }
194
195    /// Creates a draining producer, which *moves* items from the tail of the
196    /// vector.
197    ///
198    /// Unsafe because we're moving from beyond `vec.len()`, so the caller must
199    /// ensure that data is initialized and not read after the borrow is
200    /// released.
201    unsafe fn from_vec(vec: &mut Vec<T>, len: usize) -> DrainProducer<'_, T> {
202        let start = vec.len();
203        assert!(vec.capacity() - start >= len);
204
205        // The pointer is derived from `Vec` directly, not through a `Deref`,
206        // so it has provenance over the whole allocation.
207        let ptr = vec.as_mut_ptr().add(start);
208        DrainProducer::new(slice::from_raw_parts_mut(ptr, len))
209    }
210}
211
212impl<'data, T: 'data + Send> Producer for DrainProducer<'data, T> {
213    type IntoIter = SliceDrain<'data, T>;
214    type Item = T;
215
216    fn into_iter(mut self) -> Self::IntoIter {
217        // replace the slice so we don't drop it twice
218        let slice = mem::take(&mut self.slice);
219        SliceDrain {
220            iter: slice.iter_mut(),
221        }
222    }
223
224    fn split_at(mut self, index: usize) -> (Self, Self) {
225        // replace the slice so we don't drop it twice
226        let slice = mem::take(&mut self.slice);
227        let (left, right) = slice.split_at_mut(index);
228        unsafe { (DrainProducer::new(left), DrainProducer::new(right)) }
229    }
230}
231
232impl<'data, T: 'data + Send> Drop for DrainProducer<'data, T> {
233    fn drop(&mut self) {
234        // extract the slice so we can use `Drop for [T]`
235        let slice_ptr: *mut [T] = mem::take::<&'data mut [T]>(&mut self.slice);
236        unsafe { ptr::drop_in_place::<[T]>(slice_ptr) };
237    }
238}
239
240/// ////////////////////////////////////////////////////////////////////////
241
242// like std::vec::Drain, without updating a source Vec
243pub(crate) struct SliceDrain<'data, T> {
244    iter: slice::IterMut<'data, T>,
245}
246
247impl<'data, T: 'data> Iterator for SliceDrain<'data, T> {
248    type Item = T;
249
250    fn next(&mut self) -> Option<T> {
251        // Coerce the pointer early, so we don't keep the
252        // reference that's about to be invalidated.
253        let ptr: *const T = self.iter.next()?;
254        Some(unsafe { ptr::read(ptr) })
255    }
256
257    fn size_hint(&self) -> (usize, Option<usize>) {
258        self.iter.size_hint()
259    }
260
261    fn count(self) -> usize {
262        self.iter.len()
263    }
264}
265
266impl<'data, T: 'data> DoubleEndedIterator for SliceDrain<'data, T> {
267    fn next_back(&mut self) -> Option<Self::Item> {
268        // Coerce the pointer early, so we don't keep the
269        // reference that's about to be invalidated.
270        let ptr: *const T = self.iter.next_back()?;
271        Some(unsafe { ptr::read(ptr) })
272    }
273}
274
275impl<'data, T: 'data> ExactSizeIterator for SliceDrain<'data, T> {
276    fn len(&self) -> usize {
277        self.iter.len()
278    }
279}
280
281impl<'data, T: 'data> iter::FusedIterator for SliceDrain<'data, T> {}
282
283impl<'data, T: 'data> Drop for SliceDrain<'data, T> {
284    fn drop(&mut self) {
285        // extract the iterator so we can use `Drop for [T]`
286        let slice_ptr: *mut [T] = mem::replace(&mut self.iter, [].iter_mut()).into_slice();
287        unsafe { ptr::drop_in_place::<[T]>(slice_ptr) };
288    }
289}