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
pub fn digest(mut data: &[u8]) -> u32 {
let len: u32 = data.len() as u32;
let mut hash: u32 = len;
if len == 0 {
return 0;
}
let remainder = len & 3;
let block_count = len >> 2;
for _i in 0..block_count {
hash += u16::from_le_bytes([data[0], data[1]]) as u32;
let temp = (u16::from_le_bytes([data[2], data[3]]) as u32) << 11 ^ hash;
hash = hash << 16 ^ temp;
data = &data[4..];
hash += hash >> 11;
}
match remainder {
0 => {
}
1 => {
hash += data[0] as u32;
hash ^= hash << 10;
hash += hash >> 1;
},
2 => {
hash += u16::from_le_bytes([data[0], data[1]]) as u32;
hash ^= hash << 11;
hash += hash >> 17;
},
3 => {
hash += u16::from_le_bytes([data[0], data[1]]) as u32;
hash ^= hash << 16;
hash ^= (data[2] as u32) << 18;
hash += hash >> 11;
},
_ => {
panic!("Please investigate");
}
}
hash ^= hash << 3;
hash += hash >> 5;
hash ^= hash << 4;
hash += hash >> 17;
hash ^= hash << 25;
hash += hash >> 6;
return hash;
}
#[cfg(test)]
mod hiesh_tests {
use super::digest;
#[test]
fn test_digest() {
assert_eq!(digest("Hello World!".as_bytes()), 1774740540);
assert_eq!(digest("Hsieh Hash".as_bytes()), 1552477933);
assert_eq!(digest("SuperFastHash".as_bytes()), 2245601745);
}
}