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
use std::mem::transmute;
pub trait FixedInt: Sized + Copy {
const REQUIRED_SPACE: usize;
fn required_space() -> usize;
fn encode_fixed(self, src: &mut [u8]);
fn decode_fixed(src: &[u8]) -> Self;
fn encode_fixed_light<'a>(&'a self) -> &'a [u8];
fn encode_fixed_vec(self) -> Vec<u8> {
let mut v = Vec::new();
v.resize(Self::required_space(), 0);
self.encode_fixed(&mut v[..]);
v
}
fn decode_fixed_vec(v: &Vec<u8>) -> Self {
assert_eq!(v.len(), Self::required_space());
Self::decode_fixed(&v[..])
}
}
macro_rules! impl_fixedint {
($t:ty, $sz:expr) => {
impl FixedInt for $t {
const REQUIRED_SPACE: usize = $sz;
fn required_space() -> usize {
Self::REQUIRED_SPACE
}
fn encode_fixed_light<'a>(&'a self) -> &'a [u8] {
return unsafe {
std::slice::from_raw_parts(
transmute::<&$t, *const u8>(&self),
Self::REQUIRED_SPACE,
)
};
}
fn encode_fixed(self, dst: &mut [u8]) {
assert_eq!(dst.len(), Self::REQUIRED_SPACE);
let encoded = unsafe { &*(&self as *const $t as *const [u8; $sz]) };
dst.clone_from_slice(encoded);
}
fn decode_fixed(src: &[u8]) -> $t {
assert_eq!(src.len(), Self::REQUIRED_SPACE);
return unsafe { *(src.as_ptr() as *const $t) };
}
}
};
}
impl_fixedint!(usize, 8);
impl_fixedint!(u64, 8);
impl_fixedint!(u32, 4);
impl_fixedint!(u16, 2);
impl_fixedint!(isize, 8);
impl_fixedint!(i64, 8);
impl_fixedint!(i32, 4);
impl_fixedint!(i16, 2);