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
use super::*;
use generic_array::{ArrayBuilder, ArrayConsumer};
pub unsafe trait Select<T, N: ArrayLength<T>> {
fn select(self, true_values: NumericArray<T, N>, false_values: NumericArray<T, N>) -> NumericArray<T, N>;
}
pub unsafe trait Permute<T, N: ArrayLength<T>> {
fn permute(self, values: &NumericArray<T, N>) -> NumericArray<T, N>;
}
unsafe impl<T, N: ArrayLength<T>> Permute<T, N> for NumericArray<usize, N>
where
N: ArrayLength<usize>,
T: Clone,
{
fn permute(self, values: &NumericArray<T, N>) -> NumericArray<T, N> {
unsafe {
let mut indices = ArrayConsumer::new(self.0);
let (index_iter, index_position) = indices.iter_position();
let mut destination = ArrayBuilder::new();
{
let (destination_iter, destination_position) = destination.iter_position();
destination_iter.zip(index_iter).for_each(|(dst, index)| {
let value = values.get_unchecked(*index).clone();
*index_position += 1;
ptr::write(dst, value);
*destination_position += 1;
});
}
NumericArray::new(destination.into_inner())
}
}
}
unsafe impl<T, N: ArrayLength<T>> Select<T, N> for NumericArray<bool, N>
where
N: ArrayLength<bool>,
{
fn select(self, true_values: NumericArray<T, N>, false_values: NumericArray<T, N>) -> NumericArray<T, N> {
unsafe {
let mut mask = ArrayConsumer::new(self.0);
let mut true_values = ArrayConsumer::new(true_values.0);
let mut false_values = ArrayConsumer::new(false_values.0);
let (mask_iter, mask_position) = mask.iter_position();
let (true_values_iter, true_values_position) = true_values.iter_position();
let (false_values_iter, false_values_position) = false_values.iter_position();
let mut destination = ArrayBuilder::new();
{
let (destination_iter, destination_position) = destination.iter_position();
destination_iter
.zip(mask_iter.zip(true_values_iter.zip(false_values_iter)))
.for_each(|(dst, (m, (t, f)))| {
let t = ptr::read(t);
let f = ptr::read(f);
let m = ptr::read(m);
*mask_position += 1;
*true_values_position += 1;
*false_values_position += 1;
ptr::write(dst, if m { t } else { f });
*destination_position += 1;
});
}
NumericArray::new(destination.into_inner())
}
}
}