319 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
		
		
			
		
	
	
			319 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
|   | ////////////////////////////////////////////////////////////////////////////// | ||
|  | // | ||
|  | // (C) Copyright Ion Gaztanaga 2012-2012. | ||
|  | // 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 http://www.boost.org/libs/move for documentation. | ||
|  | // | ||
|  | ////////////////////////////////////////////////////////////////////////////// | ||
|  | 
 | ||
|  | //! \file | ||
|  | //! This header defines core utilities to ease the development | ||
|  | //! of move-aware functions. This header minimizes dependencies | ||
|  | //! from other libraries. | ||
|  | 
 | ||
|  | #ifndef BOOST_MOVE_MOVE_UTILITY_CORE_HPP | ||
|  | #define BOOST_MOVE_MOVE_UTILITY_CORE_HPP | ||
|  | 
 | ||
|  | #ifndef BOOST_CONFIG_HPP | ||
|  | #  include <boost/config.hpp> | ||
|  | #endif | ||
|  | # | ||
|  | #if defined(BOOST_HAS_PRAGMA_ONCE) | ||
|  | #  pragma once | ||
|  | #endif | ||
|  | 
 | ||
|  | #include <boost/move/detail/config_begin.hpp> | ||
|  | #include <boost/move/detail/workaround.hpp>  //forceinline | ||
|  | #include <boost/move/core.hpp> | ||
|  | #include <boost/move/detail/meta_utils.hpp> | ||
|  | #include <boost/static_assert.hpp> | ||
|  | 
 | ||
|  | #if defined(BOOST_NO_CXX11_RVALUE_REFERENCES) && !defined(BOOST_MOVE_DOXYGEN_INVOKED) | ||
|  | 
 | ||
|  |    namespace boost { | ||
|  | 
 | ||
|  |    template<class T> | ||
|  |    struct enable_move_utility_emulation | ||
|  |    { | ||
|  |       static const bool value = true; | ||
|  |    }; | ||
|  |      | ||
|  |    ////////////////////////////////////////////////////////////////////////////// | ||
|  |    // | ||
|  |    //                            move() | ||
|  |    // | ||
|  |    ////////////////////////////////////////////////////////////////////////////// | ||
|  | 
 | ||
|  |    template <class T> | ||
|  |    BOOST_MOVE_FORCEINLINE typename ::boost::move_detail::enable_if_and | ||
|  |       < T & | ||
|  |       , enable_move_utility_emulation<T> | ||
|  |       , has_move_emulation_disabled<T> | ||
|  |       >::type | ||
|  |          move(T& x) BOOST_NOEXCEPT | ||
|  |    { | ||
|  |       return x; | ||
|  |    } | ||
|  | 
 | ||
|  |    template <class T> | ||
|  |    BOOST_MOVE_FORCEINLINE typename ::boost::move_detail::enable_if_and | ||
|  |       < rv<T>& | ||
|  |       , enable_move_utility_emulation<T> | ||
|  |       , has_move_emulation_enabled<T> | ||
|  |       >::type | ||
|  |          move(T& x) BOOST_NOEXCEPT | ||
|  |    { | ||
|  |       return *BOOST_MOVE_TO_RV_CAST(::boost::rv<T>*, ::boost::move_detail::addressof(x) ); | ||
|  |    } | ||
|  | 
 | ||
|  |    template <class T> | ||
|  |    BOOST_MOVE_FORCEINLINE typename ::boost::move_detail::enable_if_and | ||
|  |       < rv<T>& | ||
|  |       , enable_move_utility_emulation<T> | ||
|  |       , has_move_emulation_enabled<T> | ||
|  |       >::type | ||
|  |          move(rv<T>& x) BOOST_NOEXCEPT | ||
|  |    { | ||
|  |       return x; | ||
|  |    } | ||
|  | 
 | ||
|  |    ////////////////////////////////////////////////////////////////////////////// | ||
|  |    // | ||
|  |    //                            forward() | ||
|  |    // | ||
|  |    ////////////////////////////////////////////////////////////////////////////// | ||
|  | 
 | ||
|  |    template <class T> | ||
|  |    BOOST_MOVE_FORCEINLINE typename ::boost::move_detail::enable_if_and | ||
|  |       < T & | ||
|  |       , enable_move_utility_emulation<T> | ||
|  |       , ::boost::move_detail::is_rv<T> | ||
|  |       >::type | ||
|  |          forward(const typename ::boost::move_detail::identity<T>::type &x) BOOST_NOEXCEPT | ||
|  |    { | ||
|  |       return const_cast<T&>(x); | ||
|  |    } | ||
|  | 
 | ||
|  |    template <class T> | ||
|  |    BOOST_MOVE_FORCEINLINE typename ::boost::move_detail::enable_if_and | ||
|  |       < const T & | ||
|  |       , enable_move_utility_emulation<T> | ||
|  |       , ::boost::move_detail::is_not_rv<T> | ||
|  |       >::type | ||
|  |          forward(const typename ::boost::move_detail::identity<T>::type &x) BOOST_NOEXCEPT | ||
|  |    { | ||
|  |       return x; | ||
|  |    } | ||
|  | 
 | ||
|  |    ////////////////////////////////////////////////////////////////////////////// | ||
|  |    // | ||
|  |    //                        move_if_not_lvalue_reference() | ||
|  |    // | ||
|  |    ////////////////////////////////////////////////////////////////////////////// | ||
|  | 
 | ||
|  |    template <class T> | ||
|  |    BOOST_MOVE_FORCEINLINE typename ::boost::move_detail::enable_if_and | ||
|  |       < T & | ||
|  |       , enable_move_utility_emulation<T> | ||
|  |       , ::boost::move_detail::is_rv<T> | ||
|  |       >::type | ||
|  |          move_if_not_lvalue_reference(const typename ::boost::move_detail::identity<T>::type &x) BOOST_NOEXCEPT | ||
|  |    { | ||
|  |       return const_cast<T&>(x); | ||
|  |    } | ||
|  | 
 | ||
|  |    template <class T> | ||
|  |    BOOST_MOVE_FORCEINLINE typename ::boost::move_detail::enable_if_and | ||
|  |       < typename ::boost::move_detail::add_lvalue_reference<T>::type | ||
|  |       , enable_move_utility_emulation<T> | ||
|  |       , ::boost::move_detail::is_not_rv<T> | ||
|  |       , ::boost::move_detail::or_ | ||
|  |          < ::boost::move_detail::is_lvalue_reference<T> | ||
|  |          , has_move_emulation_disabled<T> | ||
|  |          > | ||
|  |       >::type | ||
|  |          move_if_not_lvalue_reference(typename ::boost::move_detail::remove_reference<T>::type &x) BOOST_NOEXCEPT | ||
|  |    { | ||
|  |       return x; | ||
|  |    } | ||
|  | 
 | ||
|  |    template <class T> | ||
|  |    BOOST_MOVE_FORCEINLINE typename ::boost::move_detail::enable_if_and | ||
|  |       < rv<T>& | ||
|  |       , enable_move_utility_emulation<T> | ||
|  |       , ::boost::move_detail::is_not_rv<T> | ||
|  |       , ::boost::move_detail::and_ | ||
|  |          < ::boost::move_detail::not_< ::boost::move_detail::is_lvalue_reference<T> > | ||
|  |          , has_move_emulation_enabled<T> | ||
|  |          > | ||
|  |       >::type | ||
|  |          move_if_not_lvalue_reference(typename ::boost::move_detail::remove_reference<T>::type &x) BOOST_NOEXCEPT | ||
|  |    { | ||
|  |       return move(x); | ||
|  |    } | ||
|  | 
 | ||
|  |    }  //namespace boost | ||
|  | 
 | ||
|  | #else    //#if defined(BOOST_NO_CXX11_RVALUE_REFERENCES) && !defined(BOOST_MOVE_DOXYGEN_INVOKED) | ||
|  | 
 | ||
|  |    #if defined(BOOST_MOVE_USE_STANDARD_LIBRARY_MOVE) | ||
|  |       #include <utility> | ||
|  | 
 | ||
|  |       namespace boost{ | ||
|  | 
 | ||
|  |       using ::std::move; | ||
|  |       using ::std::forward; | ||
|  | 
 | ||
|  |       }  //namespace boost | ||
|  | 
 | ||
|  |    #else //!BOOST_MOVE_USE_STANDARD_LIBRARY_MOVE | ||
|  | 
 | ||
|  |       namespace boost { | ||
|  | 
 | ||
|  |       //! This trait's internal boolean `value` is false in compilers with rvalue references | ||
|  |       //! and true in compilers without rvalue references. | ||
|  |       //! | ||
|  |       //! A user can specialize this trait for a type T to false to SFINAE out `move` and `forward` | ||
|  |       //! so that the user can define a different move emulation for that type in namespace boost | ||
|  |       //! (e.g. another Boost library for its types) and avoid any overload ambiguity. | ||
|  |       template<class T> | ||
|  |       struct enable_move_utility_emulation | ||
|  |       { | ||
|  |          static const bool value = false; | ||
|  |       }; | ||
|  | 
 | ||
|  |       ////////////////////////////////////////////////////////////////////////////// | ||
|  |       // | ||
|  |       //                                  move | ||
|  |       // | ||
|  |       ////////////////////////////////////////////////////////////////////////////// | ||
|  | 
 | ||
|  |       #if defined(BOOST_MOVE_DOXYGEN_INVOKED) | ||
|  |          //! This function provides a way to convert a reference into a rvalue reference | ||
|  |          //! in compilers with rvalue references. For other compilers if `T` is Boost.Move | ||
|  |          //! enabled type then it converts `T&` into <tt>::boost::rv<T> &</tt> so that | ||
|  |          //! move emulation is activated, else it returns `T &`. | ||
|  |          template <class T> | ||
|  |          rvalue_reference move(input_reference) noexcept; | ||
|  | 
 | ||
|  |       #elif defined(BOOST_MOVE_OLD_RVALUE_REF_BINDING_RULES) | ||
|  | 
 | ||
|  |          //Old move approach, lvalues could bind to rvalue references | ||
|  |          template <class T> | ||
|  |          BOOST_MOVE_FORCEINLINE typename ::boost::move_detail::remove_reference<T>::type && move(T&& t) BOOST_NOEXCEPT | ||
|  |          {  return t;   } | ||
|  | 
 | ||
|  |       #else //BOOST_MOVE_OLD_RVALUE_REF_BINDING_RULES | ||
|  | 
 | ||
|  |          template <class T> | ||
|  |          BOOST_MOVE_FORCEINLINE typename ::boost::move_detail::remove_reference<T>::type && move(T&& t) BOOST_NOEXCEPT | ||
|  |          { return static_cast<typename ::boost::move_detail::remove_reference<T>::type &&>(t); } | ||
|  | 
 | ||
|  |       #endif   //BOOST_MOVE_OLD_RVALUE_REF_BINDING_RULES | ||
|  | 
 | ||
|  |       ////////////////////////////////////////////////////////////////////////////// | ||
|  |       // | ||
|  |       //                                  forward | ||
|  |       // | ||
|  |       ////////////////////////////////////////////////////////////////////////////// | ||
|  | 
 | ||
|  | 
 | ||
|  |       #if defined(BOOST_MOVE_DOXYGEN_INVOKED) | ||
|  |          //! This function provides limited form of forwarding that is usually enough for | ||
|  |          //! in-place construction and avoids the exponential overloading for | ||
|  |          //! achieve the limited forwarding in C++03. | ||
|  |          //! | ||
|  |          //! For compilers with rvalue references this function provides perfect forwarding. | ||
|  |          //! | ||
|  |          //! Otherwise: | ||
|  |          //! * If input_reference binds to const ::boost::rv<T> & then it output_reference is | ||
|  |          //!   ::boost::rv<T> & | ||
|  |          //! | ||
|  |          //! * Else, output_reference is equal to input_reference. | ||
|  |          template <class T> output_reference forward(input_reference) noexcept; | ||
|  |       #elif defined(BOOST_MOVE_OLD_RVALUE_REF_BINDING_RULES) | ||
|  | 
 | ||
|  |          //Old move approach, lvalues could bind to rvalue references | ||
|  | 
 | ||
|  |          template <class T> | ||
|  |          BOOST_MOVE_FORCEINLINE T&& forward(typename ::boost::move_detail::identity<T>::type&& t) BOOST_NOEXCEPT | ||
|  |          {  return t;   } | ||
|  | 
 | ||
|  |       #else //Old move | ||
|  | 
 | ||
|  |          template <class T> | ||
|  |          BOOST_MOVE_FORCEINLINE T&& forward(typename ::boost::move_detail::remove_reference<T>::type& t) BOOST_NOEXCEPT | ||
|  |          {  return static_cast<T&&>(t);   } | ||
|  | 
 | ||
|  |          template <class T> | ||
|  |          BOOST_MOVE_FORCEINLINE T&& forward(typename ::boost::move_detail::remove_reference<T>::type&& t) BOOST_NOEXCEPT | ||
|  |          { | ||
|  |             //"boost::forward<T> error: 'T' is a lvalue reference, can't forward as rvalue."; | ||
|  |             BOOST_STATIC_ASSERT(!boost::move_detail::is_lvalue_reference<T>::value); | ||
|  |             return static_cast<T&&>(t); | ||
|  |          } | ||
|  | 
 | ||
|  |       #endif   //BOOST_MOVE_DOXYGEN_INVOKED | ||
|  | 
 | ||
|  |       ////////////////////////////////////////////////////////////////////////////// | ||
|  |       // | ||
|  |       //                         move_if_not_lvalue_reference | ||
|  |       // | ||
|  |       ////////////////////////////////////////////////////////////////////////////// | ||
|  | 
 | ||
|  | 
 | ||
|  |       #if defined(BOOST_MOVE_DOXYGEN_INVOKED) | ||
|  |          //! <b>Effects</b>: Calls `boost::move` if `input_reference` is not a lvalue reference. | ||
|  |          //!   Otherwise returns the reference | ||
|  |          template <class T> output_reference move_if_not_lvalue_reference(input_reference) noexcept; | ||
|  |       #elif defined(BOOST_MOVE_OLD_RVALUE_REF_BINDING_RULES) | ||
|  | 
 | ||
|  |          //Old move approach, lvalues could bind to rvalue references | ||
|  | 
 | ||
|  |          template <class T> | ||
|  |          BOOST_MOVE_FORCEINLINE T&& move_if_not_lvalue_reference(typename ::boost::move_detail::identity<T>::type&& t) BOOST_NOEXCEPT | ||
|  |          {  return t;   } | ||
|  | 
 | ||
|  |       #else //Old move | ||
|  | 
 | ||
|  |          template <class T> | ||
|  |          BOOST_MOVE_FORCEINLINE T&& move_if_not_lvalue_reference(typename ::boost::move_detail::remove_reference<T>::type& t) BOOST_NOEXCEPT | ||
|  |          {  return static_cast<T&&>(t);   } | ||
|  | 
 | ||
|  |          template <class T> | ||
|  |          BOOST_MOVE_FORCEINLINE T&& move_if_not_lvalue_reference(typename ::boost::move_detail::remove_reference<T>::type&& t) BOOST_NOEXCEPT | ||
|  |          { | ||
|  |             //"boost::forward<T> error: 'T' is a lvalue reference, can't forward as rvalue."; | ||
|  |             BOOST_STATIC_ASSERT(!boost::move_detail::is_lvalue_reference<T>::value); | ||
|  |             return static_cast<T&&>(t); | ||
|  |          } | ||
|  | 
 | ||
|  |       #endif   //BOOST_MOVE_DOXYGEN_INVOKED | ||
|  | 
 | ||
|  |       }  //namespace boost { | ||
|  | 
 | ||
|  |    #endif   //#if defined(BOOST_MOVE_USE_STANDARD_LIBRARY_MOVE) | ||
|  | 
 | ||
|  | #endif   //BOOST_NO_CXX11_RVALUE_REFERENCES | ||
|  | 
 | ||
|  | #if !defined(BOOST_MOVE_DOXYGEN_INVOKED) | ||
|  | 
 | ||
|  | namespace boost{ | ||
|  | namespace move_detail{ | ||
|  | 
 | ||
|  | template <typename T> | ||
|  | typename boost::move_detail::add_rvalue_reference<T>::type declval(); | ||
|  | 
 | ||
|  | }  //namespace move_detail{ | ||
|  | }  //namespace boost{ | ||
|  | 
 | ||
|  | #endif   //#if !defined(BOOST_MOVE_DOXYGEN_INVOKED) | ||
|  | 
 | ||
|  | 
 | ||
|  | #include <boost/move/detail/config_end.hpp> | ||
|  | 
 | ||
|  | #endif //#ifndef BOOST_MOVE_MOVE_UTILITY_CORE_HPP |