Removes randomness in sequential map test case
[libcds.git] / cds / opt / hash.h
index 6d5a88d78c2a8716d5b5fbadb7cbdffe24135569..2949b506bf15078e9bba897a8b441030a0980baf 100644 (file)
@@ -1,11 +1,39 @@
-//$$CDS-header$$
+/*
+    This file is a part of libcds - Concurrent Data Structures library
 
-#ifndef __CDS_OPT_HASH_H
-#define __CDS_OPT_HASH_H
+    (C) Copyright Maxim Khizhinsky (libcds.dev@gmail.com) 2006-2017
+
+    Source code repo: http://github.com/khizmax/libcds/
+    Download: http://sourceforge.net/projects/libcds/files/
+
+    Redistribution and use in source and binary forms, with or without
+    modification, are permitted provided that the following conditions are met:
+
+    * Redistributions of source code must retain the above copyright notice, this
+      list of conditions and the following disclaimer.
+
+    * Redistributions in binary form must reproduce the above copyright notice,
+      this list of conditions and the following disclaimer in the documentation
+      and/or other materials provided with the distribution.
+
+    THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+    AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+    IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+    DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
+    FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+    DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
+    SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
+    CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
+    OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+    OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+*/
+
+#ifndef CDSLIB_OPT_HASH_H
+#define CDSLIB_OPT_HASH_H
 
 #include <tuple>
+#include <functional>
 #include <cds/opt/options.h>
-#include <cds/details/hash_functor_selector.h>
 
 namespace cds { namespace opt {
 
@@ -15,8 +43,6 @@ namespace cds { namespace opt {
 
         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.
-        If standard C++ library of the compiler you use does not provide TR1 implementation
-        the \p cds library automatically selects <tt>boost::hash</tt>.
     */
     template <typename Functor>
     struct hash {
@@ -30,23 +56,22 @@ namespace cds { namespace opt {
 
     namespace v {
         //@cond
-        using cds::details::hash;
+        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> or <tt> boost::hash<Q> </tt>
-            depending of compiler you use.
+            Otherwise, the result is <tt> std::hash<Q> </tt>.
 
-            Note that default hash function like <tt> std::hash<Q> </tt> or <tt> boost::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
+            typedef Hash    type;   ///< resulting implementation of hash functor
         };
 
         template <>
@@ -56,14 +81,13 @@ namespace cds { namespace opt {
                 template <typename Q>
                 size_t operator()( Q const& key ) const
                 {
-                    return hash<Q>()( key );
+                    return std::hash<Q>()( key );
                 }
             };
         };
         //@endcond
     }   // namespace v
 
-#ifdef CDS_CXX11_VARIADIC_TEMPLATE_SUPPORT
     //@cond
     namespace details {
         template <class> struct hash_list;
@@ -82,14 +106,12 @@ namespace cds { namespace opt {
             hash_list( hash_tuple_type const& t)
                 : hash_tuple( t )
             {}
-#       ifdef CDS_MOVE_SEMANTICS_SUPPORT
             hash_list( hash_tuple_type&& t)
-                : hash_tuple( std::forward<hash_tuple_type>(t) )
+                : hash_tuple( std::forward<hash_tuple_type>(t))
             {}
-#       endif
 
             template <size_t I, typename T>
-            typename std::enable_if< (I == sizeof...(Functors)) >::type apply( size_t * dest, T const& v ) const
+            typename std::enable_if< (I == sizeof...(Functors)) >::type apply( size_t * /*dest*/, T const& /*v*/ ) const
             {}
 
             template <size_t I, typename T>
@@ -108,21 +130,19 @@ namespace cds { namespace opt {
     } // 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
-//#if CDS_COMPILER != CDS_COMPILER_INTEL
-    // Intel C++ compiler does not support
     template <typename Functor> struct hash< std::tuple<Functor> >;
-//#endif
     //@endcond
 
-    /// Multi-functor hash option setter - specialization for std::tuple
+    /// Multi-functor hash option setter - specialization for \p std::tuple
     template <typename... Functors>
     struct hash< std::tuple<Functors...> >
     {
-//#   if CDS_COMPILER == CDS_COMPILER_INTEL
-        //static_assert( sizeof...(Functors) > 1, "At least, two functors must be provided. Single functor is not supported" );
-//#   endif
         //@cond
         template <typename Base> struct pack: public Base
         {
@@ -131,400 +151,6 @@ namespace cds { namespace opt {
         //@endcond
     };
 
-#else   // no variadic template support
-    namespace details {
-        template <typename T> struct hash_list;
-        template <typename F1, typename F2>
-        struct hash_list< std::tuple<F1, F2> >
-        {
-            static size_t const size = 2;
-            typedef size_t values[size];
-            typedef std::tuple<F1, F2>  hash_tuple_type;
-
-            hash_tuple_type hash_tuple;
-
-            hash_list()
-            {}
-            hash_list( hash_tuple_type const& t)
-                : hash_tuple( t )
-            {}
-#       ifdef CDS_MOVE_SEMANTICS_SUPPORT
-            hash_list( hash_tuple_type&& t)
-                : hash_tuple( t )
-            {}
-#       endif
-
-            template <typename T>
-            void operator()( size_t * dest, T const& v ) const
-            {
-                dest[0] = std::get<0>( hash_tuple )( v );
-                dest[1] = std::get<1>( hash_tuple )( v );
-            }
-        };
-
-        template <typename F1, typename F2, typename F3>
-        struct hash_list< std::tuple<F1, F2, F3> >
-        {
-            static size_t const size = 3;
-            typedef size_t values[size];
-            typedef std::tuple<F1, F2, F3> hash_tuple_type;
-
-            hash_tuple_type hash_tuple;
-
-            hash_list()
-            {}
-            hash_list( hash_tuple_type const& t)
-                : hash_tuple( t )
-            {}
-#       ifdef CDS_MOVE_SEMANTICS_SUPPORT
-            hash_list( hash_tuple_type&& t)
-                : hash_tuple( t )
-            {}
-#       endif
-
-            template <typename T>
-            void operator()( size_t * dest, T const& v ) const
-            {
-                dest[0] = std::get<0>( hash_tuple )( v );
-                dest[1] = std::get<1>( hash_tuple )( v );
-                dest[2] = std::get<2>( hash_tuple )( v );
-            }
-        };
-
-        template <typename F1, typename F2, typename F3, typename F4>
-        struct hash_list< std::tuple<F1, F2, F3, F4> >
-        {
-            static size_t const size = 4;
-            typedef size_t values[size];
-            typedef std::tuple<F1, F2, F3, F4> hash_tuple_type;
-
-            hash_tuple_type hash_tuple;
-
-            hash_list()
-            {}
-            hash_list( hash_tuple_type const& t)
-                : hash_tuple( t )
-            {}
-#       ifdef CDS_MOVE_SEMANTICS_SUPPORT
-            hash_list( hash_tuple_type&& t)
-                : hash_tuple( t )
-            {}
-#       endif
-
-            template <typename T>
-            void operator()( size_t * dest, T const& v ) const
-            {
-                dest[0] = std::get<0>( hash_tuple )( v );
-                dest[1] = std::get<1>( hash_tuple )( v );
-                dest[2] = std::get<2>( hash_tuple )( v );
-                dest[3] = std::get<3>( hash_tuple )( v );
-            }
-        };
-
-        template <typename F1, typename F2, typename F3, typename F4, typename F5>
-        struct hash_list< std::tuple<F1, F2, F3, F4, F5> >
-        {
-            static size_t const size = 5;
-            typedef size_t values[size];
-            typedef std::tuple<F1, F2, F3, F4, F5> hash_tuple_type;
-
-            hash_tuple_type hash_tuple;
-
-            hash_list()
-            {}
-            hash_list( hash_tuple_type const& t)
-                : hash_tuple( t )
-            {}
-#       ifdef CDS_MOVE_SEMANTICS_SUPPORT
-            hash_list( hash_tuple_type&& t)
-                : hash_tuple( t )
-            {}
-#       endif
-
-            template <typename T>
-            void operator()( size_t * dest, T const& v ) const
-            {
-                dest[0] = std::get<0>( hash_tuple )( v );
-                dest[1] = std::get<1>( hash_tuple )( v );
-                dest[2] = std::get<2>( hash_tuple )( v );
-                dest[3] = std::get<3>( hash_tuple )( v );
-                dest[4] = std::get<4>( hash_tuple )( v );
-            }
-        };
-
-        template <typename F1, typename F2, typename F3, typename F4, typename F5, typename F6>
-        struct hash_list< std::tuple<F1, F2, F3, F4, F5, F6> >
-        {
-            static size_t const size = 6;
-            typedef size_t values[size];
-            typedef std::tuple<F1, F2, F3, F4, F5, F6> hash_tuple_type;
-
-            hash_tuple_type hash_tuple;
-
-            hash_list()
-            {}
-            hash_list( hash_tuple_type const& t)
-                : hash_tuple( t )
-            {}
-#       ifdef CDS_MOVE_SEMANTICS_SUPPORT
-            hash_list( hash_tuple_type&& t)
-                : hash_tuple( t )
-            {}
-#       endif
-
-            template <typename T>
-            void operator()( size_t * dest, T const& v ) const
-            {
-                dest[0] = std::get<0>( hash_tuple )( v );
-                dest[1] = std::get<1>( hash_tuple )( v );
-                dest[2] = std::get<2>( hash_tuple )( v );
-                dest[3] = std::get<3>( hash_tuple )( v );
-                dest[4] = std::get<4>( hash_tuple )( v );
-                dest[5] = std::get<5>( hash_tuple )( v );
-            }
-        };
-
-        template <typename F1, typename F2, typename F3, typename F4, typename F5, typename F6, typename F7>
-        struct hash_list< std::tuple<F1, F2, F3, F4, F5, F6, F7> >
-        {
-            static size_t const size = 7;
-            typedef size_t values[size];
-            typedef std::tuple<F1, F2, F3, F4, F5, F6, F7> hash_tuple_type;
-
-            hash_tuple_type hash_tuple;
-
-            hash_list()
-            {}
-            hash_list( hash_tuple_type const& t)
-                : hash_tuple( t )
-            {}
-#       ifdef CDS_MOVE_SEMANTICS_SUPPORT
-            hash_list( hash_tuple_type&& t)
-                : hash_tuple( t )
-            {}
-#       endif
-
-            template <typename T>
-            void operator()( size_t * dest, T const& v ) const
-            {
-                dest[0] = std::get<0>( hash_tuple )( v );
-                dest[1] = std::get<1>( hash_tuple )( v );
-                dest[2] = std::get<2>( hash_tuple )( v );
-                dest[3] = std::get<3>( hash_tuple )( v );
-                dest[4] = std::get<4>( hash_tuple )( v );
-                dest[5] = std::get<5>( hash_tuple )( v );
-                dest[6] = std::get<6>( hash_tuple )( v );
-            }
-        };
-
-        template <typename F1, typename F2, typename F3, typename F4, typename F5, typename F6, typename F7, typename F8>
-        struct hash_list< std::tuple<F1, F2, F3, F4, F5, F6, F7, F8> >
-        {
-            static size_t const size = 8;
-            typedef size_t values[size];
-            typedef std::tuple<F1, F2, F3, F4, F5, F6, F7, F8> hash_tuple_type;
-
-            hash_tuple_type hash_tuple;
-
-            hash_list()
-            {}
-            hash_list( hash_tuple_type const& t)
-                : hash_tuple( t )
-            {}
-#       ifdef CDS_MOVE_SEMANTICS_SUPPORT
-            hash_list( hash_tuple_type&& t)
-                : hash_tuple( t )
-            {}
-#       endif
-
-            template <typename T>
-            void operator()( size_t * dest, T const& v ) const
-            {
-                dest[0] = std::get<0>( hash_tuple )( v );
-                dest[1] = std::get<1>( hash_tuple )( v );
-                dest[2] = std::get<2>( hash_tuple )( v );
-                dest[3] = std::get<3>( hash_tuple )( v );
-                dest[4] = std::get<4>( hash_tuple )( v );
-                dest[5] = std::get<5>( hash_tuple )( v );
-                dest[6] = std::get<6>( hash_tuple )( v );
-                dest[7] = std::get<7>( hash_tuple )( v );
-            }
-        };
-
-#if !((CDS_COMPILER == CDS_COMPILER_MSVC || CDS_COMPILER == CDS_COMPILER_INTEL) && _MSC_VER == 1700)
-        // MSVC 11: max count of argument is 8
-
-        template <typename F1, typename F2, typename F3, typename F4, typename F5, typename F6, typename F7, typename F8, typename F9>
-        struct hash_list< std::tuple<F1, F2, F3, F4, F5, F6, F7, F8, F9> >
-        {
-            static size_t const size = 9;
-            typedef size_t values[size];
-            typedef std::tuple<F1, F2, F3, F4, F5, F6, F7, F8, F9> hash_tuple_type;
-
-            hash_tuple_type hash_tuple;
-
-            hash_list()
-            {}
-            hash_list( hash_tuple_type const& t)
-                : hash_tuple( t )
-            {}
-#       ifdef CDS_MOVE_SEMANTICS_SUPPORT
-            hash_list( hash_tuple_type&& t)
-                : hash_tuple( t )
-            {}
-#       endif
-
-            template <typename T>
-            void operator()( size_t * dest, T const& v ) const
-            {
-                dest[0] = std::get<0>( hash_tuple )( v );
-                dest[1] = std::get<1>( hash_tuple )( v );
-                dest[2] = std::get<2>( hash_tuple )( v );
-                dest[3] = std::get<3>( hash_tuple )( v );
-                dest[4] = std::get<4>( hash_tuple )( v );
-                dest[5] = std::get<5>( hash_tuple )( v );
-                dest[6] = std::get<6>( hash_tuple )( v );
-                dest[7] = std::get<7>( hash_tuple )( v );
-                dest[8] = std::get<8>( hash_tuple )( v );
-            }
-        };
-
-        template <typename F1, typename F2, typename F3, typename F4, typename F5, typename F6, typename F7, typename F8, typename F9,
-                  typename F10>
-        struct hash_list< std::tuple<F1, F2, F3, F4, F5, F6, F7, F8, F9, F10> >
-        {
-            static size_t const size = 10;
-            typedef size_t values[size];
-            typedef std::tuple<F1, F2, F3, F4, F5, F6, F7, F8, F9, F10> hash_tuple_type;
-
-            hash_tuple_type hash_tuple;
-
-            hash_list()
-            {}
-            hash_list( hash_tuple_type const& t)
-                : hash_tuple( t )
-            {}
-#       ifdef CDS_MOVE_SEMANTICS_SUPPORT
-            hash_list( hash_tuple_type&& t)
-                : hash_tuple( t )
-            {}
-#       endif
-
-            template <typename T>
-            void operator()( size_t * dest, T const& v ) const
-            {
-                dest[0] = std::get<0>( hash_tuple )( v );
-                dest[1] = std::get<1>( hash_tuple )( v );
-                dest[2] = std::get<2>( hash_tuple )( v );
-                dest[3] = std::get<3>( hash_tuple )( v );
-                dest[4] = std::get<4>( hash_tuple )( v );
-                dest[5] = std::get<5>( hash_tuple )( v );
-                dest[6] = std::get<6>( hash_tuple )( v );
-                dest[7] = std::get<7>( hash_tuple )( v );
-                dest[8] = std::get<8>( hash_tuple )( v );
-                dest[9] = std::get<9>( hash_tuple )( v );
-            }
-        };
-#endif
-    } // namespace details
-
-    template< typename F1, typename F2 >
-    struct hash< std::tuple< F1, F2 > >
-    {
-        //@cond
-        template <typename Base> struct pack: public Base
-        {
-            typedef details::hash_list< std::tuple<F1, F2> >  hash;
-        };
-        //@endcond
-    };
-    template< typename F1, typename F2, typename F3 >
-    struct hash< std::tuple< F1, F2, F3 > >
-    {
-        //@cond
-        template <typename Base> struct pack: public Base
-        {
-            typedef details::hash_list< std::tuple<F1, F2, F3> >  hash;
-        };
-        //@endcond
-    };
-    template< typename F1, typename F2, typename F3, typename F4 >
-    struct hash< std::tuple< F1, F2, F3, F4 > >
-    {
-        //@cond
-        template <typename Base> struct pack: public Base
-        {
-            typedef details::hash_list< std::tuple<F1, F2, F3, F4> >  hash;
-        };
-        //@endcond
-    };
-    template< typename F1, typename F2, typename F3, typename F4, typename F5 >
-    struct hash< std::tuple< F1, F2, F3, F4, F5 > >
-    {
-        //@cond
-        template <typename Base> struct pack: public Base
-        {
-            typedef details::hash_list< std::tuple<F1, F2, F3, F4, F5> >  hash;
-        };
-        //@endcond
-    };
-    template< typename F1, typename F2, typename F3, typename F4, typename F5, typename F6 >
-    struct hash< std::tuple< F1, F2, F3, F4, F5, F6 > >
-    {
-        //@cond
-        template <typename Base> struct pack: public Base
-        {
-            typedef details::hash_list< std::tuple<F1, F2, F3, F4, F5, F6> >  hash;
-        };
-        //@endcond
-    };
-    template< typename F1, typename F2, typename F3, typename F4, typename F5, typename F6, typename F7 >
-    struct hash< std::tuple< F1, F2, F3, F4, F5, F6, F7 > >
-    {
-        //@cond
-        template <typename Base> struct pack: public Base
-        {
-            typedef details::hash_list< std::tuple<F1, F2, F3, F4, F5, F6, F7> >  hash;
-        };
-        //@endcond
-    };
-    template< typename F1, typename F2, typename F3, typename F4, typename F5, typename F6, typename F7, typename F8 >
-    struct hash< std::tuple< F1, F2, F3, F4, F5, F6, F7, F8 > >
-    {
-        //@cond
-        template <typename Base> struct pack: public Base
-        {
-            typedef details::hash_list< std::tuple<F1, F2, F3, F4, F5, F6, F7, F8> >  hash;
-        };
-        //@endcond
-    };
-
-#if !((CDS_COMPILER == CDS_COMPILER_MSVC || CDS_COMPILER == CDS_COMPILER_INTEL) && _MSC_VER == 1700)
-    // MSVC 11: max count of argument is 8
-
-    template< typename F1, typename F2, typename F3, typename F4, typename F5, typename F6, typename F7, typename F8, typename F9 >
-    struct hash< std::tuple< F1, F2, F3, F4, F5, F6, F7, F8, F9 > >
-    {
-        //@cond
-        template <typename Base> struct pack: public Base
-        {
-            typedef details::hash_list< std::tuple<F1, F2, F3, F4, F5, F6, F7, F8, F9> >  hash;
-        };
-        //@endcond
-    };
-    template< typename F1, typename F2, typename F3, typename F4, typename F5, typename F6, typename F7, typename F8, typename F9,
-              typename F10 >
-    struct hash< std::tuple< F1, F2, F3, F4, F5, F6, F7, F8, F9, F10 > >
-    {
-        //@cond
-        template <typename Base> struct pack: public Base
-        {
-            typedef details::hash_list< std::tuple<F1, F2, F3, F4, F5, F6, F7, F8, F9, F10> >  hash;
-        };
-        //@endcond
-    };
-#endif  // !MSVC11
-#endif  // #ifdef CDS_CXX11_VARIADIC_TEMPLATE_SUPPORT
 
     //@cond
     namespace details {
@@ -545,11 +171,9 @@ namespace cds { namespace opt {
             hash_list_wrapper( hash_tuple_type const& t)
                 : m_wrappedList( t )
             {}
-#       ifdef CDS_MOVE_SEMANTICS_SUPPORT
             hash_list_wrapper( hash_tuple_type&& t)
-                : m_wrappedList( std::forward<hash_tuple_type>(t) )
+                : m_wrappedList( std::forward<hash_tuple_type>(t))
             {}
-#       endif
 
             void operator()( size_t * dest, wrapped_type const& what ) const
             {
@@ -568,4 +192,4 @@ namespace cds { namespace opt {
 
 }} // namespace cds::opt
 
-#endif // #ifndef __CDS_OPT_HASH_H
+#endif // #ifndef CDSLIB_OPT_HASH_H