191 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			191 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			C++
		
	
	
	
| // -*- C++ -*-
 | |
| //===----------------------------------------------------------------------===//
 | |
| //
 | |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
 | |
| // See https://llvm.org/LICENSE.txt for license information.
 | |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #ifndef _LIBCPP___ITERATOR_ADVANCE_H
 | |
| #define _LIBCPP___ITERATOR_ADVANCE_H
 | |
| 
 | |
| #include <__config>
 | |
| #include <__debug>
 | |
| #include <__function_like.h>
 | |
| #include <__iterator/concepts.h>
 | |
| #include <__iterator/incrementable_traits.h>
 | |
| #include <concepts>
 | |
| #include <limits>
 | |
| #include <type_traits>
 | |
| #include <utility>
 | |
| 
 | |
| #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
 | |
| #pragma GCC system_header
 | |
| #endif
 | |
| 
 | |
| _LIBCPP_PUSH_MACROS
 | |
| #include <__undef_macros>
 | |
| 
 | |
| _LIBCPP_BEGIN_NAMESPACE_STD
 | |
| 
 | |
| template <class _InputIter>
 | |
| inline _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR_AFTER_CXX14 void
 | |
| __advance(_InputIter& __i, typename iterator_traits<_InputIter>::difference_type __n, input_iterator_tag) {
 | |
|   for (; __n > 0; --__n)
 | |
|     ++__i;
 | |
| }
 | |
| 
 | |
| template <class _BiDirIter>
 | |
| inline _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR_AFTER_CXX14 void
 | |
| __advance(_BiDirIter& __i, typename iterator_traits<_BiDirIter>::difference_type __n, bidirectional_iterator_tag) {
 | |
|   if (__n >= 0)
 | |
|     for (; __n > 0; --__n)
 | |
|       ++__i;
 | |
|   else
 | |
|     for (; __n < 0; ++__n)
 | |
|       --__i;
 | |
| }
 | |
| 
 | |
| template <class _RandIter>
 | |
| inline _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR_AFTER_CXX14 void
 | |
| __advance(_RandIter& __i, typename iterator_traits<_RandIter>::difference_type __n, random_access_iterator_tag) {
 | |
|   __i += __n;
 | |
| }
 | |
| 
 | |
| template <
 | |
|     class _InputIter, class _Distance,
 | |
|     class = typename enable_if<is_integral<decltype(_VSTD::__convert_to_integral(declval<_Distance>()))>::value>::type>
 | |
| inline _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR_AFTER_CXX14 void advance(_InputIter& __i, _Distance __orig_n) {
 | |
|   typedef decltype(_VSTD::__convert_to_integral(__orig_n)) _IntegralSize;
 | |
|   _IntegralSize __n = __orig_n;
 | |
|   _LIBCPP_ASSERT(__n >= 0 || __is_cpp17_bidirectional_iterator<_InputIter>::value,
 | |
|                  "Attempt to advance(it, n) with negative n on a non-bidirectional iterator");
 | |
|   _VSTD::__advance(__i, __n, typename iterator_traits<_InputIter>::iterator_category());
 | |
| }
 | |
| 
 | |
| #if !defined(_LIBCPP_HAS_NO_RANGES)
 | |
| 
 | |
| namespace ranges {
 | |
| // [range.iter.op.advance]
 | |
| struct __advance_fn final : __function_like {
 | |
| private:
 | |
|   template <class _Tp>
 | |
|   static constexpr _Tp __abs(_Tp __n) noexcept {
 | |
|     return __n < 0 ? -__n : __n;
 | |
|   }
 | |
| 
 | |
|   template <class _Ip>
 | |
|   static constexpr void __advance_forward(_Ip& __i, iter_difference_t<_Ip> __n) {
 | |
|     while (__n > 0) {
 | |
|       --__n;
 | |
|       ++__i;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   template <class _Ip>
 | |
|   static constexpr void __advance_backward(_Ip& __i, iter_difference_t<_Ip> __n) {
 | |
|     while (__n < 0) {
 | |
|       ++__n;
 | |
|       --__i;
 | |
|     }
 | |
|   }
 | |
| 
 | |
| public:
 | |
|   constexpr explicit __advance_fn(__tag __x) noexcept : __function_like(__x) {}
 | |
| 
 | |
|   // Preconditions: If `I` does not model `bidirectional_iterator`, `n` is not negative.
 | |
|   template <input_or_output_iterator _Ip>
 | |
|   constexpr void operator()(_Ip& __i, iter_difference_t<_Ip> __n) const {
 | |
|     _LIBCPP_ASSERT(__n >= 0 || bidirectional_iterator<_Ip>,
 | |
|                    "If `n < 0`, then `bidirectional_iterator<I>` must be true.");
 | |
| 
 | |
|     // If `I` models `random_access_iterator`, equivalent to `i += n`.
 | |
|     if constexpr (random_access_iterator<_Ip>) {
 | |
|       __i += __n;
 | |
|       return;
 | |
|     } else if constexpr (bidirectional_iterator<_Ip>) {
 | |
|       // Otherwise, if `n` is non-negative, increments `i` by `n`.
 | |
|       __advance_forward(__i, __n);
 | |
|       // Otherwise, decrements `i` by `-n`.
 | |
|       __advance_backward(__i, __n);
 | |
|       return;
 | |
|     } else {
 | |
|       // Otherwise, if `n` is non-negative, increments `i` by `n`.
 | |
|       __advance_forward(__i, __n);
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // Preconditions: Either `assignable_from<I&, S> || sized_sentinel_for<S, I>` is modeled, or [i, bound) denotes a range.
 | |
|   template <input_or_output_iterator _Ip, sentinel_for<_Ip> _Sp>
 | |
|   constexpr void operator()(_Ip& __i, _Sp __bound) const {
 | |
|     // If `I` and `S` model `assignable_from<I&, S>`, equivalent to `i = std::move(bound)`.
 | |
|     if constexpr (assignable_from<_Ip&, _Sp>) {
 | |
|       __i = std::move(__bound);
 | |
|     }
 | |
|     // Otherwise, if `S` and `I` model `sized_sentinel_for<S, I>`, equivalent to `ranges::advance(i, bound - i)`.
 | |
|     else if constexpr (sized_sentinel_for<_Sp, _Ip>) {
 | |
|       (*this)(__i, __bound - __i);
 | |
|     }
 | |
|     // Otherwise, while `bool(i != bound)` is true, increments `i`.
 | |
|     else {
 | |
|       while (__i != __bound) {
 | |
|         ++__i;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // Preconditions:
 | |
|   //   * If `n > 0`, [i, bound) denotes a range.
 | |
|   //   * If `n == 0`, [i, bound) or [bound, i) denotes a range.
 | |
|   //   * If `n < 0`, [bound, i) denotes a range, `I` models `bidirectional_iterator`, and `I` and `S` model `same_as<I, S>`.
 | |
|   // Returns: `n - M`, where `M` is the difference between the the ending and starting position.
 | |
|   template <input_or_output_iterator _Ip, sentinel_for<_Ip> _Sp>
 | |
|   constexpr iter_difference_t<_Ip> operator()(_Ip& __i, iter_difference_t<_Ip> __n, _Sp __bound) const {
 | |
|     _LIBCPP_ASSERT(__n >= 0 || (bidirectional_iterator<_Ip> && same_as<_Ip, _Sp>),
 | |
|                    "If `n < 0`, then `bidirectional_iterator<I> && same_as<I, S>` must be true.");
 | |
|     // If `S` and `I` model `sized_sentinel_for<S, I>`:
 | |
|     if constexpr (sized_sentinel_for<_Sp, _Ip>) {
 | |
|       // If |n| >= |bound - i|, equivalent to `ranges::advance(i, bound)`.
 | |
|       if (const auto __M = __bound - __i; __abs(__n) >= __abs(__M)) {
 | |
|         (*this)(__i, __bound);
 | |
|         return __n - __M;
 | |
|       }
 | |
| 
 | |
|       // Otherwise, equivalent to `ranges::advance(i, n)`.
 | |
|       (*this)(__i, __n);
 | |
|       return 0;
 | |
|     } else {
 | |
|       // Otherwise, if `n` is non-negative, while `bool(i != bound)` is true, increments `i` but at
 | |
|       // most `n` times.
 | |
|       while (__i != __bound && __n > 0) {
 | |
|         ++__i;
 | |
|         --__n;
 | |
|       }
 | |
| 
 | |
|       // Otherwise, while `bool(i != bound)` is true, decrements `i` but at most `-n` times.
 | |
|       if constexpr (bidirectional_iterator<_Ip> && same_as<_Ip, _Sp>) {
 | |
|         while (__i != __bound && __n < 0) {
 | |
|           --__i;
 | |
|           ++__n;
 | |
|         }
 | |
|       }
 | |
|       return __n;
 | |
|     }
 | |
| 
 | |
|     _LIBCPP_UNREACHABLE();
 | |
|   }
 | |
| };
 | |
| 
 | |
| inline constexpr auto advance = __advance_fn(__function_like::__tag());
 | |
| } // namespace ranges
 | |
| 
 | |
| #endif // !defined(_LIBCPP_HAS_NO_RANGES)
 | |
| 
 | |
| _LIBCPP_END_NAMESPACE_STD
 | |
| 
 | |
| _LIBCPP_POP_MACROS
 | |
| 
 | |
| #endif // _LIBCPP___ITERATOR_ADVANCE_H
 |