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 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143
use core::mem::transmute; pub(crate) trait Convert<To> { fn convert(self) -> To; fn convert_ref(&self) -> &To; fn convert_mut_ref(&mut self) -> &mut To; } macro_rules! convert { ($from:ty, $to:ty) => { impl Convert<$to> for $from { #[inline(always)] fn convert(self) -> $to { unsafe { transmute(self) } } #[inline(always)] fn convert_ref(&self) -> &$to { unsafe { transmute(self) } } #[inline(always)] fn convert_mut_ref(&mut self) -> &mut $to { unsafe { transmute(self) } } } impl Convert<$from> for $to { #[inline(always)] fn convert(self) -> $from { unsafe { transmute(self) } } #[inline(always)] fn convert_ref(&self) -> &$from { unsafe { transmute(self) } } #[inline(always)] fn convert_mut_ref(&mut self) -> &mut $from { unsafe { transmute(self) } } } }; } convert!(u128, [u64; 2]); convert!(u128, [u32; 4]); convert!(u128, [u16; 8]); convert!(u128, [u8; 16]); convert!([u64; 2], [u32; 4]); convert!([u64; 2], [u16; 8]); convert!([u64; 2], [u8; 16]); convert!([u32; 4], [u16; 8]); convert!([u32; 4], [u8; 16]); convert!([u16; 8], [u8; 16]); convert!(u64, [u32; 2]); convert!(u64, [u16; 4]); convert!(u64, [u8; 8]); convert!([u32; 2], [u16; 4]); convert!([u32; 2], [u8; 8]); convert!(u32, [u16; 2]); convert!(u32, [u8; 4]); convert!([u16; 2], [u8; 4]); convert!(u16, [u8; 2]); convert!([f64; 2], [u8; 16]); convert!([f32; 4], [u8; 16]); convert!(f64, [u8; 8]); convert!([f32; 2], [u8; 8]); convert!(f32, [u8; 4]); macro_rules! as_array { ($input:expr, $len:expr) => {{ { #[inline(always)] fn as_array<T>(slice: &[T]) -> &[T; $len] { assert_eq!(slice.len(), $len); unsafe { &*(slice.as_ptr() as *const [_; $len]) } } as_array($input) } }} } pub(crate) trait ReadFromSlice { fn read_u16(&self) -> (u16, &[u8]); fn read_u32(&self) -> (u32, &[u8]); fn read_u64(&self) -> (u64, &[u8]); fn read_u128(&self) -> (u128, &[u8]); fn read_last_u16(&self) -> u16; fn read_last_u32(&self) -> u32; fn read_last_u64(&self) -> u64; fn read_last_u128(&self) -> u128; } impl ReadFromSlice for [u8] { #[inline(always)] fn read_u16(&self) -> (u16, &[u8]) { let (value, rest) = self.split_at(2); (as_array!(value, 2).convert(), rest) } #[inline(always)] fn read_u32(&self) -> (u32, &[u8]) { let (value, rest) = self.split_at(4); (as_array!(value, 4).convert(), rest) } #[inline(always)] fn read_u64(&self) -> (u64, &[u8]) { let (value, rest) = self.split_at(8); (as_array!(value, 8).convert(), rest) } #[inline(always)] fn read_u128(&self) -> (u128, &[u8]) { let (value, rest) = self.split_at(16); (as_array!(value, 16).convert(), rest) } #[inline(always)] fn read_last_u16(&self) -> u16 { let (_, value) = self.split_at(self.len() - 2); as_array!(value, 2).convert() } #[inline(always)] fn read_last_u32(&self) -> u32 { let (_, value) = self.split_at(self.len() - 4); as_array!(value, 4).convert() } #[inline(always)] fn read_last_u64(&self) -> u64 { let (_, value) = self.split_at(self.len() - 8); as_array!(value, 8).convert() } #[inline(always)] fn read_last_u128(&self) -> u128 { let (_, value) = self.split_at(self.len() - 16); as_array!(value, 16).convert() } }