Chris@16
|
1 //////////////////////////////////////////////////////////////////////////////
|
Chris@16
|
2 //
|
Chris@101
|
3 // (C) Copyright Ion Gaztanaga 2005-2013. Distributed under the Boost
|
Chris@16
|
4 // Software License, Version 1.0. (See accompanying file
|
Chris@16
|
5 // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
Chris@16
|
6 //
|
Chris@16
|
7 // See http://www.boost.org/libs/container for documentation.
|
Chris@16
|
8 //
|
Chris@16
|
9 //////////////////////////////////////////////////////////////////////////////
|
Chris@16
|
10
|
Chris@16
|
11 #ifndef BOOST_CONTAINER_DETAIL_MULTIALLOCATION_CHAIN_HPP
|
Chris@16
|
12 #define BOOST_CONTAINER_DETAIL_MULTIALLOCATION_CHAIN_HPP
|
Chris@16
|
13
|
Chris@101
|
14 #ifndef BOOST_CONFIG_HPP
|
Chris@101
|
15 # include <boost/config.hpp>
|
Chris@101
|
16 #endif
|
Chris@101
|
17
|
Chris@101
|
18 #if defined(BOOST_HAS_PRAGMA_ONCE)
|
Chris@101
|
19 # pragma once
|
Chris@101
|
20 #endif
|
Chris@101
|
21
|
Chris@101
|
22 #include <boost/container/detail/config_begin.hpp>
|
Chris@101
|
23 #include <boost/container/detail/workaround.hpp>
|
Chris@101
|
24 // container
|
Chris@16
|
25 #include <boost/container/container_fwd.hpp>
|
Chris@101
|
26 // container/detail
|
Chris@101
|
27 #include <boost/container/detail/to_raw_pointer.hpp>
|
Chris@101
|
28 #include <boost/container/detail/transform_iterator.hpp>
|
Chris@16
|
29 #include <boost/container/detail/type_traits.hpp>
|
Chris@101
|
30 // intrusive
|
Chris@16
|
31 #include <boost/intrusive/slist.hpp>
|
Chris@16
|
32 #include <boost/intrusive/pointer_traits.hpp>
|
Chris@101
|
33 // move
|
Chris@101
|
34 #include <boost/move/utility_core.hpp>
|
Chris@16
|
35
|
Chris@16
|
36 namespace boost {
|
Chris@16
|
37 namespace container {
|
Chris@16
|
38 namespace container_detail {
|
Chris@16
|
39
|
Chris@16
|
40 template<class VoidPointer>
|
Chris@16
|
41 class basic_multiallocation_chain
|
Chris@16
|
42 {
|
Chris@16
|
43 private:
|
Chris@16
|
44 typedef bi::slist_base_hook<bi::void_pointer<VoidPointer>
|
Chris@16
|
45 ,bi::link_mode<bi::normal_link>
|
Chris@16
|
46 > node;
|
Chris@16
|
47
|
Chris@16
|
48 typedef typename boost::intrusive::pointer_traits
|
Chris@16
|
49 <VoidPointer>::template rebind_pointer<char>::type char_ptr;
|
Chris@16
|
50 typedef typename boost::intrusive::
|
Chris@16
|
51 pointer_traits<char_ptr>::difference_type difference_type;
|
Chris@16
|
52
|
Chris@16
|
53 typedef bi::slist< node
|
Chris@16
|
54 , bi::linear<true>
|
Chris@16
|
55 , bi::cache_last<true>
|
Chris@101
|
56 , bi::size_type<typename boost::container::container_detail::make_unsigned<difference_type>::type>
|
Chris@16
|
57 > slist_impl_t;
|
Chris@16
|
58 slist_impl_t slist_impl_;
|
Chris@16
|
59
|
Chris@16
|
60 typedef typename boost::intrusive::pointer_traits
|
Chris@16
|
61 <VoidPointer>::template rebind_pointer<node>::type node_ptr;
|
Chris@16
|
62 typedef typename boost::intrusive::
|
Chris@16
|
63 pointer_traits<node_ptr> node_ptr_traits;
|
Chris@16
|
64
|
Chris@16
|
65 static node & to_node(const VoidPointer &p)
|
Chris@16
|
66 { return *static_cast<node*>(static_cast<void*>(container_detail::to_raw_pointer(p))); }
|
Chris@16
|
67
|
Chris@16
|
68 static VoidPointer from_node(node &n)
|
Chris@16
|
69 { return node_ptr_traits::pointer_to(n); }
|
Chris@16
|
70
|
Chris@16
|
71 static node_ptr to_node_ptr(const VoidPointer &p)
|
Chris@16
|
72 { return node_ptr_traits::static_cast_from(p); }
|
Chris@16
|
73
|
Chris@16
|
74 BOOST_MOVABLE_BUT_NOT_COPYABLE(basic_multiallocation_chain)
|
Chris@16
|
75
|
Chris@16
|
76 public:
|
Chris@16
|
77
|
Chris@16
|
78 typedef VoidPointer void_pointer;
|
Chris@16
|
79 typedef typename slist_impl_t::iterator iterator;
|
Chris@16
|
80 typedef typename slist_impl_t::size_type size_type;
|
Chris@16
|
81
|
Chris@16
|
82 basic_multiallocation_chain()
|
Chris@16
|
83 : slist_impl_()
|
Chris@16
|
84 {}
|
Chris@16
|
85
|
Chris@16
|
86 basic_multiallocation_chain(const void_pointer &b, const void_pointer &before_e, size_type n)
|
Chris@16
|
87 : slist_impl_(to_node_ptr(b), to_node_ptr(before_e), n)
|
Chris@16
|
88 {}
|
Chris@16
|
89
|
Chris@16
|
90 basic_multiallocation_chain(BOOST_RV_REF(basic_multiallocation_chain) other)
|
Chris@16
|
91 : slist_impl_(::boost::move(other.slist_impl_))
|
Chris@16
|
92 {}
|
Chris@16
|
93
|
Chris@16
|
94 basic_multiallocation_chain& operator=(BOOST_RV_REF(basic_multiallocation_chain) other)
|
Chris@16
|
95 {
|
Chris@16
|
96 slist_impl_ = ::boost::move(other.slist_impl_);
|
Chris@16
|
97 return *this;
|
Chris@16
|
98 }
|
Chris@16
|
99
|
Chris@16
|
100 bool empty() const
|
Chris@16
|
101 { return slist_impl_.empty(); }
|
Chris@16
|
102
|
Chris@16
|
103 size_type size() const
|
Chris@16
|
104 { return slist_impl_.size(); }
|
Chris@16
|
105
|
Chris@16
|
106 iterator before_begin()
|
Chris@16
|
107 { return slist_impl_.before_begin(); }
|
Chris@16
|
108
|
Chris@16
|
109 iterator begin()
|
Chris@16
|
110 { return slist_impl_.begin(); }
|
Chris@16
|
111
|
Chris@16
|
112 iterator end()
|
Chris@16
|
113 { return slist_impl_.end(); }
|
Chris@16
|
114
|
Chris@16
|
115 iterator last()
|
Chris@16
|
116 { return slist_impl_.last(); }
|
Chris@16
|
117
|
Chris@16
|
118 void clear()
|
Chris@16
|
119 { slist_impl_.clear(); }
|
Chris@16
|
120
|
Chris@16
|
121 iterator insert_after(iterator it, void_pointer m)
|
Chris@16
|
122 { return slist_impl_.insert_after(it, to_node(m)); }
|
Chris@16
|
123
|
Chris@16
|
124 void push_front(const void_pointer &m)
|
Chris@16
|
125 { return slist_impl_.push_front(to_node(m)); }
|
Chris@16
|
126
|
Chris@16
|
127 void push_back(const void_pointer &m)
|
Chris@16
|
128 { return slist_impl_.push_back(to_node(m)); }
|
Chris@16
|
129
|
Chris@16
|
130 void_pointer pop_front()
|
Chris@16
|
131 {
|
Chris@16
|
132 node & n = slist_impl_.front();
|
Chris@16
|
133 void_pointer ret = from_node(n);
|
Chris@16
|
134 slist_impl_.pop_front();
|
Chris@16
|
135 return ret;
|
Chris@16
|
136 }
|
Chris@16
|
137
|
Chris@16
|
138 void splice_after(iterator after_this, basic_multiallocation_chain &x, iterator before_b, iterator before_e, size_type n)
|
Chris@16
|
139 { slist_impl_.splice_after(after_this, x.slist_impl_, before_b, before_e, n); }
|
Chris@16
|
140
|
Chris@16
|
141 void splice_after(iterator after_this, basic_multiallocation_chain &x)
|
Chris@16
|
142 { slist_impl_.splice_after(after_this, x.slist_impl_); }
|
Chris@16
|
143
|
Chris@16
|
144 void erase_after(iterator before_b, iterator e, size_type n)
|
Chris@16
|
145 { slist_impl_.erase_after(before_b, e, n); }
|
Chris@16
|
146
|
Chris@16
|
147 void_pointer incorporate_after(iterator after_this, const void_pointer &b, size_type unit_bytes, size_type num_units)
|
Chris@16
|
148 {
|
Chris@16
|
149 typedef typename boost::intrusive::pointer_traits<char_ptr> char_pointer_traits;
|
Chris@16
|
150 char_ptr elem = char_pointer_traits::static_cast_from(b);
|
Chris@16
|
151 if(num_units){
|
Chris@16
|
152 char_ptr prev_elem = elem;
|
Chris@16
|
153 elem += unit_bytes;
|
Chris@16
|
154 for(size_type i = 0; i != num_units-1; ++i, elem += unit_bytes){
|
Chris@16
|
155 ::new (container_detail::to_raw_pointer(prev_elem)) void_pointer(elem);
|
Chris@16
|
156 prev_elem = elem;
|
Chris@16
|
157 }
|
Chris@16
|
158 slist_impl_.incorporate_after(after_this, to_node_ptr(b), to_node_ptr(prev_elem), num_units);
|
Chris@16
|
159 }
|
Chris@16
|
160 return elem;
|
Chris@16
|
161 }
|
Chris@16
|
162
|
Chris@16
|
163 void incorporate_after(iterator after_this, void_pointer b, void_pointer before_e, size_type n)
|
Chris@16
|
164 { slist_impl_.incorporate_after(after_this, to_node_ptr(b), to_node_ptr(before_e), n); }
|
Chris@16
|
165
|
Chris@16
|
166 void swap(basic_multiallocation_chain &x)
|
Chris@16
|
167 { slist_impl_.swap(x.slist_impl_); }
|
Chris@16
|
168
|
Chris@16
|
169 static iterator iterator_to(const void_pointer &p)
|
Chris@16
|
170 { return slist_impl_t::s_iterator_to(to_node(p)); }
|
Chris@16
|
171
|
Chris@16
|
172 std::pair<void_pointer, void_pointer> extract_data()
|
Chris@16
|
173 {
|
Chris@16
|
174 std::pair<void_pointer, void_pointer> ret
|
Chris@16
|
175 (slist_impl_.begin().operator->()
|
Chris@16
|
176 ,slist_impl_.last().operator->());
|
Chris@16
|
177 slist_impl_.clear();
|
Chris@16
|
178 return ret;
|
Chris@16
|
179 }
|
Chris@16
|
180 };
|
Chris@16
|
181
|
Chris@16
|
182 template<class T>
|
Chris@16
|
183 struct cast_functor
|
Chris@16
|
184 {
|
Chris@16
|
185 typedef typename container_detail::add_reference<T>::type result_type;
|
Chris@16
|
186 template<class U>
|
Chris@16
|
187 result_type operator()(U &ptr) const
|
Chris@16
|
188 { return *static_cast<T*>(static_cast<void*>(&ptr)); }
|
Chris@16
|
189 };
|
Chris@16
|
190
|
Chris@16
|
191 template<class MultiallocationChain, class T>
|
Chris@16
|
192 class transform_multiallocation_chain
|
Chris@16
|
193 : public MultiallocationChain
|
Chris@16
|
194 {
|
Chris@16
|
195 private:
|
Chris@16
|
196 BOOST_MOVABLE_BUT_NOT_COPYABLE(transform_multiallocation_chain)
|
Chris@16
|
197 //transform_multiallocation_chain(const transform_multiallocation_chain &);
|
Chris@16
|
198 //transform_multiallocation_chain & operator=(const transform_multiallocation_chain &);
|
Chris@16
|
199
|
Chris@16
|
200 typedef typename MultiallocationChain::void_pointer void_pointer;
|
Chris@16
|
201 typedef typename boost::intrusive::pointer_traits
|
Chris@16
|
202 <void_pointer> void_pointer_traits;
|
Chris@16
|
203 typedef typename void_pointer_traits::template
|
Chris@16
|
204 rebind_pointer<T>::type pointer;
|
Chris@16
|
205 typedef typename boost::intrusive::pointer_traits
|
Chris@16
|
206 <pointer> pointer_traits;
|
Chris@16
|
207
|
Chris@16
|
208 static pointer cast(const void_pointer &p)
|
Chris@16
|
209 { return pointer_traits::static_cast_from(p); }
|
Chris@16
|
210
|
Chris@16
|
211 public:
|
Chris@16
|
212 typedef transform_iterator
|
Chris@16
|
213 < typename MultiallocationChain::iterator
|
Chris@16
|
214 , container_detail::cast_functor <T> > iterator;
|
Chris@16
|
215 typedef typename MultiallocationChain::size_type size_type;
|
Chris@16
|
216
|
Chris@16
|
217 transform_multiallocation_chain()
|
Chris@16
|
218 : MultiallocationChain()
|
Chris@16
|
219 {}
|
Chris@16
|
220
|
Chris@16
|
221 transform_multiallocation_chain(BOOST_RV_REF(transform_multiallocation_chain) other)
|
Chris@16
|
222 : MultiallocationChain(::boost::move(static_cast<MultiallocationChain&>(other)))
|
Chris@16
|
223 {}
|
Chris@16
|
224
|
Chris@16
|
225 transform_multiallocation_chain(BOOST_RV_REF(MultiallocationChain) other)
|
Chris@16
|
226 : MultiallocationChain(::boost::move(static_cast<MultiallocationChain&>(other)))
|
Chris@16
|
227 {}
|
Chris@16
|
228
|
Chris@16
|
229 transform_multiallocation_chain& operator=(BOOST_RV_REF(transform_multiallocation_chain) other)
|
Chris@16
|
230 {
|
Chris@16
|
231 return static_cast<MultiallocationChain&>
|
Chris@16
|
232 (this->MultiallocationChain::operator=(::boost::move(static_cast<MultiallocationChain&>(other))));
|
Chris@16
|
233 }
|
Chris@16
|
234 /*
|
Chris@16
|
235 void push_front(const pointer &mem)
|
Chris@16
|
236 { holder_.push_front(mem); }
|
Chris@16
|
237
|
Chris@16
|
238 void push_back(const pointer &mem)
|
Chris@16
|
239 { return holder_.push_back(mem); }
|
Chris@16
|
240
|
Chris@16
|
241 void swap(transform_multiallocation_chain &other_chain)
|
Chris@16
|
242 { holder_.swap(other_chain.holder_); }
|
Chris@16
|
243
|
Chris@16
|
244 void splice_after(iterator after_this, transform_multiallocation_chain &x, iterator before_b, iterator before_e, size_type n)
|
Chris@16
|
245 { holder_.splice_after(after_this.base(), x.holder_, before_b.base(), before_e.base(), n); }
|
Chris@16
|
246
|
Chris@16
|
247 void incorporate_after(iterator after_this, pointer b, pointer before_e, size_type n)
|
Chris@16
|
248 { holder_.incorporate_after(after_this.base(), b, before_e, n); }
|
Chris@16
|
249 */
|
Chris@16
|
250 pointer pop_front()
|
Chris@16
|
251 { return cast(this->MultiallocationChain::pop_front()); }
|
Chris@16
|
252 /*
|
Chris@16
|
253 bool empty() const
|
Chris@16
|
254 { return holder_.empty(); }
|
Chris@16
|
255
|
Chris@16
|
256 iterator before_begin()
|
Chris@16
|
257 { return iterator(holder_.before_begin()); }
|
Chris@16
|
258 */
|
Chris@16
|
259 iterator begin()
|
Chris@16
|
260 { return iterator(this->MultiallocationChain::begin()); }
|
Chris@16
|
261 /*
|
Chris@16
|
262 iterator end()
|
Chris@16
|
263 { return iterator(holder_.end()); }
|
Chris@16
|
264
|
Chris@16
|
265 iterator last()
|
Chris@16
|
266 { return iterator(holder_.last()); }
|
Chris@16
|
267
|
Chris@16
|
268 size_type size() const
|
Chris@16
|
269 { return holder_.size(); }
|
Chris@16
|
270
|
Chris@16
|
271 void clear()
|
Chris@16
|
272 { holder_.clear(); }
|
Chris@16
|
273 */
|
Chris@16
|
274 iterator insert_after(iterator it, pointer m)
|
Chris@16
|
275 { return iterator(this->MultiallocationChain::insert_after(it.base(), m)); }
|
Chris@16
|
276
|
Chris@16
|
277 static iterator iterator_to(const pointer &p)
|
Chris@16
|
278 { return iterator(MultiallocationChain::iterator_to(p)); }
|
Chris@16
|
279
|
Chris@16
|
280 std::pair<pointer, pointer> extract_data()
|
Chris@16
|
281 {
|
Chris@16
|
282 std::pair<void_pointer, void_pointer> data(this->MultiallocationChain::extract_data());
|
Chris@16
|
283 return std::pair<pointer, pointer>(cast(data.first), cast(data.second));
|
Chris@16
|
284 }
|
Chris@16
|
285 /*
|
Chris@16
|
286 MultiallocationChain &extract_multiallocation_chain()
|
Chris@16
|
287 { return holder_; }*/
|
Chris@16
|
288 };
|
Chris@16
|
289
|
Chris@16
|
290 }}}
|
Chris@16
|
291
|
Chris@16
|
292 // namespace container_detail {
|
Chris@16
|
293 // namespace container {
|
Chris@16
|
294 // namespace boost {
|
Chris@16
|
295
|
Chris@16
|
296 #include <boost/container/detail/config_end.hpp>
|
Chris@16
|
297
|
Chris@16
|
298 #endif //BOOST_CONTAINER_DETAIL_MULTIALLOCATION_CHAIN_HPP
|