2 * Copyright (c) 2012-2014 Glen Joseph Fernandes
3 * glenfe at live dot com
5 * Distributed under the Boost Software License,
6 * Version 1.0. (See accompanying file LICENSE_1_0.txt
7 * or copy at http://boost.org/LICENSE_1_0.txt)
9 #ifndef BOOST_SMART_PTR_ALLOCATE_SHARED_ARRAY_HPP
10 #define BOOST_SMART_PTR_ALLOCATE_SHARED_ARRAY_HPP
12 #include <boost/smart_ptr/detail/array_count_impl.hpp>
13 #include <boost/smart_ptr/detail/sp_if_array.hpp>
16 template<class T, class A>
17 inline typename boost::detail::sp_if_array<T>::type
18 allocate_shared(const A& allocator, std::size_t size) {
19 typedef typename boost::detail::array_inner<T>::type T1;
20 typedef typename boost::detail::array_base<T1>::type T2;
21 typedef boost::detail::ms_init_tag R1;
22 typedef boost::detail::as_allocator<A, T, R1> A1;
23 typedef boost::detail::ms_in_allocator_tag D1;
24 std::size_t n1 = size * boost::detail::array_total<T1>::size;
28 A1 a1(allocator, size, &p2);
29 shared_ptr<T> s1(p1, d1, a1);
30 A1* a2 = static_cast<A1*>(s1._internal_get_untyped_deleter());
32 #if !defined(BOOST_NO_CXX11_ALLOCATOR)
33 boost::detail::as_init(allocator, p2, n1);
35 boost::detail::ms_init(p2, n1);
38 p1 = reinterpret_cast<T1*>(p2);
39 return shared_ptr<T>(s1, p1);
42 template<class T, class A>
43 inline typename boost::detail::sp_if_size_array<T>::type
44 allocate_shared(const A& allocator) {
45 typedef typename boost::detail::array_inner<T>::type T1;
46 typedef typename boost::detail::array_base<T1>::type T2;
47 typedef boost::detail::ms_init_tag R1;
48 typedef boost::detail::as_allocator<A, T, R1> A1;
49 typedef boost::detail::ms_in_allocator_tag D1;
51 N = boost::detail::array_total<T>::size
56 A1 a1(allocator, &p2);
57 shared_ptr<T> s1(p1, d1, a1);
58 A1* a2 = static_cast<A1*>(s1._internal_get_untyped_deleter());
60 #if !defined(BOOST_NO_CXX11_ALLOCATOR)
61 boost::detail::as_init(allocator, p2, N);
63 boost::detail::ms_init(p2, N);
66 p1 = reinterpret_cast<T1*>(p2);
67 return shared_ptr<T>(s1, p1);
70 template<class T, class A>
71 inline typename boost::detail::sp_if_array<T>::type
72 allocate_shared(const A& allocator, std::size_t size,
73 const typename boost::detail::array_inner<T>::type& value) {
74 typedef typename boost::detail::array_inner<T>::type T1;
75 typedef typename boost::detail::array_base<T1>::type T2;
77 typedef boost::detail::ms_init_tag R1;
78 typedef boost::detail::as_allocator<A, T, R1> A1;
79 typedef boost::detail::ms_in_allocator_tag D1;
81 M = boost::detail::array_total<T1>::size
83 std::size_t n1 = M * size;
86 T3* p3 = reinterpret_cast<T3*>(&value);
88 A1 a1(allocator, size, &p2);
89 shared_ptr<T> s1(p1, d1, a1);
90 A1* a2 = static_cast<A1*>(s1._internal_get_untyped_deleter());
92 #if !defined(BOOST_NO_CXX11_ALLOCATOR)
93 boost::detail::as_init<T2, A, M>(allocator, p2, n1, p3);
95 boost::detail::ms_init<T2, M>(p2, n1, p3);
98 p1 = reinterpret_cast<T1*>(p2);
99 return shared_ptr<T>(s1, p1);
102 template<class T, class A>
103 inline typename boost::detail::sp_if_size_array<T>::type
104 allocate_shared(const A& allocator,
105 const typename boost::detail::array_inner<T>::type& value) {
106 typedef typename boost::detail::array_inner<T>::type T1;
107 typedef typename boost::detail::array_base<T1>::type T2;
109 typedef boost::detail::ms_init_tag R1;
110 typedef boost::detail::as_allocator<A, T, R1> A1;
111 typedef boost::detail::ms_in_allocator_tag D1;
113 N = boost::detail::array_total<T>::size,
114 M = boost::detail::array_total<T1>::size
118 T3* p3 = reinterpret_cast<T3*>(&value);
120 A1 a1(allocator, &p2);
121 shared_ptr<T> s1(p1, d1, a1);
122 A1* a2 = static_cast<A1*>(s1._internal_get_untyped_deleter());
124 #if !defined(BOOST_NO_CXX11_ALLOCATOR)
125 boost::detail::as_init<T2, A, M>(allocator, p2, N, p3);
127 boost::detail::ms_init<T2, M>(p2, N, p3);
130 p1 = reinterpret_cast<T1*>(p2);
131 return shared_ptr<T>(s1, p1);
134 template<class T, class A>
135 inline typename boost::detail::sp_if_array<T>::type
136 allocate_shared_noinit(const A& allocator, std::size_t size) {
137 typedef typename boost::detail::array_inner<T>::type T1;
138 typedef typename boost::detail::array_base<T1>::type T2;
139 typedef boost::detail::ms_noinit_tag R1;
140 typedef boost::detail::as_allocator<A, T, R1> A1;
141 typedef boost::detail::ms_in_allocator_tag D1;
142 std::size_t n1 = size * boost::detail::array_total<T1>::size;
146 A1 a1(allocator, size, &p2);
147 shared_ptr<T> s1(p1, d1, a1);
148 A1* a2 = static_cast<A1*>(s1._internal_get_untyped_deleter());
150 boost::detail::ms_noinit(p2, n1);
152 p1 = reinterpret_cast<T1*>(p2);
153 return shared_ptr<T>(s1, p1);
156 template<class T, class A>
157 inline typename boost::detail::sp_if_size_array<T>::type
158 allocate_shared_noinit(const A& allocator) {
159 typedef typename boost::detail::array_inner<T>::type T1;
160 typedef typename boost::detail::array_base<T1>::type T2;
161 typedef boost::detail::ms_noinit_tag R1;
162 typedef boost::detail::as_allocator<A, T, R1> A1;
163 typedef boost::detail::ms_in_allocator_tag D1;
165 N = boost::detail::array_total<T>::size
170 A1 a1(allocator, &p2);
171 shared_ptr<T> s1(p1, d1, a1);
172 A1* a2 = static_cast<A1*>(s1._internal_get_untyped_deleter());
174 boost::detail::ms_noinit(p2, N);
176 p1 = reinterpret_cast<T1*>(p2);
177 return shared_ptr<T>(s1, p1);