-
Notifications
You must be signed in to change notification settings - Fork 31
/
std_simd.rs
66 lines (51 loc) · 1.86 KB
/
std_simd.rs
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
#![cfg(feature = "portable-simd")]
use core::simd::prelude::*;
use std::mem::size_of;
use crate::premul_alpha::unpremultiply_f32_scalar;
/// Divides input by alpha, on encountering zero in alpha, stores zero in output
///
///
pub fn unpremultiply_std_simd(input: &mut [f32], alpha: &[f32]) {
// Compiler explorer
// url: https://rust.godbolt.org/z/EK88W4j98
// 8 seems to work out nicely
const VECTOR_SIZE: usize = size_of::<Simd<f32, 8>>() / size_of::<f32>();
let in_chunk = input.chunks_exact_mut(VECTOR_SIZE);
let alpha_chunk = alpha.chunks_exact(VECTOR_SIZE);
let zeros = f32x8::splat(0.0);
for (chunk, alpha_values) in in_chunk.zip(alpha_chunk) {
let ab = f32x8::from_slice(chunk);
let al = f32x8::from_slice(alpha_values);
let mask = al.simd_eq(zeros);
// divide
let div_result = ab / al;
// remove effects of division bu zero
// analogous to _mm_andnot_ps or (!mask & b)
let result = (!mask).select(div_result, zeros);
// store
result.copy_to_slice(chunk);
}
// handle remainder
unpremultiply_f32_scalar(
input.chunks_exact_mut(VECTOR_SIZE).into_remainder(),
alpha.chunks_exact(VECTOR_SIZE).remainder()
);
}
#[test]
fn test_unpremultiply_simd_scalar() {
use nanorand::Rng;
use crate::premul_alpha::unpremultiply_f32_scalar;
let mut in_array = [0.0f32; 256];
nanorand::WyRand::new().fill(&mut in_array);
let mut in_copy = in_array;
let mut in_alpha = [0.0; 256];
nanorand::WyRand::new().fill(&mut in_alpha);
unpremultiply_f32_scalar(&mut in_array, &in_alpha);
unpremultiply_std_simd(&mut in_copy, &in_alpha);
for (a, b) in in_array.iter().zip(&in_copy) {
let diff = a - b;
assert!(a.is_finite());
assert!(b.is_finite());
assert!(diff < f32::EPSILON, "{a} {b}");
}
}