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
116
117
118
119
120
121
122
123
124
125
126
/// Primitive cast between types using `as`
pub trait To<T> {
    fn to(self) -> T;
}

/// blanket implementation to ensure reflexive `To` is the identity function
impl<T> To<T> for T {
    #[inline(always)]
    fn to(self) -> Self {
        self
    }
}

macro_rules! impl_to {
    ($ty1:ty, $($ty:ty,)*) => {
$(
    impl To<$ty> for $ty1 {
        #[inline(always)]
        fn to(self) -> $ty {
            self as $ty
        }
    }
    impl To<$ty1> for $ty {
        #[inline(always)]
        fn to(self) -> $ty1 {
            self as $ty1
        }
    }
)*

impl_to!($($ty,)*);

};
    () => {};
}

impl_to!(u8, i8, u16, i16, u32, i32, u64, i64, u128, i128, f32, f64,);

#[cfg(feature = "half")]
mod half_impl {
    use super::*;

    macro_rules! impl_to_half {
        ($ty1:ty, $($ty:ty,)*) => {

    impl To<half::f16> for $ty1 {
        #[inline(always)]
        fn to(self) -> half::f16 {
            (self as f32).to()
        }
    }
    impl To<half::bf16> for $ty1 {
        #[inline(always)]
        fn to(self) -> half::bf16 {
            (self as f32).to()
        }
    }
    impl To<$ty1> for half::f16 {
        #[inline(always)]
        fn to(self) -> $ty1 {
            self.to_f32().to()
        }
    }
    impl To<$ty1> for half::bf16 {
        #[inline(always)]
        fn to(self) -> $ty1 {
            self.to_f32().to()
        }
    }

    impl_to_half!($($ty,)*);
        };
        () => {};
    }

    impl_to_half!(u8, i8, u16, i16, u32, i32, u64, i64, u128, i128,);

    impl To<half::f16> for f32 {
        #[inline(always)]
        fn to(self) -> half::f16 {
            half::f16::from_f32(self)
        }
    }
    impl To<half::bf16> for f32 {
        #[inline(always)]
        fn to(self) -> half::bf16 {
            half::bf16::from_f32(self)
        }
    }
    impl To<half::f16> for f64 {
        #[inline(always)]
        fn to(self) -> half::f16 {
            half::f16::from_f64(self)
        }
    }
    impl To<half::bf16> for f64 {
        #[inline(always)]
        fn to(self) -> half::bf16 {
            half::bf16::from_f64(self)
        }
    }
    impl To<f32> for half::f16 {
        #[inline(always)]
        fn to(self) -> f32 {
            self.to_f32()
        }
    }
    impl To<f32> for half::bf16 {
        #[inline(always)]
        fn to(self) -> f32 {
            self.to_f32()
        }
    }
    impl To<f64> for half::f16 {
        #[inline(always)]
        fn to(self) -> f64 {
            self.to_f64()
        }
    }
    impl To<f64> for half::bf16 {
        #[inline(always)]
        fn to(self) -> f64 {
            self.to_f64()
        }
    }
}