1 // (C) Copyright Jeremy Siek 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)
6 #ifndef BOOST_ITERATOR_CONCEPTS_HPP
7 #define BOOST_ITERATOR_CONCEPTS_HPP
11 // Adapted to new iterator concepts
12 // 22 Nov 2002 Thomas Witt
13 // Added interoperable concept.
15 #include <boost/concept_check.hpp>
16 #include <boost/iterator/iterator_categories.hpp>
18 // Use boost::detail::iterator_traits to work around some MSVC/Dinkumware problems.
19 #include <boost/detail/iterator.hpp>
21 #include <boost/type_traits/is_same.hpp>
22 #include <boost/type_traits/is_integral.hpp>
23 #include <boost/type_traits/is_convertible.hpp>
25 #include <boost/mpl/bool.hpp>
26 #include <boost/mpl/if.hpp>
27 #include <boost/mpl/and.hpp>
28 #include <boost/mpl/or.hpp>
30 #include <boost/static_assert.hpp>
32 // Use boost/limits to work around missing limits headers on some compilers
33 #include <boost/limits.hpp>
34 #include <boost/config.hpp>
38 namespace boost_concepts {
39 // Used a different namespace here (instead of "boost") so that the
40 // concept descriptions do not take for granted the names in
43 // We use this in place of STATIC_ASSERT((is_convertible<...>))
44 // because some compilers (CWPro7.x) can't detect convertibility.
46 // Of course, that just gets us a different error at the moment with
47 // some tests, since new iterator category deduction still depends
48 // on convertibility detection. We might need some specializations
49 // to support this compiler.
50 template <class Target, class Source>
51 struct static_assert_base_and_derived
53 static_assert_base_and_derived(Target* = (Source*)0) {}
56 //===========================================================================
57 // Iterator Access Concepts
59 template <typename Iterator>
60 class ReadableIteratorConcept {
62 typedef BOOST_DEDUCED_TYPENAME boost::detail::iterator_traits<Iterator>::value_type value_type;
65 boost::function_requires< boost::AssignableConcept<Iterator> >();
66 boost::function_requires< boost::CopyConstructibleConcept<Iterator> >();
69 boost::ignore_unused_variable_warning(v);
76 , typename ValueType = BOOST_DEDUCED_TYPENAME boost::detail::iterator_traits<Iterator>::value_type
78 class WritableIteratorConcept {
82 boost::function_requires< boost::CopyConstructibleConcept<Iterator> >();
89 template <typename Iterator>
90 class SwappableIteratorConcept {
94 std::iter_swap(i1, i2);
100 template <typename Iterator>
101 class LvalueIteratorConcept
104 typedef typename boost::detail::iterator_traits<Iterator>::value_type value_type;
107 value_type& r = const_cast<value_type&>(*i);
108 boost::ignore_unused_variable_warning(r);
114 //===========================================================================
115 // Iterator Traversal Concepts
117 template <typename Iterator>
118 class IncrementableIteratorConcept {
120 typedef typename boost::iterator_traversal<Iterator>::type traversal_category;
123 boost::function_requires< boost::AssignableConcept<Iterator> >();
124 boost::function_requires< boost::CopyConstructibleConcept<Iterator> >();
127 (boost::is_convertible<
129 , boost::incrementable_traversal_tag
139 template <typename Iterator>
140 class SinglePassIteratorConcept {
142 typedef typename boost::iterator_traversal<Iterator>::type traversal_category;
143 typedef typename boost::detail::iterator_traits<Iterator>::difference_type difference_type;
146 boost::function_requires< IncrementableIteratorConcept<Iterator> >();
147 boost::function_requires< boost::EqualityComparableConcept<Iterator> >();
150 (boost::is_convertible<
152 , boost::single_pass_traversal_tag
158 template <typename Iterator>
159 class ForwardTraversalConcept {
161 typedef typename boost::iterator_traversal<Iterator>::type traversal_category;
162 typedef typename boost::detail::iterator_traits<Iterator>::difference_type difference_type;
165 boost::function_requires< SinglePassIteratorConcept<Iterator> >();
166 boost::function_requires<
167 boost::DefaultConstructibleConcept<Iterator> >();
169 typedef boost::mpl::and_<
170 boost::is_integral<difference_type>,
171 boost::mpl::bool_< std::numeric_limits<difference_type>::is_signed >
172 > difference_type_is_signed_integral;
174 BOOST_STATIC_ASSERT(difference_type_is_signed_integral::value);
176 (boost::is_convertible<
178 , boost::forward_traversal_tag
184 template <typename Iterator>
185 class BidirectionalTraversalConcept {
187 typedef typename boost::iterator_traversal<Iterator>::type traversal_category;
190 boost::function_requires< ForwardTraversalConcept<Iterator> >();
193 (boost::is_convertible<
195 , boost::bidirectional_traversal_tag
205 template <typename Iterator>
206 class RandomAccessTraversalConcept {
208 typedef typename boost::iterator_traversal<Iterator>::type traversal_category;
209 typedef typename boost::detail::iterator_traits<Iterator>::difference_type
213 boost::function_requires< BidirectionalTraversalConcept<Iterator> >();
216 (boost::is_convertible<
218 , boost::random_access_traversal_tag
233 //===========================================================================
234 // Iterator Interoperability Concept
239 template <typename Iterator1, typename Iterator2>
240 void interop_single_pass_constraints(Iterator1 const& i1, Iterator2 const& i2)
250 template <typename Iterator1, typename Iterator2>
251 void interop_rand_access_constraints(Iterator1 const& i1, Iterator2 const& i2,
252 boost::random_access_traversal_tag, boost::random_access_traversal_tag)
255 typename boost::detail::iterator_traits<Iterator2>::difference_type n;
268 template <typename Iterator1, typename Iterator2>
269 void interop_rand_access_constraints(Iterator1 const& i1, Iterator2 const& i2,
270 boost::single_pass_traversal_tag, boost::single_pass_traversal_tag)
273 } // namespace detail
275 template <typename Iterator, typename ConstIterator>
276 class InteroperableIteratorConcept
279 typedef typename boost::detail::pure_traversal_tag<
280 typename boost::iterator_traversal<
283 >::type traversal_category;
285 typedef typename boost::detail::pure_traversal_tag<
286 typename boost::iterator_traversal<
289 >::type const_traversal_category;
293 boost::function_requires< SinglePassIteratorConcept<Iterator> >();
294 boost::function_requires< SinglePassIteratorConcept<ConstIterator> >();
296 detail::interop_single_pass_constraints(i, ci);
297 detail::interop_rand_access_constraints(i, ci, traversal_category(), const_traversal_category());
305 } // namespace boost_concepts
308 #endif // BOOST_ITERATOR_CONCEPTS_HPP