]> git.proxmox.com Git - ceph.git/blame - ceph/src/boost/libs/icl/include/boost/icl/detail/element_iterator.hpp
bump version to 12.2.2-pve1
[ceph.git] / ceph / src / boost / libs / icl / include / boost / icl / detail / element_iterator.hpp
CommitLineData
7c673cae
FG
1/*-----------------------------------------------------------------------------+
2Copyright (c) 2009-2009: Joachim Faulhaber
3+------------------------------------------------------------------------------+
4 Distributed under the Boost Software License, Version 1.0.
5 (See accompanying file LICENCE.txt or copy at
6 http://www.boost.org/LICENSE_1_0.txt)
7+-----------------------------------------------------------------------------*/
8#ifndef BOOST_ICL_DETAIL_ELEMENT_ITERATOR_HPP_JOFA_091104
9#define BOOST_ICL_DETAIL_ELEMENT_ITERATOR_HPP_JOFA_091104
10
11#include <boost/mpl/if.hpp>
12#include <boost/iterator/iterator_facade.hpp>
13#include <boost/config/warning_disable.hpp>
14#include <boost/icl/type_traits/succ_pred.hpp>
15#include <boost/icl/detail/mapped_reference.hpp>
16
17namespace boost{namespace icl
18{
19
20//------------------------------------------------------------------------------
21template<class Type>
22struct is_std_pair
23{
24 typedef is_std_pair<Type> type;
25 BOOST_STATIC_CONSTANT(bool, value = false);
26};
27
28template<class FirstT, class SecondT>
29struct is_std_pair<std::pair<FirstT, SecondT> >
30{
31 typedef is_std_pair<std::pair<FirstT, SecondT> > type;
32 BOOST_STATIC_CONSTANT(bool, value = true);
33};
34
35
36//------------------------------------------------------------------------------
37template<class Type>
38struct first_element
39{
40 typedef Type type;
41};
42
43template<class FirstT, class SecondT>
44struct first_element<std::pair<FirstT, SecondT> >
45{
46 typedef FirstT type;
47};
48
49//------------------------------------------------------------------------------
50template <class SegmentIteratorT> class element_iterator;
51
52template<class IteratorT>
53struct is_reverse
54{
55 typedef is_reverse type;
56 BOOST_STATIC_CONSTANT(bool, value = false);
57};
58
59template<class BaseIteratorT>
60struct is_reverse<std::reverse_iterator<BaseIteratorT> >
61{
62 typedef is_reverse<std::reverse_iterator<BaseIteratorT> > type;
63 BOOST_STATIC_CONSTANT(bool, value = true);
64};
65
66template<class BaseIteratorT>
67struct is_reverse<icl::element_iterator<BaseIteratorT> >
68{
69 typedef is_reverse<icl::element_iterator<BaseIteratorT> > type;
70 BOOST_STATIC_CONSTANT(bool, value = is_reverse<BaseIteratorT>::value);
71};
72
73//------------------------------------------------------------------------------
74template<class SegmentT>
75struct elemental;
76
77#ifdef ICL_USE_INTERVAL_TEMPLATE_TEMPLATE
78
79 template<class DomainT, ICL_COMPARE Compare, ICL_INTERVAL(ICL_COMPARE) Interval>
80 struct elemental<ICL_INTERVAL_TYPE(Interval,DomainT,Compare) >
81 {
82 typedef ICL_INTERVAL_TYPE(Interval,DomainT,Compare) segment_type;
83 typedef segment_type interval_type;
84 typedef DomainT type;
85 typedef DomainT domain_type;
86 typedef DomainT codomain_type;
87 typedef DomainT transit_type;
88 };
89
90 template< class DomainT, class CodomainT,
91 ICL_COMPARE Compare, ICL_INTERVAL(ICL_COMPARE) Interval >
92 struct elemental<std::pair<ICL_INTERVAL_TYPE(Interval,DomainT,Compare)const, CodomainT> >
93 {
94 typedef std::pair<ICL_INTERVAL_TYPE(Interval,DomainT,Compare), CodomainT> segment_type;
95 typedef ICL_INTERVAL_TYPE(Interval,DomainT,Compare) interval_type;
96 typedef std::pair<DomainT, CodomainT> type;
97 typedef DomainT domain_type;
98 typedef CodomainT codomain_type;
99 typedef mapped_reference<DomainT, CodomainT> transit_type;
100 };
101
102#else //ICL_USE_INTERVAL_TEMPLATE_TYPE
103
104 template<ICL_INTERVAL(ICL_COMPARE) Interval>
105 struct elemental
106 {
107 typedef ICL_INTERVAL_TYPE(Interval,DomainT,Compare) segment_type;
108 typedef segment_type interval_type;
109 typedef typename interval_traits<interval_type>::domain_type domain_type;
110 typedef domain_type type;
111 typedef domain_type codomain_type;
112 typedef domain_type transit_type;
113 };
114
115 template< class CodomainT, ICL_INTERVAL(ICL_COMPARE) Interval >
116 struct elemental<std::pair<ICL_INTERVAL_TYPE(Interval,DomainT,Compare)const, CodomainT> >
117 {
118 typedef std::pair<ICL_INTERVAL_TYPE(Interval,DomainT,Compare), CodomainT> segment_type;
119 typedef ICL_INTERVAL_TYPE(Interval,DomainT,Compare) interval_type;
120 typedef typename interval_traits<interval_type>::domain_type domain_type;
121 typedef CodomainT codomain_type;
122 typedef std::pair<domain_type, codomain_type> type;
123 typedef mapped_reference<domain_type, codomain_type> transit_type;
124 };
125
126#endif //ICL_USE_INTERVAL_TEMPLATE_TEMPLATE
127
128
129//------------------------------------------------------------------------------
130//- struct segment_adapter
131//------------------------------------------------------------------------------
132template<class SegmentIteratorT, class SegmentT>
133struct segment_adapter;
134
135#ifdef ICL_USE_INTERVAL_TEMPLATE_TEMPLATE
136
137template<class SegmentIteratorT, class DomainT, ICL_COMPARE Compare, ICL_INTERVAL(ICL_COMPARE) Interval>
138struct segment_adapter<SegmentIteratorT, ICL_INTERVAL_TYPE(Interval,DomainT,Compare) >
139{
140 typedef segment_adapter type;
141 typedef ICL_INTERVAL_TYPE(Interval,DomainT,Compare) segment_type;
142 typedef segment_type interval_type;
143 typedef typename interval_type::difference_type domain_difference_type;
144 typedef DomainT domain_type;
145 typedef DomainT codomain_type;
146 typedef domain_type element_type;
147 typedef domain_type& transit_type;
148
149 static domain_type first (const SegmentIteratorT& leaper){ return leaper->first(); }
150 static domain_type last (const SegmentIteratorT& leaper){ return leaper->last(); }
151 static domain_difference_type length(const SegmentIteratorT& leaper){ return leaper->length();}
152
153 static transit_type transient_element(domain_type& inter_pos, const SegmentIteratorT& leaper,
154 const domain_difference_type& sneaker)
155 {
156 inter_pos = is_reverse<SegmentIteratorT>::value ? leaper->last() - sneaker
157 : leaper->first() + sneaker;
158 return inter_pos;
159 }
160};
161
162template < class SegmentIteratorT, class DomainT, class CodomainT,
163 ICL_COMPARE Compare, ICL_INTERVAL(ICL_COMPARE) Interval >
164struct segment_adapter<SegmentIteratorT, std::pair<ICL_INTERVAL_TYPE(Interval,DomainT,Compare)const, CodomainT> >
165{
166 typedef segment_adapter type;
167 typedef ICL_INTERVAL_TYPE(Interval,DomainT,Compare) interval_type;
168 typedef DomainT domain_type;
169 typedef std::pair<DomainT, CodomainT> element_type;
170 typedef CodomainT codomain_type;
171 typedef mapped_reference<DomainT, CodomainT> transit_type;
172 typedef typename difference_type_of<interval_traits<interval_type> >::type
173 domain_difference_type;
174
175 static domain_type first (const SegmentIteratorT& leaper){ return leaper->first.first(); }
176 static domain_type last (const SegmentIteratorT& leaper){ return leaper->first.last(); }
177 static domain_difference_type length(const SegmentIteratorT& leaper){ return leaper->first.length();}
178
179 static transit_type transient_element(domain_type& inter_pos, const SegmentIteratorT& leaper,
180 const domain_difference_type& sneaker)
181 {
182 inter_pos = is_reverse<SegmentIteratorT>::value ? leaper->first.last() - sneaker
183 : leaper->first.first() + sneaker;
184 return transit_type(inter_pos, leaper->second);
185 }
186};
187
188#else // ICL_USE_INTERVAL_TEMPLATE_TYPE
189
190template<class SegmentIteratorT, ICL_INTERVAL(ICL_COMPARE) Interval>
191struct segment_adapter
192{
193 typedef segment_adapter type;
194 typedef ICL_INTERVAL_TYPE(Interval,DomainT,Compare) segment_type;
195 typedef segment_type interval_type;
196 typedef typename interval_traits<interval_type>::domain_type domain_type;
197 typedef domain_type codomain_type;
198 typedef domain_type element_type;
199 typedef domain_type& transit_type;
200 typedef typename difference_type_of<interval_traits<interval_type> >::type
201 domain_difference_type;
202
203 static domain_type first (const SegmentIteratorT& leaper){ return leaper->first(); }
204 static domain_type last (const SegmentIteratorT& leaper){ return leaper->last(); }
205 static domain_difference_type length(const SegmentIteratorT& leaper){ return icl::length(*leaper);}
206
207 static transit_type transient_element(domain_type& inter_pos, const SegmentIteratorT& leaper,
208 const domain_difference_type& sneaker)
209 {
210 inter_pos = is_reverse<SegmentIteratorT>::value ? icl::last(*leaper) - sneaker
211 : icl::first(*leaper) + sneaker;
212 return inter_pos;
213 }
214};
215
216template < class SegmentIteratorT, class CodomainT, ICL_INTERVAL(ICL_COMPARE) Interval >
217struct segment_adapter<SegmentIteratorT, std::pair<ICL_INTERVAL_TYPE(Interval,DomainT,Compare)const, CodomainT> >
218{
219 typedef segment_adapter type;
220 typedef ICL_INTERVAL_TYPE(Interval,DomainT,Compare) interval_type;
221 typedef typename interval_traits<interval_type>::domain_type domain_type;
222 typedef CodomainT codomain_type;
223 typedef std::pair<domain_type, codomain_type> element_type;
224 typedef mapped_reference<domain_type, CodomainT> transit_type;
225 typedef typename difference_type_of<interval_traits<interval_type> >::type
226 domain_difference_type;
227
228 static domain_type first (const SegmentIteratorT& leaper){ return leaper->first.first(); }
229 static domain_type last (const SegmentIteratorT& leaper){ return leaper->first.last(); }
230 static domain_difference_type length(const SegmentIteratorT& leaper){ return icl::length(leaper->first);}
231
232 static transit_type transient_element(domain_type& inter_pos, const SegmentIteratorT& leaper,
233 const domain_difference_type& sneaker)
234 {
235 inter_pos = is_reverse<SegmentIteratorT>::value ? icl::last(leaper->first) - sneaker
236 : icl::first(leaper->first) + sneaker;
237 return transit_type(inter_pos, leaper->second);
238 }
239};
240
241#endif // ICL_USE_INTERVAL_TEMPLATE_TEMPLATE
242
243template <class SegmentIteratorT>
244class element_iterator
245 : public boost::iterator_facade<
246 element_iterator<SegmentIteratorT>
247 , typename elemental<typename SegmentIteratorT::value_type>::transit_type
248 , boost::bidirectional_traversal_tag
249 , typename elemental<typename SegmentIteratorT::value_type>::transit_type
250 >
251{
252public:
253 typedef element_iterator type;
254 typedef SegmentIteratorT segment_iterator;
255 typedef typename SegmentIteratorT::value_type segment_type;
256 typedef typename first_element<segment_type>::type interval_type;
257 typedef typename elemental<segment_type>::type element_type;
258 typedef typename elemental<segment_type>::domain_type domain_type;
259 typedef typename elemental<segment_type>::codomain_type codomain_type;
260 typedef typename elemental<segment_type>::transit_type transit_type;
261 typedef transit_type value_type;
262 typedef typename difference_type_of<interval_traits<interval_type> >::type
263 domain_difference_type;
264
265private:
266 typedef typename segment_adapter<segment_iterator,segment_type>::type adapt;
267
268 struct enabler{};
269
270public:
271 element_iterator()
272 : _saltator(identity_element<segment_iterator>::value())
273 , _reptator(identity_element<domain_difference_type>::value()){}
274
275 explicit element_iterator(segment_iterator jumper)
276 : _saltator(jumper), _reptator(identity_element<domain_difference_type>::value()) {}
277
278 template <class SaltatorT>
279 element_iterator
280 ( element_iterator<SaltatorT> const& other
281 , typename enable_if<boost::is_convertible<SaltatorT*,SegmentIteratorT*>, enabler>::type = enabler())
282 : _saltator(other._saltator), _reptator(other._reptator) {}
283
284private:
285 friend class boost::iterator_core_access;
286 template <class> friend class element_iterator;
287
288 template <class SaltatorT>
289 bool equal(element_iterator<SaltatorT> const& other) const
290 {
291 return this->_saltator == other._saltator
292 && this->_reptator == other._reptator;
293 }
294
295 void increment()
296 {
297 if(_reptator < icl::pred(adapt::length(_saltator)))
298 ++_reptator;
299 else
300 {
301 ++_saltator;
302 _reptator = identity_element<domain_difference_type>::value();
303 }
304 }
305
306 void decrement()
307 {
308 if(identity_element<domain_difference_type>::value() < _reptator)
309 --_reptator;
310 else
311 {
312 --_saltator;
313 _reptator = adapt::length(_saltator);
314 --_reptator;
315 }
316 }
317
318 value_type dereference()const
319 {
320 return adapt::transient_element(_inter_pos, _saltator, _reptator);
321 }
322
323private:
324 segment_iterator _saltator; // satltare: to jump : the fast moving iterator
325 mutable domain_difference_type _reptator; // reptare: to sneak : the slow moving iterator 0 based
326 mutable domain_type _inter_pos; // inter position : Position within the current segment
327 // _saltator->first.first() <= _inter_pos <= _saltator->first.last()
328};
329
330}} // namespace icl boost
331
332#endif // BOOST_ICL_DETAIL_ELEMENT_ITERATOR_HPP_JOFA_091104
333
334
335