1 | // Copyright David Abrahams 2002. |
---|
2 | // Distributed under the Boost Software License, Version 1.0. (See |
---|
3 | // accompanying file LICENSE_1_0.txt or copy at |
---|
4 | // http://www.boost.org/LICENSE_1_0.txt) |
---|
5 | #ifndef REFERENT_STORAGE_DWA200278_HPP |
---|
6 | # define REFERENT_STORAGE_DWA200278_HPP |
---|
7 | # include <boost/mpl/if.hpp> |
---|
8 | # include <cstddef> |
---|
9 | |
---|
10 | namespace boost { namespace python { namespace detail { |
---|
11 | |
---|
12 | struct alignment_dummy; |
---|
13 | typedef void (*function_ptr)(); |
---|
14 | typedef int (alignment_dummy::*member_ptr); |
---|
15 | typedef int (alignment_dummy::*member_function_ptr)(); |
---|
16 | |
---|
17 | # define BOOST_PYTHON_ALIGNER(T, n) \ |
---|
18 | typename mpl::if_c< \ |
---|
19 | sizeof(T) <= size, T, char>::type t##n |
---|
20 | |
---|
21 | // Storage for size bytes, aligned to all fundamental types no larger than size |
---|
22 | template <std::size_t size> |
---|
23 | union aligned_storage |
---|
24 | { |
---|
25 | BOOST_PYTHON_ALIGNER(char, 0); |
---|
26 | BOOST_PYTHON_ALIGNER(short, 1); |
---|
27 | BOOST_PYTHON_ALIGNER(int, 2); |
---|
28 | BOOST_PYTHON_ALIGNER(long, 3); |
---|
29 | BOOST_PYTHON_ALIGNER(float, 4); |
---|
30 | BOOST_PYTHON_ALIGNER(double, 5); |
---|
31 | BOOST_PYTHON_ALIGNER(long double, 6); |
---|
32 | BOOST_PYTHON_ALIGNER(void*, 7); |
---|
33 | BOOST_PYTHON_ALIGNER(function_ptr, 8); |
---|
34 | BOOST_PYTHON_ALIGNER(member_ptr, 9); |
---|
35 | BOOST_PYTHON_ALIGNER(member_function_ptr, 10); |
---|
36 | char bytes[size]; |
---|
37 | }; |
---|
38 | |
---|
39 | # undef BOOST_PYTHON_ALIGNER |
---|
40 | |
---|
41 | // Compute the size of T's referent. We wouldn't need this at all, |
---|
42 | // but sizeof() is broken in CodeWarriors <= 8.0 |
---|
43 | template <class T> struct referent_size; |
---|
44 | |
---|
45 | # ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION |
---|
46 | |
---|
47 | template <class T> |
---|
48 | struct referent_size<T&> |
---|
49 | { |
---|
50 | BOOST_STATIC_CONSTANT( |
---|
51 | std::size_t, value = sizeof(T)); |
---|
52 | }; |
---|
53 | |
---|
54 | # else |
---|
55 | |
---|
56 | template <class T> struct referent_size |
---|
57 | { |
---|
58 | static T f(); |
---|
59 | BOOST_STATIC_CONSTANT(std::size_t, value = sizeof(f())); |
---|
60 | }; |
---|
61 | |
---|
62 | # endif |
---|
63 | |
---|
64 | // A metafunction returning a POD type which can store U, where T == |
---|
65 | // U&. If T is not a reference type, returns a POD which can store T. |
---|
66 | template <class T> |
---|
67 | struct referent_storage |
---|
68 | { |
---|
69 | typedef aligned_storage< |
---|
70 | ::boost::python::detail::referent_size<T>::value |
---|
71 | > type; |
---|
72 | }; |
---|
73 | |
---|
74 | }}} // namespace boost::python::detail |
---|
75 | |
---|
76 | #endif // REFERENT_STORAGE_DWA200278_HPP |
---|