#include "util/hash.h"
#include <string>
#include "port/lang.h"
#include "util/coding.h"
#include "util/hash128.h"
#include "util/math128.h"
#include "util/xxhash.h"
#include "util/xxph3.h"
namespace ROCKSDB_NAMESPACE {
uint64_t (*kGetSliceNPHash64UnseededFnPtr)(const Slice&) = &GetSliceHash64;
uint32_t Hash(const char* data, size_t n, uint32_t seed) {
const uint32_t m = 0xc6a4a793;
const uint32_t r = 24;
const char* limit = data + n;
uint32_t h = static_cast<uint32_t>(seed ^ (n * m));
while (data + 4 <= limit) {
uint32_t w = DecodeFixed32(data);
data += 4;
h += w;
h *= m;
h ^= (h >> 16);
}
switch (limit - data) {
case 3:
h += static_cast<uint32_t>(static_cast<int8_t>(data[2])) << 16;
FALLTHROUGH_INTENDED;
case 2:
h += static_cast<uint32_t>(static_cast<int8_t>(data[1])) << 8;
FALLTHROUGH_INTENDED;
case 1:
h += static_cast<uint32_t>(static_cast<int8_t>(data[0]));
h *= m;
h ^= (h >> r);
break;
}
return h;
}
uint64_t Hash64(const char* data, size_t n, uint64_t seed) {
return XXPH3_64bits_withSeed(data, n, seed);
}
uint64_t Hash64(const char* data, size_t n) {
return XXPH3_64bits(data, n);
}
uint64_t GetSlicePartsNPHash64(const SliceParts& data, uint64_t seed) {
size_t concat_len = 0;
for (int i = 0; i < data.num_parts; ++i) {
concat_len += data.parts[i].size();
}
std::string concat_data;
concat_data.reserve(concat_len);
for (int i = 0; i < data.num_parts; ++i) {
concat_data.append(data.parts[i].data(), data.parts[i].size());
}
assert(concat_data.size() == concat_len);
return NPHash64(concat_data.data(), concat_len, seed);
}
Unsigned128 Hash128(const char* data, size_t n, uint64_t seed) {
auto h = XXH3_128bits_withSeed(data, n, seed);
return (Unsigned128{h.high64} << 64) | (h.low64);
}
Unsigned128 Hash128(const char* data, size_t n) {
auto h = XXH3_128bits(data, n);
return (Unsigned128{h.high64} << 64) | (h.low64);
}
void Hash2x64(const char* data, size_t n, uint64_t* high64, uint64_t* low64) {
auto h = XXH3_128bits(data, n);
*high64 = h.high64;
*low64 = h.low64;
}
void Hash2x64(const char* data, size_t n, uint64_t seed, uint64_t* high64,
uint64_t* low64) {
auto h = XXH3_128bits_withSeed(data, n, seed);
*high64 = h.high64;
*low64 = h.low64;
}
namespace {
inline uint64_t XXH3_avalanche(uint64_t h64) {
h64 ^= h64 >> 37;
h64 *= 0x165667919E3779F9U;
h64 ^= h64 >> 32;
return h64;
}
inline uint64_t XXH3_unavalanche(uint64_t h64) {
h64 ^= h64 >> 32;
h64 *= 0x8da8ee41d6df849U; h64 ^= h64 >> 37;
return h64;
}
}
void BijectiveHash2x64(uint64_t in_high64, uint64_t in_low64, uint64_t seed,
uint64_t* out_high64, uint64_t* out_low64) {
const uint64_t bitflipl = 0x59973f0033362349U - seed;
const uint64_t bitfliph = 0xc202797692d63d58U + seed;
Unsigned128 tmp128 =
Multiply64to128(in_low64 ^ in_high64 ^ bitflipl, 0x9E3779B185EBCA87U);
uint64_t lo = Lower64of128(tmp128);
uint64_t hi = Upper64of128(tmp128);
lo += 0x3c0000000000000U; in_high64 ^= bitfliph;
hi += in_high64 + (Lower32of64(in_high64) * uint64_t{0x85EBCA76});
lo ^= EndianSwapValue(hi);
tmp128 = Multiply64to128(lo, 0xC2B2AE3D27D4EB4FU);
lo = Lower64of128(tmp128);
hi = Upper64of128(tmp128) + (hi * 0xC2B2AE3D27D4EB4FU);
*out_low64 = XXH3_avalanche(lo);
*out_high64 = XXH3_avalanche(hi);
}
void BijectiveUnhash2x64(uint64_t in_high64, uint64_t in_low64, uint64_t seed,
uint64_t* out_high64, uint64_t* out_low64) {
const uint64_t bitflipl = 0x59973f0033362349U - seed;
const uint64_t bitfliph = 0xc202797692d63d58U + seed;
uint64_t lo = XXH3_unavalanche(in_low64);
uint64_t hi = XXH3_unavalanche(in_high64);
lo *= 0xba79078168d4baf; hi -= Upper64of128(Multiply64to128(lo, 0xC2B2AE3D27D4EB4FU));
hi *= 0xba79078168d4baf; lo ^= EndianSwapValue(hi);
lo -= 0x3c0000000000000U;
lo *= 0x887493432badb37U; hi -= Upper64of128(Multiply64to128(lo, 0x9E3779B185EBCA87U));
uint32_t tmp32 = Lower32of64(hi) * 0xb6c92f47; hi -= tmp32;
hi = (hi & 0xFFFFFFFF00000000U) -
((tmp32 * uint64_t{0x85EBCA76}) & 0xFFFFFFFF00000000U) + tmp32;
hi ^= bitfliph;
lo ^= hi ^ bitflipl;
*out_high64 = hi;
*out_low64 = lo;
}
void BijectiveHash2x64(uint64_t in_high64, uint64_t in_low64,
uint64_t* out_high64, uint64_t* out_low64) {
BijectiveHash2x64(in_high64, in_low64, 0, out_high64, out_low64);
}
void BijectiveUnhash2x64(uint64_t in_high64, uint64_t in_low64,
uint64_t* out_high64, uint64_t* out_low64) {
BijectiveUnhash2x64(in_high64, in_low64, 0, out_high64, out_low64);
}
}