mirror of
https://github.com/boostorg/unordered.git
synced 2025-07-30 11:27:15 +02:00
dropped foa_mixer in favor of internal mix policy governed by hash_traits
This commit is contained in:
@ -21,6 +21,8 @@
|
|||||||
#include <boost/cstdint.hpp>
|
#include <boost/cstdint.hpp>
|
||||||
#include <boost/predef.h>
|
#include <boost/predef.h>
|
||||||
#include <boost/type_traits/is_nothrow_swappable.hpp>
|
#include <boost/type_traits/is_nothrow_swappable.hpp>
|
||||||
|
#include <boost/unordered/detail/xmx.hpp>
|
||||||
|
#include <boost/unordered/hash_traits.hpp>
|
||||||
#include <climits>
|
#include <climits>
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
#include <cstddef>
|
#include <cstddef>
|
||||||
@ -752,6 +754,19 @@ struct table_arrays
|
|||||||
value_type *elements;
|
value_type *elements;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct no_mix
|
||||||
|
{
|
||||||
|
static inline std::size_t mix(std::size_t x)noexcept{return x;}
|
||||||
|
};
|
||||||
|
|
||||||
|
struct xmx_mix
|
||||||
|
{
|
||||||
|
static inline std::size_t mix(std::size_t x)noexcept
|
||||||
|
{
|
||||||
|
return xmx(x);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
struct if_constexpr_void_else{void operator()()const{}};
|
struct if_constexpr_void_else{void operator()()const{}};
|
||||||
|
|
||||||
template<bool B,typename F,typename G=if_constexpr_void_else>
|
template<bool B,typename F,typename G=if_constexpr_void_else>
|
||||||
@ -827,6 +842,11 @@ table:empty_value<Hash,0>,empty_value<Pred,1>,empty_value<Allocator,1>
|
|||||||
static constexpr auto N=group_type::N;
|
static constexpr auto N=group_type::N;
|
||||||
using size_policy=pow2_size_policy;
|
using size_policy=pow2_size_policy;
|
||||||
using prober=pow2_quadratic_prober;
|
using prober=pow2_quadratic_prober;
|
||||||
|
using mix_policy=typename std::conditional<
|
||||||
|
hash_traits<Hash>::is_avalanching::value,
|
||||||
|
no_mix,
|
||||||
|
xmx_mix
|
||||||
|
>::type;
|
||||||
using alloc_traits=boost::allocator_traits<Allocator>;
|
using alloc_traits=boost::allocator_traits<Allocator>;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
@ -1158,7 +1178,7 @@ public:
|
|||||||
template<typename Key>
|
template<typename Key>
|
||||||
BOOST_FORCEINLINE iterator find(const Key& x)
|
BOOST_FORCEINLINE iterator find(const Key& x)
|
||||||
{
|
{
|
||||||
auto hash=h()(x);
|
auto hash=hash_for(x);
|
||||||
return find_impl(x,position_for(hash),hash);
|
return find_impl(x,position_for(hash),hash);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1285,6 +1305,12 @@ private:
|
|||||||
return std::get<0>(k);
|
return std::get<0>(k);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<typename Key>
|
||||||
|
inline std::size_t hash_for(const Key& x)
|
||||||
|
{
|
||||||
|
return mix_policy::mix(h()(x));
|
||||||
|
}
|
||||||
|
|
||||||
inline std::size_t position_for(std::size_t hash)const
|
inline std::size_t position_for(std::size_t hash)const
|
||||||
{
|
{
|
||||||
return position_for(hash,arrays);
|
return position_for(hash,arrays);
|
||||||
@ -1341,7 +1367,7 @@ private:
|
|||||||
BOOST_FORCEINLINE std::pair<iterator,bool> emplace_impl(Args&&... args)
|
BOOST_FORCEINLINE std::pair<iterator,bool> emplace_impl(Args&&... args)
|
||||||
{
|
{
|
||||||
const auto &k=key_from(std::forward<Args>(args)...);
|
const auto &k=key_from(std::forward<Args>(args)...);
|
||||||
auto hash=h()(k);
|
auto hash=hash_for(k);
|
||||||
auto pos0=position_for(hash);
|
auto pos0=position_for(hash);
|
||||||
auto it=find_impl(k,pos0,hash);
|
auto it=find_impl(k,pos0,hash);
|
||||||
|
|
||||||
@ -1398,13 +1424,13 @@ private:
|
|||||||
template<typename Value>
|
template<typename Value>
|
||||||
void unchecked_insert(Value&& x)
|
void unchecked_insert(Value&& x)
|
||||||
{
|
{
|
||||||
auto hash=h()(key_from(x));
|
auto hash=hash_for(key_from(x));
|
||||||
unchecked_emplace_at(position_for(hash),hash,std::forward<Value>(x));
|
unchecked_emplace_at(position_for(hash),hash,std::forward<Value>(x));
|
||||||
}
|
}
|
||||||
|
|
||||||
void nosize_transfer_element(value_type* p,const arrays_type& arrays_)
|
void nosize_transfer_element(value_type* p,const arrays_type& arrays_)
|
||||||
{
|
{
|
||||||
auto hash=h()(key_from(*p));
|
auto hash=hash_for(key_from(*p));
|
||||||
nosize_unchecked_emplace_at(
|
nosize_unchecked_emplace_at(
|
||||||
arrays_,position_for(hash,arrays_),hash,type_policy::move(*p));
|
arrays_,position_for(hash,arrays_),hash,type_policy::move(*p));
|
||||||
destroy_element(p);
|
destroy_element(p);
|
||||||
|
@ -1,175 +0,0 @@
|
|||||||
/* Hash mixer for boost::unordered::unordered_flat_[map|set].
|
|
||||||
*
|
|
||||||
* Copyright 2022 Joaquin M Lopez Munoz.
|
|
||||||
* Copyright 2022 Peter Dimov.
|
|
||||||
* Distributed under the Boost Software License, Version 1.0.
|
|
||||||
* (See accompanying file LICENSE_1_0.txt or copy at
|
|
||||||
* http://www.boost.org/LICENSE_1_0.txt)
|
|
||||||
*
|
|
||||||
* See https://www.boost.org/libs/unordered for library home page.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef BOOST_UNORDERED_DETAIL_FOA_MIXER_HPP
|
|
||||||
#define BOOST_UNORDERED_DETAIL_FOA_MIXER_HPP
|
|
||||||
|
|
||||||
#include <boost/config.hpp>
|
|
||||||
#include <boost/container_hash/hash_fwd.hpp>
|
|
||||||
#include <boost/core/empty_value.hpp>
|
|
||||||
#include <boost/cstdint.hpp>
|
|
||||||
#include <climits>
|
|
||||||
#include <cstddef>
|
|
||||||
#include <type_traits>
|
|
||||||
#include <utility>
|
|
||||||
|
|
||||||
namespace boost{
|
|
||||||
namespace unordered{
|
|
||||||
namespace detail{
|
|
||||||
namespace foa{
|
|
||||||
|
|
||||||
/* mixer<Hash> is functionally equivalent to Hash except if Hash is
|
|
||||||
* boost::hash<T> for any of the Ts where boost::hash implements a
|
|
||||||
* trivial hashing function not fit for open-addressing hash container:
|
|
||||||
* in these cases, the result of boost::hash is post-mixed using
|
|
||||||
*
|
|
||||||
* - 64 bits: xmx (TODO: Peter Dimov to explain)
|
|
||||||
* - 32 bits: xmx33 (TODO: Peter Dimov to explain)
|
|
||||||
*/
|
|
||||||
|
|
||||||
#if defined(BOOST_GCC)
|
|
||||||
/* GCC's -Wshadow triggers at scenarios like this:
|
|
||||||
*
|
|
||||||
* struct foo{};
|
|
||||||
* template<typename Base>
|
|
||||||
* struct derived:Base
|
|
||||||
* {
|
|
||||||
* void f(){int foo;}
|
|
||||||
* };
|
|
||||||
*
|
|
||||||
* derived<foo>x;
|
|
||||||
* x.f(); // declaration of "foo" in derived::f shadows base type "foo"
|
|
||||||
*
|
|
||||||
* This makes shadowing warnings unavoidable in general when a class template
|
|
||||||
* derives from a user-provided class, as is the case with mixer_impl.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma GCC diagnostic push
|
|
||||||
#pragma GCC diagnostic ignored "-Wshadow"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
template<typename Hash,typename MixPolicy>
|
|
||||||
class mixer_impl:empty_value<Hash>
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
using base=empty_value<Hash>;
|
|
||||||
|
|
||||||
#if BOOST_CXX_VERSION<201703L
|
|
||||||
using argument_type=typename Hash::argument_type;
|
|
||||||
using result_type=std::size_t;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
mixer_impl()=default;
|
|
||||||
template<typename... Args>
|
|
||||||
mixer_impl(Args&&... args):base{empty_init,std::forward<Args>(args)...}{}
|
|
||||||
|
|
||||||
Hash& get_base()noexcept{return base::get();}
|
|
||||||
const Hash& get_base()const noexcept{return base::get();}
|
|
||||||
|
|
||||||
template<
|
|
||||||
typename Key,
|
|
||||||
typename std::enable_if<
|
|
||||||
std::is_same<
|
|
||||||
std::size_t,
|
|
||||||
decltype(std::declval<const Hash>()(std::declval<const Key&>()))
|
|
||||||
>::value
|
|
||||||
>::type* =nullptr
|
|
||||||
>
|
|
||||||
std::size_t operator()(const Key& x)const
|
|
||||||
noexcept(noexcept(std::declval<const Hash>()(x)))
|
|
||||||
{
|
|
||||||
return MixPolicy::mix(get_base()(x));
|
|
||||||
}
|
|
||||||
|
|
||||||
friend void swap(mixer_impl& x,mixer_impl& y)
|
|
||||||
{
|
|
||||||
using std::swap;
|
|
||||||
swap(x.get_base(),y.get_base());
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
#if defined(BOOST_GCC)
|
|
||||||
#pragma GCC diagnostic pop /* ignored "-Wshadow" */
|
|
||||||
#endif
|
|
||||||
|
|
||||||
struct no_mix
|
|
||||||
{
|
|
||||||
static inline std::size_t mix(std::size_t x)noexcept{return x;}
|
|
||||||
};
|
|
||||||
|
|
||||||
#if defined(SIZE_MAX)
|
|
||||||
#if ((((SIZE_MAX >> 16) >> 16) >> 16) >> 15) != 0
|
|
||||||
#define BOOST_UNORDERED_64B_ARCHITECTURE /* >64 bits assumed as 64 bits */
|
|
||||||
#endif
|
|
||||||
#elif defined(UINTPTR_MAX) /* used as proxy for std::size_t */
|
|
||||||
#if ((((UINTPTR_MAX >> 16) >> 16) >> 16) >> 15) != 0
|
|
||||||
#define BOOST_UNORDERED_64B_ARCHITECTURE
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
struct xmx_mix
|
|
||||||
{
|
|
||||||
static inline std::size_t mix(std::size_t x)noexcept
|
|
||||||
{
|
|
||||||
#if defined(BOOST_UNORDERED_64B_ARCHITECTURE)
|
|
||||||
|
|
||||||
boost::uint64_t z=x;
|
|
||||||
|
|
||||||
z^=z>> 23;
|
|
||||||
z*=0xff51afd7ed558ccdull;
|
|
||||||
z^=z>>23;
|
|
||||||
|
|
||||||
return (std::size_t)z;
|
|
||||||
|
|
||||||
#else /* 32 bits assumed */
|
|
||||||
|
|
||||||
x^=x>>18;
|
|
||||||
x*=0x56b5aaadu;
|
|
||||||
x^=x>>16;
|
|
||||||
|
|
||||||
return x;
|
|
||||||
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
#ifdef BOOST_UNORDERED_64B_ARCHITECTURE
|
|
||||||
#undef BOOST_UNORDERED_64B_ARCHITECTURE
|
|
||||||
#endif
|
|
||||||
|
|
||||||
template<typename Hash> struct is_boost_hash:std::false_type{};
|
|
||||||
template<typename Key> struct is_boost_hash<boost::hash<Key>>:std::true_type{};
|
|
||||||
|
|
||||||
template<typename Hash> struct boost_hash_key_impl{using type=void;};
|
|
||||||
template<typename Key> struct boost_hash_key_impl<boost::hash<Key>>
|
|
||||||
{
|
|
||||||
using type=Key;
|
|
||||||
};
|
|
||||||
template<typename Hash> using boost_hash_key=
|
|
||||||
typename boost_hash_key_impl<Hash>::type;
|
|
||||||
|
|
||||||
template<typename Hash>
|
|
||||||
using mixer=typename std::conditional<
|
|
||||||
is_boost_hash<Hash>::value&&(
|
|
||||||
std::is_integral<boost_hash_key<Hash>>::value||
|
|
||||||
std::is_enum<boost_hash_key<Hash>>::value||
|
|
||||||
std::is_floating_point<boost_hash_key<Hash>>::value|| // TODO: not sure about this one
|
|
||||||
std::is_pointer<boost_hash_key<Hash>>::value),
|
|
||||||
mixer_impl<Hash,xmx_mix>,
|
|
||||||
mixer_impl<Hash,no_mix>
|
|
||||||
>::type;
|
|
||||||
|
|
||||||
} /* namespace foa */
|
|
||||||
} /* namespace detail */
|
|
||||||
} /* namespace unordered */
|
|
||||||
} /* namespace boost */
|
|
||||||
|
|
||||||
#endif
|
|
75
include/boost/unordered/detail/xmx.hpp
Normal file
75
include/boost/unordered/detail/xmx.hpp
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
/* 32b/64b xmx mix function.
|
||||||
|
*
|
||||||
|
* Copyright 2022 Peter Dimov.
|
||||||
|
* Copyright 2022 Joaquin M Lopez Munoz.
|
||||||
|
* Distributed under the Boost Software License, Version 1.0.
|
||||||
|
* (See accompanying file LICENSE_1_0.txt or copy at
|
||||||
|
* http://www.boost.org/LICENSE_1_0.txt)
|
||||||
|
*
|
||||||
|
* See https://www.boost.org/libs/unordered for library home page.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef BOOST_UNORDERED_DETAIL_XMX_HPP
|
||||||
|
#define BOOST_UNORDERED_DETAIL_XMX_HPP
|
||||||
|
|
||||||
|
#include <boost/cstdint.hpp>
|
||||||
|
#include <climits>
|
||||||
|
#include <cstddef>
|
||||||
|
|
||||||
|
namespace boost{
|
||||||
|
namespace unordered{
|
||||||
|
namespace detail{
|
||||||
|
|
||||||
|
/* Bit mixer for improvement of statistical properties of hash functions.
|
||||||
|
* The implementation is different on 64bit and 32bit architectures:
|
||||||
|
*
|
||||||
|
* - 64bit: same as xmx function in
|
||||||
|
* http://jonkagstrom.com/bit-mixer-construction/index.html
|
||||||
|
* - 32bit: generated by Hash Function Prospector
|
||||||
|
* (https://github.com/skeeto/hash-prospector) and selected as the
|
||||||
|
* best overall performer in benchmarks of Boost.Unordered flat containers.
|
||||||
|
* Score assigned by Hash Prospector: 333.7934929677524
|
||||||
|
*/
|
||||||
|
|
||||||
|
#if defined(SIZE_MAX)
|
||||||
|
#if ((((SIZE_MAX >> 16) >> 16) >> 16) >> 15) != 0
|
||||||
|
#define BOOST_UNORDERED_64B_ARCHITECTURE /* >64 bits assumed as 64 bits */
|
||||||
|
#endif
|
||||||
|
#elif defined(UINTPTR_MAX) /* used as proxy for std::size_t */
|
||||||
|
#if ((((UINTPTR_MAX >> 16) >> 16) >> 16) >> 15) != 0
|
||||||
|
#define BOOST_UNORDERED_64B_ARCHITECTURE
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static inline std::size_t xmx(std::size_t x)noexcept
|
||||||
|
{
|
||||||
|
#if defined(BOOST_UNORDERED_64B_ARCHITECTURE)
|
||||||
|
|
||||||
|
boost::uint64_t z=(boost::uint64_t)x;
|
||||||
|
|
||||||
|
z^=z>>23;
|
||||||
|
z*=0xff51afd7ed558ccdull;
|
||||||
|
z^=z>>23;
|
||||||
|
|
||||||
|
return (std::size_t)z;
|
||||||
|
|
||||||
|
#else /* 32 bits assumed */
|
||||||
|
|
||||||
|
x^=x>>18;
|
||||||
|
x*=0x56b5aaadu;
|
||||||
|
x^=x>>16;
|
||||||
|
|
||||||
|
return x;
|
||||||
|
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef BOOST_UNORDERED_64B_ARCHITECTURE
|
||||||
|
#undef BOOST_UNORDERED_64B_ARCHITECTURE
|
||||||
|
#endif
|
||||||
|
|
||||||
|
} /* namespace detail */
|
||||||
|
} /* namespace unordered */
|
||||||
|
} /* namespace boost */
|
||||||
|
|
||||||
|
#endif
|
52
include/boost/unordered/hash_traits.hpp
Normal file
52
include/boost/unordered/hash_traits.hpp
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
/* Hash function characterization.
|
||||||
|
*
|
||||||
|
* Copyright 2022 Joaquin M Lopez Munoz.
|
||||||
|
* Distributed under the Boost Software License, Version 1.0.
|
||||||
|
* (See accompanying file LICENSE_1_0.txt or copy at
|
||||||
|
* http://www.boost.org/LICENSE_1_0.txt)
|
||||||
|
*
|
||||||
|
* See https://www.boost.org/libs/unordered for library home page.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef BOOST_UNORDERED_HASH_TRAITS_HPP
|
||||||
|
#define BOOST_UNORDERED_HASH_TRAITS_HPP
|
||||||
|
|
||||||
|
#include <boost/type_traits/make_void.hpp>
|
||||||
|
#include <type_traits>
|
||||||
|
|
||||||
|
namespace boost{
|
||||||
|
namespace unordered{
|
||||||
|
|
||||||
|
namespace detail{
|
||||||
|
|
||||||
|
template<typename Hash,typename=void>
|
||||||
|
struct hash_is_avalanching
|
||||||
|
{
|
||||||
|
using type=std::false_type;
|
||||||
|
};
|
||||||
|
|
||||||
|
template<typename Hash>
|
||||||
|
struct hash_is_avalanching<Hash,void_t<typename Hash::is_avalanching>>
|
||||||
|
{
|
||||||
|
using type=std::true_type;
|
||||||
|
};
|
||||||
|
|
||||||
|
} /* namespace detail */
|
||||||
|
|
||||||
|
/* Partially specializable by users for concrete hash functions when
|
||||||
|
* actual characterization differs from default.
|
||||||
|
*/
|
||||||
|
|
||||||
|
template<typename Hash>
|
||||||
|
struct hash_traits
|
||||||
|
{
|
||||||
|
/* std::true_type if the type Hash::is_avalanching is present,
|
||||||
|
* std::false_type otherwise.
|
||||||
|
*/
|
||||||
|
using is_avalanching=typename detail::hash_is_avalanching<Hash>::type;
|
||||||
|
};
|
||||||
|
|
||||||
|
} /* namespace unordered */
|
||||||
|
} /* namespace boost */
|
||||||
|
|
||||||
|
#endif
|
Reference in New Issue
Block a user