macerator 0.3.2

Type and target-generic SIMD
Documentation
use half::f16;
use paste::paste;

use crate::{Scalar, Simd, Vector};

pub trait ReduceAdd: Scalar {
    fn reduce_add<S: Simd>(input: Vector<S, Self>) -> Self;
    fn is_accelerated<S: Simd>() -> bool;
}

impl<S: Simd, T: ReduceAdd> Vector<S, T> {
    #[inline(always)]
    pub fn reduce_add(self) -> T {
        T::reduce_add(self)
    }
}

pub trait ReduceMin: Scalar {
    fn reduce_min<S: Simd>(input: Vector<S, Self>) -> Self;
    fn is_accelerated<S: Simd>() -> bool;
}

impl<S: Simd, T: ReduceMin> Vector<S, T> {
    #[inline(always)]
    pub fn reduce_min(self) -> T {
        T::reduce_min(self)
    }
}

pub trait ReduceMax: Scalar {
    fn reduce_max<S: Simd>(input: Vector<S, Self>) -> Self;
    fn is_accelerated<S: Simd>() -> bool;
}

impl<S: Simd, T: ReduceMax> Vector<S, T> {
    #[inline(always)]
    pub fn reduce_max(self) -> T {
        T::reduce_max(self)
    }
}

macro_rules! impl_reduce {
    ($trait: ident, $name: ident, $($ty: ty),*) => {
        $(paste! {
            impl $trait for $ty {
                #[inline(always)]
                fn [<$trait:snake>]<S: Simd>(input: Vector<S, Self>) -> Self {
                    S::[<$name _ $ty>](*input)
                }
                #[inline(always)]
                fn is_accelerated<S: Simd>() -> bool {
                    S::[<$name _ $ty _supported>]()
                }
            }
        })*
    };
}

impl_reduce!(ReduceAdd, reduce_add, u8, i8, u16, i16, u32, i32, u64, i64, f16, f32, f64);
impl_reduce!(ReduceMin, reduce_min, u8, i8, u16, i16, u32, i32, u64, i64, f16, f32, f64);
impl_reduce!(ReduceMax, reduce_max, u8, i8, u16, i16, u32, i32, u64, i64, f16, f32, f64);