| 1 | // Copyright 2019 The Rust Project Developers. See the COPYRIGHT | 
| 2 | // file at the top-level directory of this distribution and at | 
|---|
| 3 | // http://rust-lang.org/COPYRIGHT. | 
|---|
| 4 | // | 
|---|
| 5 | // Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or | 
|---|
| 6 | // http://www.apache.org/licenses/LICENSE-2.0> or the MIT license | 
|---|
| 7 | // <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your | 
|---|
| 8 | // option. This file may not be copied, modified, or distributed | 
|---|
| 9 | // except according to those terms. | 
|---|
| 10 |  | 
|---|
| 11 | //! Support for lookups based on minimal perfect hashing. | 
|---|
| 12 |  | 
|---|
| 13 | // This function is based on multiplication being fast and is "good enough". Also | 
|---|
| 14 | // it can share some work between the unsalted and salted versions. | 
|---|
| 15 | #[ inline] | 
|---|
| 16 | fn my_hash(key: u32, salt: u32, n: usize) -> usize { | 
|---|
| 17 | let y: u32 = key.wrapping_add(salt).wrapping_mul(2654435769); | 
|---|
| 18 | let y: u32 = y ^ key.wrapping_mul(0x31415926); | 
|---|
| 19 | (((y as u64) * (n as u64)) >> 32) as usize | 
|---|
| 20 | } | 
|---|
| 21 |  | 
|---|
| 22 | /// Do a lookup using minimal perfect hashing. | 
|---|
| 23 | /// | 
|---|
| 24 | /// The table is stored as a sequence of "salt" values, then a sequence of | 
|---|
| 25 | /// values that contain packed key/value pairs. The strategy is to hash twice. | 
|---|
| 26 | /// The first hash retrieves a salt value that makes the second hash unique. | 
|---|
| 27 | /// The hash function doesn't have to be very good, just good enough that the | 
|---|
| 28 | /// resulting map is unique. | 
|---|
| 29 | #[ inline] | 
|---|
| 30 | pub(crate) fn mph_lookup<KV, V, FK, FV>( | 
|---|
| 31 | x: u32, | 
|---|
| 32 | salt: &[u16], | 
|---|
| 33 | kv: &[KV], | 
|---|
| 34 | fk: FK, | 
|---|
| 35 | fv: FV, | 
|---|
| 36 | default: V, | 
|---|
| 37 | ) -> V | 
|---|
| 38 | where | 
|---|
| 39 | KV: Copy, | 
|---|
| 40 | FK: Fn(KV) -> u32, | 
|---|
| 41 | FV: Fn(KV) -> V, | 
|---|
| 42 | { | 
|---|
| 43 | let s: u32 = salt[my_hash(key:x, salt:0, n:salt.len())] as u32; | 
|---|
| 44 | let key_val: KV = kv[my_hash(key:x, salt:s, n:salt.len())]; | 
|---|
| 45 | if x == fk(key_val) { | 
|---|
| 46 | fv(key_val) | 
|---|
| 47 | } else { | 
|---|
| 48 | default | 
|---|
| 49 | } | 
|---|
| 50 | } | 
|---|
| 51 |  | 
|---|