blob: 2b8287946f8ddb17f78407e7042c7a61c112894b [file] [log] [blame]
Jeff Thompsona28eed82013-08-22 16:21:10 -07001//////////////////////////////////////////////////////////////////////////////
2//
3// (C) Copyright Ion Gaztanaga 2012-2012.
4// Distributed under the Boost Software License, Version 1.0.
5// (See accompanying file LICENSE_1_0.txt or copy at
6// http://www.boost.org/LICENSE_1_0.txt)
7//
8// See http://www.boost.org/libs/move for documentation.
9//
10//////////////////////////////////////////////////////////////////////////////
11
12//! \file
13
14#ifndef BOOST_MOVE_MOVE_UTILITY_HPP
15#define BOOST_MOVE_MOVE_UTILITY_HPP
16
17#include <ndnboost/move/detail/config_begin.hpp>
18#include <ndnboost/move/core.hpp>
19#include <ndnboost/move/detail/meta_utils.hpp>
20
21#if defined(BOOST_NO_CXX11_RVALUE_REFERENCES) && !defined(BOOST_MOVE_DOXYGEN_INVOKED)
22
23 namespace ndnboost {
24
25 template<class T>
26 struct enable_move_utility_emulation
27 {
28 static const bool value = true;
29 };
30
31 //////////////////////////////////////////////////////////////////////////////
32 //
33 // move()
34 //
35 //////////////////////////////////////////////////////////////////////////////
36
37 template <class T>
38 inline typename ::ndnboost::move_detail::enable_if_c
39 < enable_move_utility_emulation<T>::value && !has_move_emulation_enabled<T>::value, T&>::type
40 move(T& x)
41 {
42 return x;
43 }
44
45 template <class T>
46 inline typename ::ndnboost::move_detail::enable_if_c
47 < enable_move_utility_emulation<T>::value && has_move_emulation_enabled<T>::value, rv<T>&>::type
48 move(T& x)
49 {
50 return *static_cast<rv<T>* >(::ndnboost::move_detail::addressof(x));
51 }
52
53 template <class T>
54 inline typename ::ndnboost::move_detail::enable_if_c
55 < enable_move_utility_emulation<T>::value && has_move_emulation_enabled<T>::value, rv<T>&>::type
56 move(rv<T>& x)
57 {
58 return x;
59 }
60
61 //////////////////////////////////////////////////////////////////////////////
62 //
63 // forward()
64 //
65 //////////////////////////////////////////////////////////////////////////////
66
67 template <class T>
68 inline typename ::ndnboost::move_detail::enable_if_c
69 < enable_move_utility_emulation<T>::value && ::ndnboost::move_detail::is_rv<T>::value, T &>::type
70 forward(const typename ::ndnboost::move_detail::identity<T>::type &x)
71 {
72 return const_cast<T&>(x);
73 }
74
75 template <class T>
76 inline typename ::ndnboost::move_detail::enable_if_c
77 < enable_move_utility_emulation<T>::value && !::ndnboost::move_detail::is_rv<T>::value, const T &>::type
78 forward(const typename ::ndnboost::move_detail::identity<T>::type &x)
79 {
80 return x;
81 }
82
83 } //namespace ndnboost
84
85#else //#if defined(BOOST_NO_CXX11_RVALUE_REFERENCES) && !defined(BOOST_MOVE_DOXYGEN_INVOKED)
86
87 #if defined(BOOST_MOVE_USE_STANDARD_LIBRARY_MOVE)
88 #include <utility>
89
90 namespace ndnboost{
91
92 using ::std::move;
93 using ::std::forward;
94
95 } //namespace ndnboost
96
97 #else //!BOOST_MOVE_USE_STANDARD_LIBRARY_MOVE
98
99 #include <ndnboost/type_traits/remove_reference.hpp>
100
101 namespace ndnboost {
102
103 //! This trait's internal boolean `value` is false in compilers with rvalue references
104 //! and true in compilers without rvalue references.
105 //!
106 //! A user can specialize this trait for a type T to false to SFINAE out `move` and `forward`
107 //! so that the user can define a different move emulation for that type in namespace ndnboost
108 //! (e.g. another Boost library for its types) and avoid any overload ambiguity.
109 template<class T>
110 struct enable_move_utility_emulation
111 {
112 static const bool value = false;
113 };
114
115 //////////////////////////////////////////////////////////////////////////////
116 //
117 // move
118 //
119 //////////////////////////////////////////////////////////////////////////////
120
121 #if defined(BOOST_MOVE_DOXYGEN_INVOKED)
122 //! This function provides a way to convert a reference into a rvalue reference
123 //! in compilers with rvalue references. For other compilers converts T & into
124 //! <i>::ndnboost::rv<T> &</i> so that move emulation is activated.
125 template <class T>
126 rvalue_reference move (input_reference);
127
128 #elif defined(BOOST_MOVE_OLD_RVALUE_REF_BINDING_RULES)
129
130 //Old move approach, lvalues could bind to rvalue references
131 template <class T>
132 inline typename remove_reference<T>::type && move(T&& t)
133 { return t; }
134
135 #else //BOOST_MOVE_OLD_RVALUE_REF_BINDING_RULES
136
137 template <class T>
138 inline typename remove_reference<T>::type && move(T&& t)
139 { return static_cast<typename remove_reference<T>::type &&>(t); }
140
141 #endif //BOOST_MOVE_OLD_RVALUE_REF_BINDING_RULES
142
143 //////////////////////////////////////////////////////////////////////////////
144 //
145 // forward
146 //
147 //////////////////////////////////////////////////////////////////////////////
148
149
150 #if defined(BOOST_MOVE_DOXYGEN_INVOKED)
151 //! This function provides limited form of forwarding that is usually enough for
152 //! in-place construction and avoids the exponential overloading for
153 //! achieve the limited forwarding in C++03.
154 //!
155 //! For compilers with rvalue references this function provides perfect forwarding.
156 //!
157 //! Otherwise:
158 //! * If input_reference binds to const ::ndnboost::rv<T> & then it output_reference is
159 //! ::ndnboost::rv<T> &
160 //!
161 //! * Else, output_reference is equal to input_reference.
162 template <class T> output_reference forward(input_reference);
163 #elif defined(BOOST_MOVE_OLD_RVALUE_REF_BINDING_RULES)
164
165 //Old move approach, lvalues could bind to rvalue references
166
167 template <class T>
168 inline T&& forward (typename ::ndnboost::move_detail::identity<T>::type&& t)
169 { return t; }
170
171 #else //Old move
172
173 //Implementation #5 from N2951, thanks to Howard Hinnant
174
175 template <class T, class U>
176 inline T&& forward(U&& t
177 , typename ::ndnboost::move_detail::enable_if_c<
178 move_detail::is_lvalue_reference<T>::value ? move_detail::is_lvalue_reference<U>::value : true>::type * = 0/*
179 , typename ::ndnboost::move_detail::enable_if_c<
180 move_detail::is_convertible
181 <typename remove_reference<U>::type*, typename remove_reference<T>::type*>::value>::type * = 0*/)
182 { return static_cast<T&&>(t); }
183
184 #endif //BOOST_MOVE_DOXYGEN_INVOKED
185
186 } //namespace ndnboost {
187
188 #endif //#if defined(BOOST_MOVE_USE_STANDARD_LIBRARY_MOVE)
189
190#endif //BOOST_NO_CXX11_RVALUE_REFERENCES
191
192#include <ndnboost/move/detail/config_end.hpp>
193
194#endif //#ifndef BOOST_MOVE_MOVE_UTILITY_HPP