Skip to content

Commit eaeef3d

Browse files
committed
std: Cache HashMap keys in TLS
This is a rebase and extension of rust-lang#31356 where we cache the keys in thread local storage. This should give us a nice speed bost in creating hash maps along with mostly retaining the property that all maps have a nondeterministic iteration order. Closes rust-lang#27243
1 parent 764ef92 commit eaeef3d

File tree

1 file changed

+27
-2
lines changed
  • src/libstd/collections/hash

1 file changed

+27
-2
lines changed

src/libstd/collections/hash/map.rs

+27-2
Original file line numberDiff line numberDiff line change
@@ -1667,8 +1667,33 @@ impl RandomState {
16671667
#[allow(deprecated)] // rand
16681668
#[stable(feature = "hashmap_build_hasher", since = "1.7.0")]
16691669
pub fn new() -> RandomState {
1670-
let mut r = rand::thread_rng();
1671-
RandomState { k0: r.gen(), k1: r.gen() }
1670+
// Historically this function did not cache keys from the OS and instead
1671+
// simply always called `rand::thread_rng().gen()` twice. In #31356 it
1672+
// was discovered, however, that because we re-seed the thread-local RNG
1673+
// from the OS periodically that this can cause excessive slowdown when
1674+
// many hash maps are created on a thread. To solve this performance
1675+
// trap we cache the first set of randomly generated keys per-thread.
1676+
//
1677+
// In doing this, however, we lose the property that all hash maps have
1678+
// nondeterministic iteration order as all of those created on the same
1679+
// thread would have the same hash keys. This property has been nice in
1680+
// the past as it allows for maximal flexibility in the implementation
1681+
// of `HashMap` itself.
1682+
//
1683+
// The constraint here (if there even is one) is just that maps created
1684+
// on the same thread have the same iteration order, and that *may* be
1685+
// relied upon even though it is not a documented guarantee at all of
1686+
// the `HashMap` type. In any case we've decided that this is reasonable
1687+
// for now, so caching keys thread-locally seems fine.
1688+
thread_local!(static KEYS: (u64, u64) = {
1689+
let r = rand::OsRng::new();
1690+
let mut r = r.expect("failed to create an OS RNG");
1691+
(r.gen(), r.gen())
1692+
});
1693+
1694+
KEYS.with(|&(k0, k1)| {
1695+
RandomState { k0: k0, k1: k1 }
1696+
})
16721697
}
16731698
}
16741699

0 commit comments

Comments
 (0)