Chris@16
|
1 // (C) Copyright 2002-2008, Fernando Luis Cacciola Carballal.
|
Chris@16
|
2 //
|
Chris@16
|
3 // Distributed under the Boost Software License, Version 1.0. (See
|
Chris@16
|
4 // accompanying file LICENSE_1_0.txt or copy at
|
Chris@16
|
5 // http://www.boost.org/LICENSE_1_0.txt)
|
Chris@16
|
6 //
|
Chris@16
|
7 // 21 Ago 2002 (Created) Fernando Cacciola
|
Chris@16
|
8 // 24 Dec 2007 (Refactored and worked around various compiler bugs) Fernando Cacciola, Niels Dekker
|
Chris@16
|
9 // 23 May 2008 (Fixed operator= const issue, added initialized_value) Niels Dekker, Fernando Cacciola
|
Chris@16
|
10 // 21 Ago 2008 (Added swap) Niels Dekker, Fernando Cacciola
|
Chris@16
|
11 // 20 Feb 2009 (Fixed logical const-ness issues) Niels Dekker, Fernando Cacciola
|
Chris@16
|
12 // 03 Apr 2010 (Added initialized<T>, suggested by Jeffrey Hellrung, fixing #3472) Niels Dekker
|
Chris@16
|
13 // 30 May 2010 (Made memset call conditional, fixing #3869) Niels Dekker
|
Chris@16
|
14 //
|
Chris@16
|
15 #ifndef BOOST_UTILITY_VALUE_INIT_21AGO2002_HPP
|
Chris@16
|
16 #define BOOST_UTILITY_VALUE_INIT_21AGO2002_HPP
|
Chris@16
|
17
|
Chris@16
|
18 // Note: The implementation of boost::value_initialized had to deal with the
|
Chris@16
|
19 // fact that various compilers haven't fully implemented value-initialization.
|
Chris@16
|
20 // The constructor of boost::value_initialized<T> works around these compiler
|
Chris@16
|
21 // issues, by clearing the bytes of T, before constructing the T object it
|
Chris@16
|
22 // contains. More details on these issues are at libs/utility/value_init.htm
|
Chris@16
|
23
|
Chris@16
|
24 #include <boost/aligned_storage.hpp>
|
Chris@16
|
25 #include <boost/config.hpp> // For BOOST_NO_COMPLETE_VALUE_INITIALIZATION.
|
Chris@16
|
26 #include <boost/detail/workaround.hpp>
|
Chris@16
|
27 #include <boost/static_assert.hpp>
|
Chris@16
|
28 #include <boost/type_traits/cv_traits.hpp>
|
Chris@16
|
29 #include <boost/type_traits/alignment_of.hpp>
|
Chris@16
|
30 #include <boost/swap.hpp>
|
Chris@16
|
31 #include <cstring>
|
Chris@16
|
32 #include <new>
|
Chris@16
|
33
|
Chris@16
|
34 #ifdef BOOST_MSVC
|
Chris@16
|
35 #pragma warning(push)
|
Chris@16
|
36 // It is safe to ignore the following warning from MSVC 7.1 or higher:
|
Chris@16
|
37 // "warning C4351: new behavior: elements of array will be default initialized"
|
Chris@16
|
38 #pragma warning(disable: 4351)
|
Chris@16
|
39 // It is safe to ignore the following MSVC warning, which may pop up when T is
|
Chris@16
|
40 // a const type: "warning C4512: assignment operator could not be generated".
|
Chris@16
|
41 #pragma warning(disable: 4512)
|
Chris@16
|
42 #endif
|
Chris@16
|
43
|
Chris@16
|
44 #ifdef BOOST_NO_COMPLETE_VALUE_INITIALIZATION
|
Chris@16
|
45 // Implementation detail: The macro BOOST_DETAIL_VALUE_INIT_WORKAROUND_SUGGESTED
|
Chris@16
|
46 // suggests that a workaround should be applied, because of compiler issues
|
Chris@16
|
47 // regarding value-initialization.
|
Chris@16
|
48 #define BOOST_DETAIL_VALUE_INIT_WORKAROUND_SUGGESTED
|
Chris@16
|
49 #endif
|
Chris@16
|
50
|
Chris@16
|
51 // Implementation detail: The macro BOOST_DETAIL_VALUE_INIT_WORKAROUND
|
Chris@16
|
52 // switches the value-initialization workaround either on or off.
|
Chris@16
|
53 #ifndef BOOST_DETAIL_VALUE_INIT_WORKAROUND
|
Chris@16
|
54 #ifdef BOOST_DETAIL_VALUE_INIT_WORKAROUND_SUGGESTED
|
Chris@16
|
55 #define BOOST_DETAIL_VALUE_INIT_WORKAROUND 1
|
Chris@16
|
56 #else
|
Chris@16
|
57 #define BOOST_DETAIL_VALUE_INIT_WORKAROUND 0
|
Chris@16
|
58 #endif
|
Chris@16
|
59 #endif
|
Chris@16
|
60
|
Chris@16
|
61 namespace boost {
|
Chris@16
|
62
|
Chris@16
|
63 template<class T>
|
Chris@16
|
64 class initialized
|
Chris@16
|
65 {
|
Chris@16
|
66 private :
|
Chris@16
|
67 struct wrapper
|
Chris@16
|
68 {
|
Chris@16
|
69 #if !BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x592))
|
Chris@16
|
70 typename
|
Chris@16
|
71 #endif
|
Chris@16
|
72 remove_const<T>::type data;
|
Chris@16
|
73
|
Chris@101
|
74 BOOST_GPU_ENABLED
|
Chris@16
|
75 wrapper()
|
Chris@16
|
76 :
|
Chris@16
|
77 data()
|
Chris@16
|
78 {
|
Chris@16
|
79 }
|
Chris@16
|
80
|
Chris@101
|
81 BOOST_GPU_ENABLED
|
Chris@16
|
82 wrapper(T const & arg)
|
Chris@16
|
83 :
|
Chris@16
|
84 data(arg)
|
Chris@16
|
85 {
|
Chris@16
|
86 }
|
Chris@16
|
87 };
|
Chris@16
|
88
|
Chris@16
|
89 mutable
|
Chris@16
|
90 #if !BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x592))
|
Chris@16
|
91 typename
|
Chris@16
|
92 #endif
|
Chris@16
|
93 aligned_storage<sizeof(wrapper), alignment_of<wrapper>::value>::type x;
|
Chris@16
|
94
|
Chris@101
|
95 BOOST_GPU_ENABLED
|
Chris@16
|
96 wrapper * wrapper_address() const
|
Chris@16
|
97 {
|
Chris@16
|
98 return static_cast<wrapper *>( static_cast<void*>(&x));
|
Chris@16
|
99 }
|
Chris@16
|
100
|
Chris@16
|
101 public :
|
Chris@16
|
102
|
Chris@101
|
103 BOOST_GPU_ENABLED
|
Chris@16
|
104 initialized()
|
Chris@16
|
105 {
|
Chris@16
|
106 #if BOOST_DETAIL_VALUE_INIT_WORKAROUND
|
Chris@16
|
107 std::memset(&x, 0, sizeof(x));
|
Chris@16
|
108 #endif
|
Chris@16
|
109 new (wrapper_address()) wrapper();
|
Chris@16
|
110 }
|
Chris@16
|
111
|
Chris@101
|
112 BOOST_GPU_ENABLED
|
Chris@16
|
113 initialized(initialized const & arg)
|
Chris@16
|
114 {
|
Chris@16
|
115 new (wrapper_address()) wrapper( static_cast<wrapper const &>(*(arg.wrapper_address())));
|
Chris@16
|
116 }
|
Chris@16
|
117
|
Chris@101
|
118 BOOST_GPU_ENABLED
|
Chris@16
|
119 explicit initialized(T const & arg)
|
Chris@16
|
120 {
|
Chris@16
|
121 new (wrapper_address()) wrapper(arg);
|
Chris@16
|
122 }
|
Chris@16
|
123
|
Chris@101
|
124 BOOST_GPU_ENABLED
|
Chris@16
|
125 initialized & operator=(initialized const & arg)
|
Chris@16
|
126 {
|
Chris@16
|
127 // Assignment is only allowed when T is non-const.
|
Chris@16
|
128 BOOST_STATIC_ASSERT( ! is_const<T>::value );
|
Chris@16
|
129 *wrapper_address() = static_cast<wrapper const &>(*(arg.wrapper_address()));
|
Chris@16
|
130 return *this;
|
Chris@16
|
131 }
|
Chris@16
|
132
|
Chris@101
|
133 BOOST_GPU_ENABLED
|
Chris@16
|
134 ~initialized()
|
Chris@16
|
135 {
|
Chris@16
|
136 wrapper_address()->wrapper::~wrapper();
|
Chris@16
|
137 }
|
Chris@16
|
138
|
Chris@101
|
139 BOOST_GPU_ENABLED
|
Chris@16
|
140 T const & data() const
|
Chris@16
|
141 {
|
Chris@16
|
142 return wrapper_address()->data;
|
Chris@16
|
143 }
|
Chris@16
|
144
|
Chris@101
|
145 BOOST_GPU_ENABLED
|
Chris@16
|
146 T& data()
|
Chris@16
|
147 {
|
Chris@16
|
148 return wrapper_address()->data;
|
Chris@16
|
149 }
|
Chris@16
|
150
|
Chris@101
|
151 BOOST_GPU_ENABLED
|
Chris@16
|
152 void swap(initialized & arg)
|
Chris@16
|
153 {
|
Chris@16
|
154 ::boost::swap( this->data(), arg.data() );
|
Chris@16
|
155 }
|
Chris@16
|
156
|
Chris@101
|
157 BOOST_GPU_ENABLED
|
Chris@16
|
158 operator T const &() const
|
Chris@16
|
159 {
|
Chris@16
|
160 return wrapper_address()->data;
|
Chris@16
|
161 }
|
Chris@16
|
162
|
Chris@101
|
163 BOOST_GPU_ENABLED
|
Chris@16
|
164 operator T&()
|
Chris@16
|
165 {
|
Chris@16
|
166 return wrapper_address()->data;
|
Chris@16
|
167 }
|
Chris@16
|
168
|
Chris@16
|
169 } ;
|
Chris@16
|
170
|
Chris@16
|
171 template<class T>
|
Chris@101
|
172 BOOST_GPU_ENABLED
|
Chris@16
|
173 T const& get ( initialized<T> const& x )
|
Chris@16
|
174 {
|
Chris@16
|
175 return x.data() ;
|
Chris@16
|
176 }
|
Chris@16
|
177
|
Chris@16
|
178 template<class T>
|
Chris@101
|
179 BOOST_GPU_ENABLED
|
Chris@16
|
180 T& get ( initialized<T>& x )
|
Chris@16
|
181 {
|
Chris@16
|
182 return x.data() ;
|
Chris@16
|
183 }
|
Chris@16
|
184
|
Chris@16
|
185 template<class T>
|
Chris@101
|
186 BOOST_GPU_ENABLED
|
Chris@16
|
187 void swap ( initialized<T> & lhs, initialized<T> & rhs )
|
Chris@16
|
188 {
|
Chris@16
|
189 lhs.swap(rhs) ;
|
Chris@16
|
190 }
|
Chris@16
|
191
|
Chris@16
|
192 template<class T>
|
Chris@16
|
193 class value_initialized
|
Chris@16
|
194 {
|
Chris@16
|
195 private :
|
Chris@16
|
196
|
Chris@16
|
197 // initialized<T> does value-initialization by default.
|
Chris@16
|
198 initialized<T> m_data;
|
Chris@16
|
199
|
Chris@16
|
200 public :
|
Chris@16
|
201
|
Chris@101
|
202 BOOST_GPU_ENABLED
|
Chris@16
|
203 value_initialized()
|
Chris@16
|
204 :
|
Chris@16
|
205 m_data()
|
Chris@16
|
206 { }
|
Chris@16
|
207
|
Chris@101
|
208 BOOST_GPU_ENABLED
|
Chris@16
|
209 T const & data() const
|
Chris@16
|
210 {
|
Chris@16
|
211 return m_data.data();
|
Chris@16
|
212 }
|
Chris@16
|
213
|
Chris@101
|
214 BOOST_GPU_ENABLED
|
Chris@16
|
215 T& data()
|
Chris@16
|
216 {
|
Chris@16
|
217 return m_data.data();
|
Chris@16
|
218 }
|
Chris@16
|
219
|
Chris@101
|
220 BOOST_GPU_ENABLED
|
Chris@16
|
221 void swap(value_initialized & arg)
|
Chris@16
|
222 {
|
Chris@16
|
223 m_data.swap(arg.m_data);
|
Chris@16
|
224 }
|
Chris@16
|
225
|
Chris@101
|
226 BOOST_GPU_ENABLED
|
Chris@16
|
227 operator T const &() const
|
Chris@16
|
228 {
|
Chris@16
|
229 return m_data;
|
Chris@16
|
230 }
|
Chris@16
|
231
|
Chris@101
|
232 BOOST_GPU_ENABLED
|
Chris@16
|
233 operator T&()
|
Chris@16
|
234 {
|
Chris@16
|
235 return m_data;
|
Chris@16
|
236 }
|
Chris@16
|
237 } ;
|
Chris@16
|
238
|
Chris@16
|
239
|
Chris@16
|
240 template<class T>
|
Chris@101
|
241 BOOST_GPU_ENABLED
|
Chris@16
|
242 T const& get ( value_initialized<T> const& x )
|
Chris@16
|
243 {
|
Chris@16
|
244 return x.data() ;
|
Chris@16
|
245 }
|
Chris@16
|
246
|
Chris@16
|
247 template<class T>
|
Chris@101
|
248 BOOST_GPU_ENABLED
|
Chris@16
|
249 T& get ( value_initialized<T>& x )
|
Chris@16
|
250 {
|
Chris@16
|
251 return x.data() ;
|
Chris@16
|
252 }
|
Chris@16
|
253
|
Chris@16
|
254 template<class T>
|
Chris@101
|
255 BOOST_GPU_ENABLED
|
Chris@16
|
256 void swap ( value_initialized<T> & lhs, value_initialized<T> & rhs )
|
Chris@16
|
257 {
|
Chris@16
|
258 lhs.swap(rhs) ;
|
Chris@16
|
259 }
|
Chris@16
|
260
|
Chris@16
|
261
|
Chris@16
|
262 class initialized_value_t
|
Chris@16
|
263 {
|
Chris@16
|
264 public :
|
Chris@16
|
265
|
Chris@101
|
266 template <class T> BOOST_GPU_ENABLED operator T() const
|
Chris@16
|
267 {
|
Chris@16
|
268 return initialized<T>().data();
|
Chris@16
|
269 }
|
Chris@16
|
270 };
|
Chris@16
|
271
|
Chris@16
|
272 initialized_value_t const initialized_value = {} ;
|
Chris@16
|
273
|
Chris@16
|
274
|
Chris@16
|
275 } // namespace boost
|
Chris@16
|
276
|
Chris@16
|
277 #ifdef BOOST_MSVC
|
Chris@16
|
278 #pragma warning(pop)
|
Chris@16
|
279 #endif
|
Chris@16
|
280
|
Chris@16
|
281 #endif
|