1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
use super::{ParallelIterator, ExactParallelIterator, IntoParallelIterator};
use super::from_par_iter::FromParallelIterator;
use std::collections::LinkedList;
use std::slice;
use std::sync::atomic::{AtomicUsize, Ordering};
mod consumer;
use self::consumer::CollectConsumer;
pub fn collect_into<PAR_ITER, T>(mut pi: PAR_ITER, v: &mut Vec<T>)
where PAR_ITER: ExactParallelIterator<Item = T>,
T: Send
{
let mut collect = Collect::new(v, pi.len());
pi.drive(collect.as_consumer());
collect.complete();
}
fn special_collect_into<PAR_ITER, T>(pi: PAR_ITER, len: usize, v: &mut Vec<T>)
where PAR_ITER: ParallelIterator<Item = T>,
T: Send
{
let mut collect = Collect::new(v, len);
pi.drive_unindexed(collect.as_consumer());
collect.complete();
}
struct Collect<'c, T: Send + 'c> {
writes: AtomicUsize,
vec: &'c mut Vec<T>,
len: usize,
}
impl<'c, T: Send + 'c> Collect<'c, T> {
fn new(vec: &'c mut Vec<T>, len: usize) -> Self {
vec.truncate(0);
vec.reserve(len);
Collect {
writes: AtomicUsize::new(0),
vec: vec,
len: len,
}
}
fn as_consumer(&mut self) -> CollectConsumer<T> {
let mut slice = self.vec.as_mut_slice();
slice = unsafe { slice::from_raw_parts_mut(slice.as_mut_ptr(), self.len) };
CollectConsumer::new(&self.writes, slice)
}
fn complete(mut self) {
unsafe {
let actual_writes = self.writes.load(Ordering::Relaxed);
assert!(actual_writes == self.len,
"expected {} total writes, but got {}",
self.len,
actual_writes);
self.vec.set_len(self.len);
}
}
}
impl<T> FromParallelIterator<T> for Vec<T>
where T: Send
{
fn from_par_iter<PAR_ITER>(par_iter: PAR_ITER) -> Self
where PAR_ITER: IntoParallelIterator<Item = T>
{
let mut par_iter = par_iter.into_par_iter();
match par_iter.opt_len() {
Some(len) => {
let mut vec = vec![];
super::collect::special_collect_into(par_iter, len, &mut vec);
vec
}
None => {
let list: LinkedList<_> = par_iter.fold(Vec::new, |mut vec, elem| {
vec.push(elem);
vec
})
.collect();
let len = list.iter().map(Vec::len).sum();
let start = Vec::with_capacity(len);
list.into_iter()
.fold(start, |mut vec, mut sub| {
vec.append(&mut sub);
vec
})
}
}
}
}