Chris@16
|
1 // Copyright David Abrahams 2001.
|
Chris@16
|
2 // Distributed under the Boost Software License, Version 1.0. (See
|
Chris@16
|
3 // accompanying file LICENSE_1_0.txt or copy at
|
Chris@16
|
4 // http://www.boost.org/LICENSE_1_0.txt)
|
Chris@16
|
5 #ifndef FORWARD_DWA20011215_HPP
|
Chris@16
|
6 # define FORWARD_DWA20011215_HPP
|
Chris@16
|
7
|
Chris@16
|
8 # include <boost/mpl/if.hpp>
|
Chris@16
|
9 # include <boost/type_traits/is_scalar.hpp>
|
Chris@16
|
10 # include <boost/type_traits/add_const.hpp>
|
Chris@16
|
11 # include <boost/type_traits/add_reference.hpp>
|
Chris@16
|
12 # include <boost/ref.hpp>
|
Chris@16
|
13 # include <boost/python/detail/value_arg.hpp>
|
Chris@16
|
14 # include <boost/python/detail/copy_ctor_mutates_rhs.hpp>
|
Chris@16
|
15 # if BOOST_WORKAROUND(BOOST_MSVC, < 1300)
|
Chris@16
|
16 # include <boost/type_traits/is_enum.hpp>
|
Chris@16
|
17 # include <boost/mpl/and.hpp>
|
Chris@16
|
18 # include <boost/mpl/not.hpp>
|
Chris@16
|
19 # else
|
Chris@16
|
20 # include <boost/mpl/or.hpp>
|
Chris@16
|
21 # endif
|
Chris@16
|
22
|
Chris@16
|
23 namespace boost { namespace python { namespace objects {
|
Chris@16
|
24
|
Chris@16
|
25 // Very much like boost::reference_wrapper<T>, except that in this
|
Chris@16
|
26 // case T can be a reference already without causing a
|
Chris@16
|
27 // reference-to-reference error.
|
Chris@16
|
28 template <class T>
|
Chris@16
|
29 struct reference_to_value
|
Chris@16
|
30 {
|
Chris@16
|
31 typedef typename add_reference<typename add_const<T>::type>::type reference;
|
Chris@16
|
32
|
Chris@16
|
33 reference_to_value(reference x) : m_value(x) {}
|
Chris@16
|
34 reference get() const { return m_value; }
|
Chris@16
|
35 private:
|
Chris@16
|
36 reference m_value;
|
Chris@16
|
37 };
|
Chris@16
|
38
|
Chris@16
|
39 // A little metaprogram which selects the type to pass through an
|
Chris@16
|
40 // intermediate forwarding function when the destination argument type
|
Chris@16
|
41 // is T.
|
Chris@16
|
42 template <class T>
|
Chris@16
|
43 struct forward
|
Chris@16
|
44 : mpl::if_<
|
Chris@16
|
45 # if BOOST_WORKAROUND(BOOST_MSVC, < 1300)
|
Chris@16
|
46 // vc6 chokes on unforwarding enums nested in classes
|
Chris@16
|
47 mpl::and_<
|
Chris@16
|
48 is_scalar<T>
|
Chris@16
|
49 , mpl::not_<
|
Chris@16
|
50 is_enum<T>
|
Chris@16
|
51 >
|
Chris@16
|
52 >
|
Chris@16
|
53 # else
|
Chris@16
|
54 mpl::or_<python::detail::copy_ctor_mutates_rhs<T>, is_scalar<T> >
|
Chris@16
|
55 # endif
|
Chris@16
|
56 , T
|
Chris@16
|
57 , reference_to_value<T>
|
Chris@16
|
58 >
|
Chris@16
|
59 {
|
Chris@16
|
60 };
|
Chris@16
|
61
|
Chris@16
|
62 # ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION
|
Chris@16
|
63 template<typename T>
|
Chris@16
|
64 struct unforward
|
Chris@16
|
65 {
|
Chris@16
|
66 typedef typename unwrap_reference<T>::type& type;
|
Chris@16
|
67 };
|
Chris@16
|
68
|
Chris@16
|
69 template<typename T>
|
Chris@16
|
70 struct unforward<reference_to_value<T> >
|
Chris@16
|
71 {
|
Chris@16
|
72 typedef T type;
|
Chris@16
|
73 };
|
Chris@16
|
74
|
Chris@16
|
75 template <typename T>
|
Chris@16
|
76 struct unforward_cref
|
Chris@16
|
77 : python::detail::value_arg<
|
Chris@16
|
78 typename unwrap_reference<T>::type
|
Chris@16
|
79 >
|
Chris@16
|
80 {
|
Chris@16
|
81 };
|
Chris@16
|
82
|
Chris@16
|
83 template<typename T>
|
Chris@16
|
84 struct unforward_cref<reference_to_value<T> >
|
Chris@16
|
85 : add_reference<typename add_const<T>::type>
|
Chris@16
|
86 {
|
Chris@16
|
87 };
|
Chris@16
|
88
|
Chris@16
|
89 # else // no partial specialization
|
Chris@16
|
90
|
Chris@16
|
91 namespace detail
|
Chris@16
|
92 {
|
Chris@16
|
93 typedef char (&yes_reference_to_value_t)[1];
|
Chris@16
|
94 typedef char (&no_reference_to_value_t)[2];
|
Chris@16
|
95
|
Chris@16
|
96 no_reference_to_value_t is_reference_to_value_test(...);
|
Chris@16
|
97
|
Chris@16
|
98 template<typename T>
|
Chris@16
|
99 yes_reference_to_value_t is_reference_to_value_test(boost::type< reference_to_value<T> >);
|
Chris@16
|
100
|
Chris@16
|
101 template<bool wrapped>
|
Chris@16
|
102 struct unforwarder
|
Chris@16
|
103 {
|
Chris@16
|
104 template <class T>
|
Chris@16
|
105 struct apply
|
Chris@16
|
106 {
|
Chris@16
|
107 typedef typename unwrap_reference<T>::type& type;
|
Chris@16
|
108 };
|
Chris@16
|
109 };
|
Chris@16
|
110
|
Chris@16
|
111 template<>
|
Chris@16
|
112 struct unforwarder<true>
|
Chris@16
|
113 {
|
Chris@16
|
114 template <class T>
|
Chris@16
|
115 struct apply
|
Chris@16
|
116 {
|
Chris@16
|
117 typedef typename T::reference type;
|
Chris@16
|
118 };
|
Chris@16
|
119 };
|
Chris@16
|
120
|
Chris@16
|
121 template<bool wrapped = false>
|
Chris@16
|
122 struct cref_unforwarder
|
Chris@16
|
123 {
|
Chris@16
|
124 template <class T>
|
Chris@16
|
125 struct apply
|
Chris@16
|
126 : python::detail::value_arg<
|
Chris@16
|
127 typename unwrap_reference<T>::type
|
Chris@16
|
128 >
|
Chris@16
|
129 {
|
Chris@16
|
130 };
|
Chris@16
|
131 };
|
Chris@16
|
132
|
Chris@16
|
133 template<>
|
Chris@16
|
134 struct cref_unforwarder<true>
|
Chris@16
|
135 {
|
Chris@16
|
136 template <class T>
|
Chris@16
|
137 struct apply
|
Chris@16
|
138 : python::detail::value_arg<
|
Chris@16
|
139 typename T::reference
|
Chris@16
|
140 >
|
Chris@16
|
141 {
|
Chris@16
|
142 };
|
Chris@16
|
143 };
|
Chris@16
|
144
|
Chris@16
|
145 template<typename T>
|
Chris@16
|
146 struct is_reference_to_value
|
Chris@16
|
147 {
|
Chris@16
|
148 BOOST_STATIC_CONSTANT(
|
Chris@16
|
149 bool, value = (
|
Chris@16
|
150 sizeof(is_reference_to_value_test(boost::type<T>()))
|
Chris@16
|
151 == sizeof(yes_reference_to_value_t)));
|
Chris@16
|
152 typedef mpl::bool_<value> type;
|
Chris@16
|
153 };
|
Chris@16
|
154 }
|
Chris@16
|
155
|
Chris@16
|
156 template <typename T>
|
Chris@16
|
157 struct unforward
|
Chris@16
|
158 : public detail::unforwarder<
|
Chris@16
|
159 detail::is_reference_to_value<T>::value
|
Chris@16
|
160 >::template apply<T>
|
Chris@16
|
161 {};
|
Chris@16
|
162
|
Chris@16
|
163 template <typename T>
|
Chris@16
|
164 struct unforward_cref
|
Chris@16
|
165 : public detail::cref_unforwarder<
|
Chris@16
|
166 detail::is_reference_to_value<T>::value
|
Chris@16
|
167 >::template apply<T>
|
Chris@16
|
168 {};
|
Chris@16
|
169
|
Chris@16
|
170 # endif // BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION
|
Chris@16
|
171
|
Chris@16
|
172 template <class T>
|
Chris@16
|
173 typename reference_to_value<T>::reference
|
Chris@16
|
174 do_unforward(reference_to_value<T> const& x, int)
|
Chris@16
|
175 {
|
Chris@16
|
176 return x.get();
|
Chris@16
|
177 }
|
Chris@16
|
178
|
Chris@16
|
179 template <class T>
|
Chris@16
|
180 typename reference_wrapper<T>::type&
|
Chris@16
|
181 do_unforward(reference_wrapper<T> const& x, int)
|
Chris@16
|
182 {
|
Chris@16
|
183 return x.get();
|
Chris@16
|
184 }
|
Chris@16
|
185
|
Chris@16
|
186 template <class T>
|
Chris@16
|
187 T const& do_unforward(T const& x, ...)
|
Chris@16
|
188 {
|
Chris@16
|
189 return x;
|
Chris@16
|
190 }
|
Chris@16
|
191
|
Chris@16
|
192 }}} // namespace boost::python::objects
|
Chris@16
|
193
|
Chris@16
|
194 #endif // FORWARD_DWA20011215_HPP
|