Deprecated: The each() function is deprecated. This message will be suppressed on further calls in /home/zhenxiangba/zhenxiangba.com/public_html/phproxy-improved-master/index.php on line 456
multiversion 0.5.1 - Docs.rs
[go: Go Back, main page]

multiversion 0.5.1

Easy function multiversioning
Documentation
use multiversion::multiversion;

#[multiversion]
#[clone(target = "[x86|x86_64]+avx")]
#[clone(target = "[x86|x86_64]+sse")]
#[clone(target = "[arm|aarch64]+neon")]
pub fn pub_add(a: &mut [f32], b: &[f32]) {
    a.iter_mut().zip(b.iter()).for_each(|(a, b)| *a += b);
}

#[multiversion]
#[clone(target = "[x86|x86_64]+avx")]
#[clone(target = "[x86|x86_64]+sse")]
#[clone(target = "[arm|aarch64]+neon")]
fn priv_add(a: &mut [f32], b: &[f32]) {
    a.iter_mut().zip(b.iter()).for_each(|(a, b)| *a += b);
}

#[multiversion]
#[clone(target = "[x86|x86_64]+avx")]
#[clone(target = "[x86|x86_64]+sse")]
#[clone(target = "[arm|aarch64]+neon")]
#[inline]
pub unsafe fn pub_unsafe_add(a: &mut [f32], b: &[f32]) {
    a.iter_mut().zip(b.iter()).for_each(|(a, b)| *a += b);
}

#[multiversion]
#[clone(target = "[x86|x86_64]+avx")]
#[clone(target = "[x86|x86_64]+sse")]
#[clone(target = "[arm|aarch64]+neon")]
#[inline]
unsafe fn priv_unsafe_add(a: &mut [f32], b: &[f32]) {
    a.iter_mut().zip(b.iter()).for_each(|(a, b)| *a += b);
}

struct Adder(f32);

impl Adder {
    #[multiversion]
    #[clone(target = "[x86|x86_64]+avx")]
    #[clone(target = "[x86|x86_64]+sse")]
    #[clone(target = "[arm|aarch64]+neon")]
    #[inline]
    pub fn pub_add(&self, x: &mut [f32]) {
        x.iter_mut().for_each(|x| *x += self.0);
    }

    #[multiversion]
    #[clone(target = "[x86|x86_64]+avx")]
    #[clone(target = "[x86|x86_64]+sse")]
    #[clone(target = "[arm|aarch64]+neon")]
    #[inline]
    fn priv_add(&self, x: &mut [f32]) {
        x.iter_mut().for_each(|x| *x += self.0);
    }

    #[multiversion]
    #[clone(target = "[x86|x86_64]+avx")]
    #[clone(target = "[x86|x86_64]+sse")]
    #[clone(target = "[arm|aarch64]+neon")]
    #[inline]
    pub unsafe fn pub_unsafe_add(&self, x: &mut [f32]) {
        x.iter_mut().for_each(|x| *x += self.0);
    }

    #[multiversion]
    #[clone(target = "[x86|x86_64]+avx")]
    #[clone(target = "[x86|x86_64]+sse")]
    #[clone(target = "[arm|aarch64]+neon")]
    #[inline]
    unsafe fn priv_unsafe_add(&self, x: &mut [f32]) {
        x.iter_mut().for_each(|x| *x += self.0);
    }
}

mod test {
    use super::*;

    #[test]
    fn test_add() {
        let mut a = vec![0f32, 2f32, 4f32];
        let b = vec![1f32, 1f32, 1f32];
        pub_add(&mut a, &b);
        assert_eq!(a, vec![1f32, 3f32, 5f32]);
        priv_add(&mut a, &b);
        assert_eq!(a, vec![2f32, 4f32, 6f32]);
        unsafe {
            pub_unsafe_add(&mut a, &b);
        }
        assert_eq!(a, vec![3f32, 5f32, 7f32]);
        unsafe {
            priv_unsafe_add(&mut a, &b);
        }
        assert_eq!(a, vec![4f32, 6f32, 8f32]);
    }

    #[test]
    fn test_add_associated() {
        let adder = Adder(1.);
        let mut a = vec![0f32, 2f32, 4f32];
        adder.pub_add(&mut a);
        assert_eq!(a, vec![1f32, 3f32, 5f32]);
        adder.priv_add(&mut a);
        assert_eq!(a, vec![2f32, 4f32, 6f32]);
        unsafe {
            adder.pub_unsafe_add(&mut a);
        }
        assert_eq!(a, vec![3f32, 5f32, 7f32]);
        unsafe {
            adder.priv_unsafe_add(&mut a);
        }
        assert_eq!(a, vec![4f32, 6f32, 8f32]);
    }
}