8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-22 22:43:03 +01:00
firebird-mirror/extern/libcds/cds/opt/hash.h
2022-10-08 20:46:39 +03:00

171 lines
5.2 KiB
C++

// Copyright (c) 2006-2018 Maxim Khizhinsky
//
// Distributed under the Boost Software License, Version 1.0. (See accompanying
// file LICENSE or copy at http://www.boost.org/LICENSE_1_0.txt)
#ifndef CDSLIB_OPT_HASH_H
#define CDSLIB_OPT_HASH_H
#include <tuple>
#include <functional>
#include <cds/opt/options.h>
namespace cds { namespace opt {
/// [type-option] Option setter for a hash function
/**
This option setter specifies hash functor used in unordered containers.
The default value of template argument \p Functor is \p cds::opt::v::hash
that is synonym for <tt>std::hash</tt> implementation of standard library.
*/
template <typename Functor>
struct hash {
//@cond
template <typename Base> struct pack: public Base
{
typedef Functor hash;
};
//@endcond
};
namespace v {
//@cond
using std::hash;
/// Metafunction selecting default hash implementation
/**
The metafunction selects appropriate hash functor implementation.
If \p Hash is not equal to opt::none, then result of metafunction is \p Hash.
Otherwise, the result is <tt> std::hash<Q> </tt>.
Note that default hash function like <tt> std::hash<Q> </tt>
is generally not suitable for complex type \p Q and its derivatives.
You should manually provide particular hash functor for such types.
*/
template <typename Hash>
struct hash_selector
{
typedef Hash type; ///< resulting implementation of hash functor
};
template <>
struct hash_selector<opt::none>
{
struct type {
template <typename Q>
size_t operator()( Q const& key ) const
{
return std::hash<Q>()( key );
}
};
};
//@endcond
} // namespace v
//@cond
namespace details {
template <class> struct hash_list;
template <typename... Functors>
struct hash_list< std::tuple<Functors...> >
{
static size_t const size = sizeof...(Functors);
typedef size_t values[size];
typedef std::tuple<Functors...> hash_tuple_type;
hash_tuple_type hash_tuple;
hash_list()
{}
hash_list( hash_tuple_type const& t)
: hash_tuple( t )
{}
hash_list( hash_tuple_type&& t)
: hash_tuple( std::forward<hash_tuple_type>(t))
{}
template <size_t I, typename T>
typename std::enable_if< (I == sizeof...(Functors)) >::type apply( size_t * /*dest*/, T const& /*v*/ ) const
{}
template <size_t I, typename T>
typename std::enable_if< (I < sizeof...(Functors)) >::type apply( size_t * dest, T const& v ) const
{
dest[I] = std::get<I>( hash_tuple )( v );
apply<I+1>( dest, v );
}
template <typename T>
void operator()( size_t * dest, T const& v ) const
{
apply<0>( dest, v );
}
};
} // namespace details
//@endcond
/// Declare tuple for hash functors \p Functors
template <typename... Functors>
using hash_tuple = details::hash_list< std::tuple< Functors... >>;
//@cond
// At least, two functors must be provided. Single functor is not supported
template <typename Functor> struct hash< std::tuple<Functor> >;
//@endcond
/// Multi-functor hash option setter - specialization for \p std::tuple
template <typename... Functors>
struct hash< std::tuple<Functors...> >
{
//@cond
template <typename Base> struct pack: public Base
{
typedef details::hash_list< std::tuple<Functors...> > hash;
};
//@endcond
};
//@cond
namespace details {
template <class HashList, typename WrappedType, typename Wrapper>
struct hash_list_wrapper {
typedef HashList hash_list;
typedef WrappedType wrapped_type;
typedef Wrapper wrapper_type;
typedef typename hash_list::hash_tuple_type hash_tuple_type;
static size_t const size = hash_list::size;
hash_list m_wrappedList;
hash_list_wrapper()
{}
hash_list_wrapper( hash_tuple_type const& t)
: m_wrappedList( t )
{}
hash_list_wrapper( hash_tuple_type&& t)
: m_wrappedList( std::forward<hash_tuple_type>(t))
{}
void operator()( size_t * dest, wrapped_type const& what ) const
{
m_wrappedList( dest, wrapper_type()( what ));
}
template <typename Q>
void operator()( size_t * dest, Q const& what) const
{
m_wrappedList( dest, what );
}
};
} // namespace details
//@endcond
}} // namespace cds::opt
#endif // #ifndef CDSLIB_OPT_HASH_H