mirror of
https://github.com/boostorg/iterator.git
synced 2026-01-29 17:42:21 +01:00
С++03 support was deprecated in 1.85 and now can be removed. This PR clears many of workarounds, which are no longer needed now. * Remove unused workaround macros (many of). * Remove BOOST_STATIC_ASSERT usages. * Minimize Boost::type_traits dependency (in favour of STL's type_traits). Closes https://github.com/boostorg/iterator/pull/82. Squashed commit of the following: commit741a627b73Author: Georgy Guminov <gogagum@gmail.com> Date: Sat Jan 25 12:13:05 2025 +0300 Replace testers with standard metafunctions. commitbf4cce6114Author: Georgy Guminov <gogagum@gmail.com> Date: Sat Jan 25 11:51:32 2025 +0300 Refactor is_lvalue_iterator.hpp. commit8d080c6c58Author: Georgy Guminov <gogagum@gmail.com> Date: Sat Jan 25 10:27:32 2025 +0300 Remove more workarounds. commit5a4ba24d36Author: Georgy Guminov <gogagum@gmail.com> Date: Sun Jan 19 16:38:30 2025 +0300 Fixes. commitfdfafce2b9Author: Georgy Guminov <gogagum@gmail.com> Date: Sat Oct 26 15:06:43 2024 +0300 Remove BOOST_NO_STRICT_ITERATOR_INTEROPERABILITY Correct static_assert messages. Fix messages & replace is_standard_layout with is_copy_constructible. commitc69ac1408aAuthor: Georgy Guminov <gogagum@gmail.com> Date: Sat Oct 26 14:48:51 2024 +0300 Correct static_assert messages. commitb5df827151Author: Georqy Guminov <gogagum@gmail.com> Date: Sun Jun 23 16:12:29 2024 +0300 Fixes. Remove some Boost.MPL usages. Remove unused includes. commit01fd35e9f8Author: Georgiy Guminov <gogagum@gmail.com> Date: Wed Jun 12 17:14:21 2024 +0300 abstract conjunction. commitc02def8acfAuthor: Georgiy Guminov <gogagum@gmail.com> Date: Wed Jun 12 16:35:43 2024 +0300 return addressof & conjunction. commit3b3d162575Author: Georgiy Guminov <gogagum@gmail.com> Date: Wed Jun 12 16:30:44 2024 +0300 Make macro more readable. commit4ab19e045fAuthor: Georgiy Guminov <gogagum@gmail.com> Date: Wed Jun 12 15:56:49 2024 +0300 Add static_assert messages. commit82b5c44cd3Author: Georgiy Guminov <gogagum@gmail.com> Date: Wed Jun 12 14:12:10 2024 +0300 Return is iterator CXX17 test. commit2d58d65462Author: Georgiy Guminov <gogagum@gmail.com> Date: Tue Jun 11 14:04:17 2024 +0300 Omitted. commita0d04d9491Author: Georgiy Guminov <gogagum@gmail.com> Date: Tue Jun 11 14:00:35 2024 +0300 Replace move with static_cast commit4a49b8a1a2Author: Georgiy Guminov <gogagum@gmail.com> Date: Mon Jun 10 21:38:53 2024 +0300 Return BOOST_NOEXCEPT commit054c013bbaAuthor: Georgiy Guminov <gogagum@gmail.com> Date: Sun Jun 9 15:20:41 2024 +0300 CXX11
248 lines
6.9 KiB
C++
248 lines
6.9 KiB
C++
// Copyright David Abrahams 2003.
|
|
// Distributed under the Boost Software License, Version 1.0. (See
|
|
// accompanying file LICENSE_1_0.txt or copy at
|
|
// http://www.boost.org/LICENSE_1_0.txt)
|
|
#ifndef COUNTING_ITERATOR_DWA200348_HPP
|
|
# define COUNTING_ITERATOR_DWA200348_HPP
|
|
|
|
# include <type_traits>
|
|
|
|
# include <boost/config.hpp>
|
|
# include <boost/detail/workaround.hpp>
|
|
# ifndef BOOST_NO_LIMITS_COMPILE_TIME_CONSTANTS
|
|
# include <limits>
|
|
# endif
|
|
# include <boost/type_traits/type_identity.hpp>
|
|
# include <boost/detail/numeric_traits.hpp>
|
|
# include <boost/iterator/iterator_adaptor.hpp>
|
|
|
|
namespace boost {
|
|
namespace iterators {
|
|
|
|
template <
|
|
class Incrementable
|
|
, class CategoryOrTraversal
|
|
, class Difference
|
|
>
|
|
class counting_iterator;
|
|
|
|
namespace detail
|
|
{
|
|
// Try to detect numeric types at compile time in ways compatible
|
|
// with the limitations of the compiler and library.
|
|
template <class T>
|
|
struct is_numeric_impl
|
|
{
|
|
|
|
# ifndef BOOST_NO_LIMITS_COMPILE_TIME_CONSTANTS
|
|
|
|
BOOST_STATIC_CONSTANT(bool, value = std::numeric_limits<T>::is_specialized);
|
|
|
|
# else
|
|
|
|
# if !BOOST_WORKAROUND(BOOST_BORLANDC, BOOST_TESTED_AT(0x551))
|
|
BOOST_STATIC_CONSTANT(
|
|
bool, value = (
|
|
std::is_convertible<int,T>::value
|
|
&& std::is_convertible<T,int>::value
|
|
));
|
|
# else
|
|
BOOST_STATIC_CONSTANT(bool, value = std::is_arithmetic<T>::value);
|
|
# endif
|
|
|
|
# endif
|
|
};
|
|
|
|
template <class T>
|
|
struct is_numeric
|
|
: std::integral_constant<bool, ::boost::iterators::detail::is_numeric_impl<T>::value>
|
|
{};
|
|
|
|
# if defined(BOOST_HAS_LONG_LONG)
|
|
template <>
|
|
struct is_numeric<boost::long_long_type>
|
|
: boost::true_type {};
|
|
|
|
template <>
|
|
struct is_numeric<boost::ulong_long_type>
|
|
: boost::true_type {};
|
|
# endif
|
|
|
|
# if defined(BOOST_HAS_INT128)
|
|
template <>
|
|
struct is_numeric<boost::int128_type>
|
|
: boost::true_type {};
|
|
|
|
template <>
|
|
struct is_numeric<boost::uint128_type>
|
|
: boost::true_type {};
|
|
# endif
|
|
|
|
// Some compilers fail to have a numeric_limits specialization
|
|
template <>
|
|
struct is_numeric<wchar_t>
|
|
: true_type {};
|
|
|
|
template <class T>
|
|
struct numeric_difference
|
|
{
|
|
typedef typename boost::detail::numeric_traits<T>::difference_type type;
|
|
};
|
|
|
|
# if defined(BOOST_HAS_INT128)
|
|
// std::numeric_limits, which is used by numeric_traits, is not specialized for __int128 in some standard libraries
|
|
template <>
|
|
struct numeric_difference<boost::int128_type>
|
|
{
|
|
typedef boost::int128_type type;
|
|
};
|
|
|
|
template <>
|
|
struct numeric_difference<boost::uint128_type>
|
|
{
|
|
typedef boost::int128_type type;
|
|
};
|
|
# endif
|
|
|
|
template <class Incrementable, class CategoryOrTraversal, class Difference>
|
|
struct counting_iterator_base
|
|
{
|
|
typedef typename detail::ia_dflt_help<
|
|
CategoryOrTraversal
|
|
, typename std::conditional<
|
|
is_numeric<Incrementable>::value
|
|
, boost::type_identity<random_access_traversal_tag>
|
|
, iterator_traversal<Incrementable>
|
|
>::type
|
|
>::type traversal;
|
|
|
|
typedef typename detail::ia_dflt_help<
|
|
Difference
|
|
, typename std::conditional<
|
|
is_numeric<Incrementable>::value
|
|
, numeric_difference<Incrementable>
|
|
, iterator_difference<Incrementable>
|
|
>::type
|
|
>::type difference;
|
|
|
|
typedef iterator_adaptor<
|
|
counting_iterator<Incrementable, CategoryOrTraversal, Difference> // self
|
|
, Incrementable // Base
|
|
, Incrementable // Value
|
|
# ifndef BOOST_ITERATOR_REF_CONSTNESS_KILLS_WRITABILITY
|
|
const // MSVC won't strip this. Instead we enable Thomas'
|
|
// criterion (see boost/iterator/detail/facade_iterator_category.hpp)
|
|
# endif
|
|
, traversal
|
|
, Incrementable const& // reference
|
|
, difference
|
|
> type;
|
|
};
|
|
|
|
// Template class distance_policy_select -- choose a policy for computing the
|
|
// distance between counting_iterators at compile-time based on whether or not
|
|
// the iterator wraps an integer or an iterator, using "poor man's partial
|
|
// specialization".
|
|
|
|
template <bool is_integer> struct distance_policy_select;
|
|
|
|
// A policy for wrapped iterators
|
|
template <class Difference, class Incrementable1, class Incrementable2>
|
|
struct iterator_distance
|
|
{
|
|
static Difference distance(Incrementable1 x, Incrementable2 y)
|
|
{
|
|
return y - x;
|
|
}
|
|
};
|
|
|
|
// A policy for wrapped numbers
|
|
template <class Difference, class Incrementable1, class Incrementable2>
|
|
struct number_distance
|
|
{
|
|
static Difference distance(Incrementable1 x, Incrementable2 y)
|
|
{
|
|
return boost::detail::numeric_distance(x, y);
|
|
}
|
|
};
|
|
}
|
|
|
|
template <
|
|
class Incrementable
|
|
, class CategoryOrTraversal = use_default
|
|
, class Difference = use_default
|
|
>
|
|
class counting_iterator
|
|
: public detail::counting_iterator_base<
|
|
Incrementable, CategoryOrTraversal, Difference
|
|
>::type
|
|
{
|
|
typedef typename detail::counting_iterator_base<
|
|
Incrementable, CategoryOrTraversal, Difference
|
|
>::type super_t;
|
|
|
|
friend class iterator_core_access;
|
|
|
|
public:
|
|
typedef typename super_t::difference_type difference_type;
|
|
|
|
BOOST_DEFAULTED_FUNCTION(counting_iterator(), {})
|
|
|
|
BOOST_DEFAULTED_FUNCTION(counting_iterator(counting_iterator const& rhs), : super_t(rhs.base()) {})
|
|
|
|
counting_iterator(Incrementable x)
|
|
: super_t(x)
|
|
{
|
|
}
|
|
|
|
# if 0
|
|
template<class OtherIncrementable>
|
|
counting_iterator(
|
|
counting_iterator<OtherIncrementable, CategoryOrTraversal, Difference> const& t
|
|
, typename enable_if_convertible<OtherIncrementable, Incrementable>::type* = 0
|
|
)
|
|
: super_t(t.base())
|
|
{}
|
|
# endif
|
|
|
|
BOOST_DEFAULTED_FUNCTION(counting_iterator& operator=(counting_iterator const& rhs), { *static_cast< super_t* >(this) = static_cast< super_t const& >(rhs); return *this; })
|
|
|
|
private:
|
|
|
|
typename super_t::reference dereference() const
|
|
{
|
|
return this->base_reference();
|
|
}
|
|
|
|
template <class OtherIncrementable>
|
|
difference_type
|
|
distance_to(counting_iterator<OtherIncrementable, CategoryOrTraversal, Difference> const& y) const
|
|
{
|
|
typedef typename std::conditional<
|
|
detail::is_numeric<Incrementable>::value
|
|
, detail::number_distance<difference_type, Incrementable, OtherIncrementable>
|
|
, detail::iterator_distance<difference_type, Incrementable, OtherIncrementable>
|
|
>::type d;
|
|
|
|
return d::distance(this->base(), y.base());
|
|
}
|
|
};
|
|
|
|
// Manufacture a counting iterator for an arbitrary incrementable type
|
|
template <class Incrementable>
|
|
inline counting_iterator<Incrementable>
|
|
make_counting_iterator(Incrementable x)
|
|
{
|
|
typedef counting_iterator<Incrementable> result_t;
|
|
return result_t(x);
|
|
}
|
|
|
|
} // namespace iterators
|
|
|
|
using iterators::counting_iterator;
|
|
using iterators::make_counting_iterator;
|
|
|
|
} // namespace boost
|
|
|
|
#endif // COUNTING_ITERATOR_DWA200348_HPP
|