forked from boostorg/iterator
Removed workarounds for older compilers from counting_iterator.hpp.
This commit is contained in:
@ -5,209 +5,171 @@
|
||||
#ifndef COUNTING_ITERATOR_DWA200348_HPP
|
||||
#define COUNTING_ITERATOR_DWA200348_HPP
|
||||
|
||||
#include <limits>
|
||||
#include <type_traits>
|
||||
|
||||
#include <boost/config.hpp>
|
||||
#include <boost/detail/workaround.hpp>
|
||||
#ifndef BOOST_NO_LIMITS_COMPILE_TIME_CONSTANTS
|
||||
#include <limits>
|
||||
#endif
|
||||
#include <boost/core/use_default.hpp>
|
||||
#include <boost/detail/numeric_traits.hpp>
|
||||
#include <boost/iterator/iterator_adaptor.hpp>
|
||||
#include <boost/iterator/iterator_categories.hpp>
|
||||
#include <boost/iterator/detail/type_traits/type_identity.hpp>
|
||||
|
||||
namespace boost {
|
||||
namespace iterators {
|
||||
|
||||
template <
|
||||
class Incrementable
|
||||
, class CategoryOrTraversal
|
||||
, class Difference
|
||||
template<
|
||||
class Incrementable,
|
||||
class CategoryOrTraversal,
|
||||
class Difference
|
||||
>
|
||||
class counting_iterator;
|
||||
|
||||
namespace detail
|
||||
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 :
|
||||
public std::integral_constant<bool, std::numeric_limits<T>::is_specialized>
|
||||
{};
|
||||
|
||||
template <>
|
||||
struct is_numeric<long long> :
|
||||
public std::true_type
|
||||
{};
|
||||
|
||||
template <>
|
||||
struct is_numeric<unsigned long long> :
|
||||
public std::true_type
|
||||
{};
|
||||
|
||||
#if defined(BOOST_HAS_INT128)
|
||||
template <>
|
||||
struct is_numeric<boost::int128_type> :
|
||||
public std::true_type
|
||||
{};
|
||||
|
||||
template <>
|
||||
struct is_numeric<boost::uint128_type> :
|
||||
public std::true_type
|
||||
{};
|
||||
#endif
|
||||
|
||||
// Some compilers fail to have a numeric_limits specialization
|
||||
template <>
|
||||
struct is_numeric<wchar_t> :
|
||||
public std::true_type
|
||||
{};
|
||||
|
||||
template <class T>
|
||||
struct numeric_difference
|
||||
{
|
||||
// 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>
|
||||
{};
|
||||
|
||||
template <>
|
||||
struct is_numeric<long long>
|
||||
: std::true_type {};
|
||||
|
||||
template <>
|
||||
struct is_numeric<unsigned long long>
|
||||
: std::true_type {};
|
||||
using type = typename boost::detail::numeric_traits<T>::difference_type;
|
||||
};
|
||||
|
||||
#if defined(BOOST_HAS_INT128)
|
||||
template <>
|
||||
struct is_numeric<boost::int128_type>
|
||||
: std::true_type {};
|
||||
// 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>
|
||||
{
|
||||
using type = boost::int128_type;
|
||||
};
|
||||
|
||||
template <>
|
||||
struct is_numeric<boost::uint128_type>
|
||||
: std::true_type {};
|
||||
template <>
|
||||
struct numeric_difference<boost::uint128_type>
|
||||
{
|
||||
using type = boost::int128_type;
|
||||
};
|
||||
#endif
|
||||
|
||||
// Some compilers fail to have a numeric_limits specialization
|
||||
template <>
|
||||
struct is_numeric<wchar_t>
|
||||
: std::true_type {};
|
||||
template <class Incrementable, class CategoryOrTraversal, class Difference>
|
||||
struct counting_iterator_base
|
||||
{
|
||||
using traversal = typename detail::ia_dflt_help<
|
||||
CategoryOrTraversal,
|
||||
typename std::conditional<
|
||||
is_numeric<Incrementable>::value,
|
||||
iterators::detail::type_identity<random_access_traversal_tag>,
|
||||
iterator_traversal<Incrementable>
|
||||
>::type
|
||||
>::type;
|
||||
|
||||
template <class T>
|
||||
struct numeric_difference
|
||||
{
|
||||
typedef typename boost::detail::numeric_traits<T>::difference_type type;
|
||||
};
|
||||
using difference = typename detail::ia_dflt_help<
|
||||
Difference,
|
||||
typename std::conditional<
|
||||
is_numeric<Incrementable>::value,
|
||||
numeric_difference<Incrementable>,
|
||||
iterator_difference<Incrementable>
|
||||
>::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
|
||||
, iterators::detail::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
|
||||
using type = iterator_adaptor<
|
||||
counting_iterator<Incrementable, CategoryOrTraversal, Difference>, // self
|
||||
Incrementable, // Base
|
||||
#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)
|
||||
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;
|
||||
};
|
||||
Incrementable, // Value
|
||||
traversal,
|
||||
Incrementable const&, // reference
|
||||
difference
|
||||
>;
|
||||
};
|
||||
|
||||
// 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
|
||||
// A distance calculation policy for wrapped iterators
|
||||
template <class Difference, class Incrementable1, class Incrementable2>
|
||||
struct iterator_distance
|
||||
{
|
||||
typedef typename detail::counting_iterator_base<
|
||||
static Difference distance(Incrementable1 x, Incrementable2 y)
|
||||
{
|
||||
return y - x;
|
||||
}
|
||||
};
|
||||
|
||||
// A distance calculation 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);
|
||||
}
|
||||
};
|
||||
|
||||
} // namespace detail
|
||||
|
||||
template<
|
||||
class Incrementable,
|
||||
class CategoryOrTraversal = use_default,
|
||||
class Difference = use_default
|
||||
>
|
||||
class counting_iterator :
|
||||
public detail::counting_iterator_base<Incrementable, CategoryOrTraversal, Difference>::type
|
||||
{
|
||||
using super_t = typename detail::counting_iterator_base<
|
||||
Incrementable, CategoryOrTraversal, Difference
|
||||
>::type super_t;
|
||||
>::type;
|
||||
|
||||
friend class iterator_core_access;
|
||||
|
||||
public:
|
||||
typedef typename super_t::difference_type difference_type;
|
||||
public:
|
||||
using reference = typename super_t::reference;
|
||||
using difference_type = typename super_t::difference_type;
|
||||
|
||||
BOOST_DEFAULTED_FUNCTION(counting_iterator(), {})
|
||||
counting_iterator() = default;
|
||||
|
||||
BOOST_DEFAULTED_FUNCTION(counting_iterator(counting_iterator const& rhs), : super_t(rhs.base()) {})
|
||||
counting_iterator(counting_iterator const&) = default;
|
||||
counting_iterator& operator=(counting_iterator const&) = default;
|
||||
|
||||
counting_iterator(Incrementable x)
|
||||
: super_t(x)
|
||||
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
|
||||
private:
|
||||
reference dereference() const
|
||||
{
|
||||
return this->base_reference();
|
||||
}
|
||||
@ -216,23 +178,21 @@ class counting_iterator
|
||||
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;
|
||||
using distance_traits = typename std::conditional<
|
||||
detail::is_numeric<Incrementable>::value,
|
||||
detail::number_distance<difference_type, Incrementable, OtherIncrementable>,
|
||||
detail::iterator_distance<difference_type, Incrementable, OtherIncrementable>
|
||||
>::type;
|
||||
|
||||
return d::distance(this->base(), y.base());
|
||||
return distance_traits::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)
|
||||
inline counting_iterator<Incrementable> make_counting_iterator(Incrementable x)
|
||||
{
|
||||
typedef counting_iterator<Incrementable> result_t;
|
||||
return result_t(x);
|
||||
return counting_iterator<Incrementable>(x);
|
||||
}
|
||||
|
||||
} // namespace iterators
|
||||
|
Reference in New Issue
Block a user