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
use crate::{
array::{Array, PrimitiveArray},
bitmap::{Bitmap, MutableBitmap},
buffer::Buffer,
types::NativeType,
};
use super::Index;
fn take_no_validity<T: NativeType, I: Index>(
values: &[T],
indices: &[I],
) -> (Buffer<T>, Option<Bitmap>) {
let values = indices
.iter()
.map(|index| values[index.to_usize()])
.collect::<Vec<_>>();
(values.into(), None)
}
fn take_values_validity<T: NativeType, I: Index>(
values: &PrimitiveArray<T>,
indices: &[I],
) -> (Buffer<T>, Option<Bitmap>) {
let values_validity = values.validity().unwrap();
let validity = indices
.iter()
.map(|index| values_validity.get_bit(index.to_usize()));
let validity = MutableBitmap::from_trusted_len_iter(validity);
let values_values = values.values();
let values = indices
.iter()
.map(|index| values_values[index.to_usize()])
.collect::<Vec<_>>();
(values.into(), validity.into())
}
fn take_indices_validity<T: NativeType, I: Index>(
values: &[T],
indices: &PrimitiveArray<I>,
) -> (Buffer<T>, Option<Bitmap>) {
let validity = indices.validity().unwrap();
let values = indices
.values()
.iter()
.enumerate()
.map(|(i, index)| {
let index = index.to_usize();
match values.get(index) {
Some(value) => *value,
None => {
if !validity.get_bit(i) {
T::default()
} else {
panic!("Out-of-bounds index {}", index)
}
}
}
})
.collect::<Vec<_>>();
(values.into(), indices.validity().cloned())
}
fn take_values_indices_validity<T: NativeType, I: Index>(
values: &PrimitiveArray<T>,
indices: &PrimitiveArray<I>,
) -> (Buffer<T>, Option<Bitmap>) {
let mut bitmap = MutableBitmap::with_capacity(indices.len());
let values_validity = values.validity().unwrap();
let values_values = values.values();
let values = indices
.iter()
.map(|index| match index {
Some(index) => {
let index = index.to_usize();
bitmap.push(values_validity.get_bit(index));
values_values[index]
}
None => {
bitmap.push(false);
T::default()
}
})
.collect::<Vec<_>>();
(values.into(), bitmap.into())
}
pub fn take<T: NativeType, I: Index>(
values: &PrimitiveArray<T>,
indices: &PrimitiveArray<I>,
) -> PrimitiveArray<T> {
let indices_has_validity = indices.null_count() > 0;
let values_has_validity = values.null_count() > 0;
let (buffer, validity) = match (values_has_validity, indices_has_validity) {
(false, false) => take_no_validity::<T, I>(values.values(), indices.values()),
(true, false) => take_values_validity::<T, I>(values, indices.values()),
(false, true) => take_indices_validity::<T, I>(values.values(), indices),
(true, true) => take_values_indices_validity::<T, I>(values, indices),
};
PrimitiveArray::<T>::new(values.data_type().clone(), buffer, validity)
}