foldhash_portable/
fast.rs1use core::hash::{BuildHasher, Hasher};
4
5use crate::seed::{gen_per_hasher_seed, GlobalSeed, SharedSeed};
6use crate::{folded_multiply, hash_bytes_long, hash_bytes_short, rotate_right, ARBITRARY3};
7
8#[derive(Clone)]
14pub struct FoldHasher<'a> {
15 accumulator: u64,
16 sponge: u128,
17 sponge_len: u8,
18 seeds: &'a [u64; 6],
19}
20
21impl<'a> FoldHasher<'a> {
22 #[inline]
25 pub const fn with_seed(per_hasher_seed: u64, shared_seed: &'a SharedSeed) -> FoldHasher<'a> {
26 FoldHasher {
27 accumulator: per_hasher_seed,
28 sponge: 0,
29 sponge_len: 0,
30 seeds: &shared_seed.seeds,
31 }
32 }
33
34 #[inline(always)]
35 fn write_num<T: Into<u128>>(&mut self, x: T) {
36 let bits: usize = 8 * core::mem::size_of::<T>();
37 if self.sponge_len as usize + bits > 128 {
38 let lo = self.sponge as u64;
39 let hi = (self.sponge >> 64) as u64;
40 self.accumulator = folded_multiply(lo ^ self.accumulator, hi ^ self.seeds[0]);
41 self.sponge = x.into();
42 self.sponge_len = bits as u8;
43 } else {
44 self.sponge |= x.into() << self.sponge_len;
45 self.sponge_len += bits as u8;
46 }
47 }
48}
49
50impl<'a> Hasher for FoldHasher<'a> {
51 #[inline(always)]
52 fn write(&mut self, bytes: &[u8]) {
53 let len = bytes.len();
59 self.accumulator = rotate_right(self.accumulator, len as u32);
60 if len <= 16 {
61 self.accumulator = hash_bytes_short(bytes, self.accumulator, self.seeds);
62 } else {
63 unsafe {
64 self.accumulator = hash_bytes_long(bytes, self.accumulator, self.seeds);
66 }
67 }
68 }
69
70 #[inline(always)]
71 fn write_u8(&mut self, i: u8) {
72 self.write_num(i);
73 }
74
75 #[inline(always)]
76 fn write_u16(&mut self, i: u16) {
77 self.write_num(i);
78 }
79
80 #[inline(always)]
81 fn write_u32(&mut self, i: u32) {
82 self.write_num(i);
83 }
84
85 #[inline(always)]
86 fn write_u64(&mut self, i: u64) {
87 self.write_num(i);
88 }
89
90 #[inline(always)]
91 fn write_u128(&mut self, i: u128) {
92 let lo = i as u64;
93 let hi = (i >> 64) as u64;
94 self.accumulator = folded_multiply(lo ^ self.accumulator, hi ^ self.seeds[0]);
95 }
96
97 #[inline(always)]
98 fn write_usize(&mut self, i: usize) {
99 #[cfg(feature = "portable")]
101 self.write_num(i as u64);
102 #[cfg(not(feature = "portable"))]
103 {
104 #[cfg(target_pointer_width = "32")]
105 self.write_num(i as u32);
106 #[cfg(target_pointer_width = "64")]
107 self.write_num(i as u64);
108 }
109 }
110
111 #[cfg(feature = "nightly")]
112 #[inline(always)]
113 fn write_str(&mut self, s: &str) {
114 self.write(s.as_bytes())
116 }
117
118 #[inline(always)]
119 fn finish(&self) -> u64 {
120 if self.sponge_len > 0 {
121 let lo = self.sponge as u64;
122 let hi = (self.sponge >> 64) as u64;
123 folded_multiply(lo ^ self.accumulator, hi ^ self.seeds[0])
124 } else {
125 self.accumulator
126 }
127 }
128}
129
130#[derive(Clone, Debug)]
132pub struct RandomState {
133 per_hasher_seed: u64,
134 global_seed: GlobalSeed,
135}
136
137impl Default for RandomState {
138 #[inline(always)]
139 fn default() -> Self {
140 Self {
141 per_hasher_seed: gen_per_hasher_seed(),
142 global_seed: GlobalSeed::new(),
143 }
144 }
145}
146
147impl BuildHasher for RandomState {
148 type Hasher = FoldHasher<'static>;
149
150 #[inline(always)]
151 fn build_hasher(&self) -> FoldHasher<'static> {
152 FoldHasher::with_seed(self.per_hasher_seed, self.global_seed.get())
153 }
154}
155
156#[derive(Clone, Debug)]
162pub struct SeedableRandomState {
163 per_hasher_seed: u64,
164 shared_seed: &'static SharedSeed,
165}
166
167impl Default for SeedableRandomState {
168 #[inline(always)]
169 fn default() -> Self {
170 Self::random()
171 }
172}
173
174impl SeedableRandomState {
175 #[inline(always)]
177 pub fn random() -> Self {
178 Self {
179 per_hasher_seed: gen_per_hasher_seed(),
180 shared_seed: SharedSeed::global_random(),
181 }
182 }
183
184 #[inline(always)]
186 pub fn fixed() -> Self {
187 Self {
188 per_hasher_seed: ARBITRARY3,
189 shared_seed: SharedSeed::global_fixed(),
190 }
191 }
192
193 #[inline(always)]
196 pub fn with_seed(per_hasher_seed: u64, shared_seed: &'static SharedSeed) -> Self {
197 Self {
199 per_hasher_seed: per_hasher_seed ^ ARBITRARY3,
200 shared_seed,
201 }
202 }
203}
204
205impl BuildHasher for SeedableRandomState {
206 type Hasher = FoldHasher<'static>;
207
208 #[inline(always)]
209 fn build_hasher(&self) -> FoldHasher<'static> {
210 FoldHasher::with_seed(self.per_hasher_seed, self.shared_seed)
211 }
212}
213
214#[derive(Clone, Debug)]
218pub struct FixedState {
219 per_hasher_seed: u64,
220}
221
222impl FixedState {
223 #[inline(always)]
225 pub const fn with_seed(per_hasher_seed: u64) -> Self {
226 Self {
228 per_hasher_seed: per_hasher_seed ^ ARBITRARY3,
229 }
230 }
231}
232
233impl Default for FixedState {
234 #[inline(always)]
235 fn default() -> Self {
236 Self {
237 per_hasher_seed: ARBITRARY3,
238 }
239 }
240}
241
242impl BuildHasher for FixedState {
243 type Hasher = FoldHasher<'static>;
244
245 #[inline(always)]
246 fn build_hasher(&self) -> FoldHasher<'static> {
247 FoldHasher::with_seed(self.per_hasher_seed, SharedSeed::global_fixed())
248 }
249}