#if !defined(phmap_utils_h_guard_) #define phmap_utils_h_guard_ // --------------------------------------------------------------------------- // Copyright (c) 2019, Gregory Popovitch - greg7mdp@gmail.com // // minimal header providing phmap::HashState // // use as: phmap::HashState().combine(0, _first_name, _last_name, _age); // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // https://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // --------------------------------------------------------------------------- #ifdef _MSC_VER #pragma warning(push) #pragma warning(disable : 4514) // unreferenced inline function has been removed #pragma warning(disable : 4710) // function not inlined #pragma warning(disable : 4711) // selected for automatic inline expansion #endif #include #include #include #include "phmap_bits.h" // --------------------------------------------------------------- // Absl forward declaration requires global scope. // --------------------------------------------------------------- #if defined(PHMAP_USE_ABSL_HASH) && !defined(phmap_fwd_decl_h_guard_) && !defined(ABSL_HASH_HASH_H_) namespace absl { template struct Hash; }; #endif namespace phmap { // --------------------------------------------------------------- // --------------------------------------------------------------- template struct phmap_mix { inline size_t operator()(size_t) const; }; template<> struct phmap_mix<4> { inline size_t operator()(size_t a) const { static constexpr uint64_t kmul = 0xcc9e2d51UL; // static constexpr uint64_t kmul = 0x3B9ACB93UL; // [greg] my own random prime uint64_t l = a * kmul; return static_cast(l ^ (l >> 32)); } }; #if defined(PHMAP_HAS_UMUL128) template<> struct phmap_mix<8> { // Very fast mixing (similar to Abseil) inline size_t operator()(size_t a) const { static constexpr uint64_t k = 0xde5fb9d2630458e9ULL; // static constexpr uint64_t k = 0x7C9D0BF0567102A5ULL; // [greg] my own random prime uint64_t h; uint64_t l = umul128(a, k, &h); return static_cast(h + l); } }; #else template<> struct phmap_mix<8> { inline size_t operator()(size_t a) const { a = (~a) + (a << 21); // a = (a << 21) - a - 1; a = a ^ (a >> 24); a = (a + (a << 3)) + (a << 8); // a * 265 a = a ^ (a >> 14); a = (a + (a << 2)) + (a << 4); // a * 21 a = a ^ (a >> 28); a = a + (a << 31); return static_cast(a); } }; #endif // -------------------------------------------- template struct fold_if_needed { inline size_t operator()(uint64_t) const; }; template<> struct fold_if_needed<4> { inline size_t operator()(uint64_t a) const { return static_cast(a ^ (a >> 32)); } }; template<> struct fold_if_needed<8> { inline size_t operator()(uint64_t a) const { return static_cast(a); } }; // --------------------------------------------------------------- // see if class T has a hash_value() friend method // --------------------------------------------------------------- template struct has_hash_value { private: typedef std::true_type yes; typedef std::false_type no; template static auto test(int) -> decltype(hash_value(std::declval()) == 1, yes()); template static no test(...); public: static constexpr bool value = std::is_same(0)), yes>::value; }; #if defined(PHMAP_USE_ABSL_HASH) && !defined(phmap_fwd_decl_h_guard_) template using Hash = ::absl::Hash; #elif !defined(PHMAP_USE_ABSL_HASH) // --------------------------------------------------------------- // phmap::Hash // --------------------------------------------------------------- template struct Hash { template ::value, int>::type = 0> size_t _hash(const T& val) const { return hash_value(val); } template ::value, int>::type = 0> size_t _hash(const T& val) const { return std::hash()(val); } inline size_t operator()(const T& val) const { return _hash(val); } }; template struct Hash { inline size_t operator()(const T *val) const noexcept { return static_cast(reinterpret_cast(val)); } }; template struct phmap_unary_function { typedef ArgumentType argument_type; typedef ResultType result_type; }; template <> struct Hash : public phmap_unary_function { inline size_t operator()(bool val) const noexcept { return static_cast(val); } }; template <> struct Hash : public phmap_unary_function { inline size_t operator()(char val) const noexcept { return static_cast(val); } }; template <> struct Hash : public phmap_unary_function { inline size_t operator()(signed char val) const noexcept { return static_cast(val); } }; template <> struct Hash : public phmap_unary_function { inline size_t operator()(unsigned char val) const noexcept { return static_cast(val); } }; #ifdef PHMAP_HAS_NATIVE_WCHAR_T template <> struct Hash : public phmap_unary_function { inline size_t operator()(wchar_t val) const noexcept { return static_cast(val); } }; #endif template <> struct Hash : public phmap_unary_function { inline size_t operator()(int16_t val) const noexcept { return static_cast(val); } }; template <> struct Hash : public phmap_unary_function { inline size_t operator()(uint16_t val) const noexcept { return static_cast(val); } }; template <> struct Hash : public phmap_unary_function { inline size_t operator()(int32_t val) const noexcept { return static_cast(val); } }; template <> struct Hash : public phmap_unary_function { inline size_t operator()(uint32_t val) const noexcept { return static_cast(val); } }; template <> struct Hash : public phmap_unary_function { inline size_t operator()(int64_t val) const noexcept { return fold_if_needed()(static_cast(val)); } }; template <> struct Hash : public phmap_unary_function { inline size_t operator()(uint64_t val) const noexcept { return fold_if_needed()(val); } }; template <> struct Hash : public phmap_unary_function { inline size_t operator()(float val) const noexcept { // -0.0 and 0.0 should return same hash uint32_t *as_int = reinterpret_cast(&val); return (val == 0) ? static_cast(0) : static_cast(*as_int); } }; template <> struct Hash : public phmap_unary_function { inline size_t operator()(double val) const noexcept { // -0.0 and 0.0 should return same hash uint64_t *as_int = reinterpret_cast(&val); return (val == 0) ? static_cast(0) : fold_if_needed()(*as_int); } }; #endif template struct Combiner { H operator()(H seed, size_t value); }; template struct Combiner { H operator()(H seed, size_t value) { return seed ^ (value + 0x9e3779b9 + (seed << 6) + (seed >> 2)); } }; template struct Combiner { H operator()(H seed, size_t value) { return seed ^ (value + size_t(0xc6a4a7935bd1e995) + (seed << 6) + (seed >> 2)); } }; // define HashState to combine member hashes... see example below // ----------------------------------------------------------------------------- template class HashStateBase { public: template static H combine(H state, const T& value, const Ts&... values); static H combine(H state) { return state; } }; template template H HashStateBase::combine(H seed, const T& v, const Ts&... vs) { return HashStateBase::combine(Combiner()( seed, phmap::Hash()(v)), vs...); } using HashState = HashStateBase; // ----------------------------------------------------------------------------- #if !defined(PHMAP_USE_ABSL_HASH) // define Hash for std::pair // ------------------------- template struct Hash> { size_t operator()(std::pair const& p) const noexcept { return phmap::HashState().combine(phmap::Hash()(p.first), p.second); } }; // define Hash for std::tuple // -------------------------- template struct Hash> { size_t operator()(std::tuple const& t) const noexcept { return _hash_helper(t); } private: template typename std::enable_if::type _hash_helper(const std::tuple &) const noexcept { return 0; } template typename std::enable_if::type _hash_helper(const std::tuple &t) const noexcept { const auto &el = std::get(t); using el_type = typename std::remove_cv::type>::type; return Combiner()( phmap::Hash()(el), _hash_helper(t)); } }; #endif } // namespace phmap #ifdef _MSC_VER #pragma warning(pop) #endif #endif // phmap_utils_h_guard_