summaryrefslogtreecommitdiff
path: root/src/support
diff options
context:
space:
mode:
authorDaniel Wirtz <dcode@dcode.io>2020-08-16 22:12:18 +0200
committerGitHub <noreply@github.com>2020-08-16 13:12:18 -0700
commit85154a1ea62a6c5d4fa698497543e2fe943e55e7 (patch)
tree716f36a2667fe75b9d5a9c34f6c6a44555b4ba4b /src/support
parent72f07db50f68001a555bd647e240e0f552dd098c (diff)
downloadbinaryen-85154a1ea62a6c5d4fa698497543e2fe943e55e7.tar.gz
binaryen-85154a1ea62a6c5d4fa698497543e2fe943e55e7.tar.bz2
binaryen-85154a1ea62a6c5d4fa698497543e2fe943e55e7.zip
Add 64-bit hash_combine (#3041)
Currently only the low 32-bits of a hash are guaranteed to be shuffled before combining with the other hash, so this PR also adds a 64-bit variant of hash_combine, including a comment on where the constants are coming from.
Diffstat (limited to 'src/support')
-rw-r--r--src/support/hash.h10
1 files changed, 9 insertions, 1 deletions
diff --git a/src/support/hash.h b/src/support/hash.h
index 50ecec563..1af462b7e 100644
--- a/src/support/hash.h
+++ b/src/support/hash.h
@@ -30,8 +30,16 @@ template<typename T> inline std::size_t hash(const T& value) {
// Combines two digests into the first digest. Use instead of `rehash` if
// `otherDigest` is another digest and not a `size_t` value.
static inline void hash_combine(std::size_t& digest, std::size_t otherDigest) {
- // see boost/container_hash/hash.hpp
+ // see: boost/container_hash/hash.hpp
+ // The constant is the N-bits reciprocal of the golden ratio:
+ // phi = (1 + sqrt(5)) / 2
+#if SIZE_MAX == UINT64_MAX
+ // trunc(2^64 / phi) = 0x9e3779b97f4a7c15
+ digest ^= otherDigest + 0x9e3779b97f4a7c15 + (digest << 12) + (digest >> 4);
+#else
+ // trunc(2^32 / phi) = 0x9e3779b9
digest ^= otherDigest + 0x9e3779b9 + (digest << 6) + (digest >> 2);
+#endif
}
// Hashes `value` and combines the resulting digest into the existing digest.