#pragma once #include <util/system/defaults.h> #define FNV32INIT 2166136261U #define FNV32PRIME 16777619U #define FNV64INIT ULL(14695981039346656037) #define FNV64PRIME ULL(1099511628211) namespace NFnvPrivate { template <class It> constexpr ui32 FnvHash32(It b, It e, ui32 init) { while (b != e) { init = (init * FNV32PRIME) ^ (unsigned char)*b++; } return init; } template <class It> constexpr ui64 FnvHash64(It b, It e, ui64 init) { while (b != e) { init = (init * FNV64PRIME) ^ (unsigned char)*b++; } return init; } template <unsigned N> struct TFnvHelper; #define DEF_FNV(t) \ template <> \ struct TFnvHelper<t> { \ static const ui##t Init = FNV##t##INIT; \ template <class It> \ static constexpr ui##t FnvHash(It b, It e, ui##t init) { \ return FnvHash##t(b, e, init); \ } \ }; DEF_FNV(32) DEF_FNV(64) #undef DEF_FNV } // namespace NFnvPrivate template <class T, class It> static constexpr T FnvHash(It b, It e, T init) { static_assert(sizeof(*b) == 1, "expect sizeof(*b) == 1"); return (T)NFnvPrivate::TFnvHelper<8 * sizeof(T)>::FnvHash(b, e, init); } template <class T, class It> static constexpr T FnvHash(It b, It e) { return FnvHash<T>(b, e, (T)NFnvPrivate::TFnvHelper<8 * sizeof(T)>::Init); } template <class T> static constexpr T FnvHash(const void* buf, size_t len, T init) { return FnvHash<T>((const unsigned char*)buf, (const unsigned char*)buf + len, init); } template <class T> static constexpr T FnvHash(const void* buf, size_t len) { return FnvHash<T>((const unsigned char*)buf, (const unsigned char*)buf + len); } template <class T, class Buf> static constexpr T FnvHash(const Buf& buf) { return FnvHash<T>(buf.data(), buf.size() * sizeof(*buf.data())); }