Skip to content

Commit a838552

Browse files
Merge portable-simd#195 - portable-simd:trait-ops
Generic `core::ops` for `Simd<T, _>` In order to maintain type soundness, we need to be sure we only implement an operation for `Simd<T, _> where T: SimdElement`... and also valid for that operation in general. While we could do this purely parametrically, it is more sound to implement the operators directly for the base scalar type arguments and then use type parameters to extend the operators to the "higher order" operations. This implements that strategy and cleans up `simd::ops` into a few submodules: - assign.rs: `core::ops::*Assign` - deref.rs: `core::ops` impls which "deref" borrowed versions of the arguments - unary.rs: encloses the logic for unary operators on `Simd`, as unary ops are much simpler This is possible since everything need not be nested in a single maze of macros anymore. The result simplifies the logic and allows reasoning about what operators are valid based on the expressed trait bounds, and also reduces the size of the trait implementation output in rustdoc, for a huge win of 4 MB off the size of `struct.Simd.html`! This addresses a common user complaint, as the original was over 5.5 MB and capable of crashing browsers! This also carries a fix for a type-inference-related breakage, by removing the autosplatting (vector + scalar binop) impls, as unfortunately the presence of autosplatting was capable of busting type inference. We will likely need to see results from a Crater run before we can understand how to re-land autosplatting.
2 parents b2dac71 + 8003b04 commit a838552

File tree

9 files changed

+377
-459
lines changed

9 files changed

+377
-459
lines changed

crates/core_simd/examples/nbody.rs

+5-5
Original file line numberDiff line numberDiff line change
@@ -97,7 +97,7 @@ mod nbody {
9797
let sun = &mut sun[0];
9898
for body in rest {
9999
let m_ratio = body.mass / SOLAR_MASS;
100-
sun.v -= body.v * m_ratio;
100+
sun.v -= body.v * Simd::splat(m_ratio);
101101
}
102102
}
103103

@@ -143,14 +143,14 @@ mod nbody {
143143
let mut i = 0;
144144
for j in 0..N_BODIES {
145145
for k in j + 1..N_BODIES {
146-
let f = r[i] * mag[i];
147-
bodies[j].v -= f * bodies[k].mass;
148-
bodies[k].v += f * bodies[j].mass;
146+
let f = r[i] * Simd::splat(mag[i]);
147+
bodies[j].v -= f * Simd::splat(bodies[k].mass);
148+
bodies[k].v += f * Simd::splat(bodies[j].mass);
149149
i += 1
150150
}
151151
}
152152
for body in bodies {
153-
body.x += dt * body.v
153+
body.x += Simd::splat(dt) * body.v
154154
}
155155
}
156156

crates/core_simd/src/math.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@ macro_rules! impl_uint_arith {
1717
/// let max = Simd::splat(MAX);
1818
/// let unsat = x + max;
1919
/// let sat = x.saturating_add(max);
20-
/// assert_eq!(x - 1, unsat);
20+
/// assert_eq!(unsat, Simd::from_array([1, 0, MAX, MAX - 1]));
2121
/// assert_eq!(sat, max);
2222
/// ```
2323
#[inline]
@@ -37,7 +37,7 @@ macro_rules! impl_uint_arith {
3737
/// let max = Simd::splat(MAX);
3838
/// let unsat = x - max;
3939
/// let sat = x.saturating_sub(max);
40-
/// assert_eq!(unsat, x + 1);
40+
/// assert_eq!(unsat, Simd::from_array([3, 2, 1, 0]));
4141
/// assert_eq!(sat, Simd::splat(0));
4242
#[inline]
4343
pub fn saturating_sub(self, second: Self) -> Self {
@@ -105,7 +105,7 @@ macro_rules! impl_int_arith {
105105
#[inline]
106106
pub fn abs(self) -> Self {
107107
const SHR: $ty = <$ty>::BITS as $ty - 1;
108-
let m = self >> SHR;
108+
let m = self >> Simd::splat(SHR);
109109
(self^m) - m
110110
}
111111

@@ -128,7 +128,7 @@ macro_rules! impl_int_arith {
128128
pub fn saturating_abs(self) -> Self {
129129
// arith shift for -1 or 0 mask based on sign bit, giving 2s complement
130130
const SHR: $ty = <$ty>::BITS as $ty - 1;
131-
let m = self >> SHR;
131+
let m = self >> Simd::splat(SHR);
132132
(self^m).saturating_sub(m)
133133
}
134134

0 commit comments

Comments
 (0)