xref: /llvm-project/libcxx/include/__iterator/common_iterator.h (revision 09e3a360581dc36d0820d3fb6da9bd7cfed87b5d)
11a29403dSzoecarver // -*- C++ -*-
21a29403dSzoecarver //===----------------------------------------------------------------------===//
31a29403dSzoecarver //
41a29403dSzoecarver // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
51a29403dSzoecarver // See https://llvm.org/LICENSE.txt for license information.
61a29403dSzoecarver // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
71a29403dSzoecarver //
81a29403dSzoecarver //===----------------------------------------------------------------------===//
91a29403dSzoecarver 
101a29403dSzoecarver #ifndef _LIBCPP___ITERATOR_COMMON_ITERATOR_H
111a29403dSzoecarver #define _LIBCPP___ITERATOR_COMMON_ITERATOR_H
121a29403dSzoecarver 
13f87aa19bSLouis Dionne #include <__assert>
1489b356f0SNikolas Klauser #include <__concepts/assignable.h>
1589b356f0SNikolas Klauser #include <__concepts/constructible.h>
1689b356f0SNikolas Klauser #include <__concepts/convertible_to.h>
1789b356f0SNikolas Klauser #include <__concepts/copyable.h>
1889b356f0SNikolas Klauser #include <__concepts/derived_from.h>
1989b356f0SNikolas Klauser #include <__concepts/equality_comparable.h>
2089b356f0SNikolas Klauser #include <__concepts/same_as.h>
211a29403dSzoecarver #include <__config>
221a29403dSzoecarver #include <__iterator/concepts.h>
231a29403dSzoecarver #include <__iterator/incrementable_traits.h>
241a29403dSzoecarver #include <__iterator/iter_move.h>
251a29403dSzoecarver #include <__iterator/iter_swap.h>
261a29403dSzoecarver #include <__iterator/iterator_traits.h>
271a29403dSzoecarver #include <__iterator/readable_traits.h>
2843562287SNikolas Klauser #include <__memory/addressof.h>
29*09e3a360SLouis Dionne #include <__type_traits/conditional.h>
30430b397fSNikolas Klauser #include <__type_traits/is_pointer.h>
31430b397fSNikolas Klauser #include <__utility/declval.h>
321a29403dSzoecarver #include <variant>
331a29403dSzoecarver 
341a29403dSzoecarver #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
351a29403dSzoecarver #  pragma GCC system_header
361a29403dSzoecarver #endif
371a29403dSzoecarver 
3892e4d679SNicole Rabjohn _LIBCPP_PUSH_MACROS
3992e4d679SNicole Rabjohn #include <__undef_macros>
4092e4d679SNicole Rabjohn 
411a29403dSzoecarver _LIBCPP_BEGIN_NAMESPACE_STD
421a29403dSzoecarver 
434f15267dSNikolas Klauser #if _LIBCPP_STD_VER >= 20
441a29403dSzoecarver 
450902eb30SLouis Dionne template <class _Iter>
460902eb30SLouis Dionne concept __can_use_postfix_proxy =
479783f28cSLouis Dionne     constructible_from<iter_value_t<_Iter>, iter_reference_t<_Iter>> && move_constructible<iter_value_t<_Iter>>;
480902eb30SLouis Dionne 
491a29403dSzoecarver template <input_or_output_iterator _Iter, sentinel_for<_Iter> _Sent>
501a29403dSzoecarver   requires(!same_as<_Iter, _Sent> && copyable<_Iter>)
511a29403dSzoecarver class common_iterator {
52844a9c0eSArthur O'Dwyer   struct __proxy {
5383ce1397SNikolas Klauser     _LIBCPP_HIDE_FROM_ABI constexpr const iter_value_t<_Iter>* operator->() const noexcept {
5477a00c0dSLouis Dionne       return std::addressof(__value_);
551a29403dSzoecarver     }
56844a9c0eSArthur O'Dwyer     iter_value_t<_Iter> __value_;
571a29403dSzoecarver   };
581a29403dSzoecarver 
59844a9c0eSArthur O'Dwyer   struct __postfix_proxy {
609783f28cSLouis Dionne     _LIBCPP_HIDE_FROM_ABI constexpr const iter_value_t<_Iter>& operator*() const noexcept { return __value_; }
61844a9c0eSArthur O'Dwyer     iter_value_t<_Iter> __value_;
621a29403dSzoecarver   };
631a29403dSzoecarver 
641a29403dSzoecarver   variant<_Iter, _Sent> __hold_;
65ca9b1d1aSLouis Dionne   template <input_or_output_iterator _OtherIter, sentinel_for<_OtherIter> _OtherSent>
66ca9b1d1aSLouis Dionne     requires(!same_as<_OtherIter, _OtherSent> && copyable<_OtherIter>)
67ca9b1d1aSLouis Dionne   friend class common_iterator;
681a29403dSzoecarver 
69ca9b1d1aSLouis Dionne public:
709783f28cSLouis Dionne   _LIBCPP_HIDE_FROM_ABI common_iterator()
719783f28cSLouis Dionne     requires default_initializable<_Iter>
729783f28cSLouis Dionne   = default;
731a29403dSzoecarver 
7477a00c0dSLouis Dionne   _LIBCPP_HIDE_FROM_ABI constexpr common_iterator(_Iter __i) : __hold_(in_place_type<_Iter>, std::move(__i)) {}
7577a00c0dSLouis Dionne   _LIBCPP_HIDE_FROM_ABI constexpr common_iterator(_Sent __s) : __hold_(in_place_type<_Sent>, std::move(__s)) {}
761a29403dSzoecarver 
771a29403dSzoecarver   template <class _I2, class _S2>
781a29403dSzoecarver     requires convertible_to<const _I2&, _Iter> && convertible_to<const _S2&, _Sent>
7983ce1397SNikolas Klauser   _LIBCPP_HIDE_FROM_ABI constexpr common_iterator(const common_iterator<_I2, _S2>& __other)
801a29403dSzoecarver       : __hold_([&]() -> variant<_Iter, _Sent> {
811638657dSKonstantin Varlamov           _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
829783f28cSLouis Dionne               !__other.__hold_.valueless_by_exception(), "Attempted to construct from a valueless common_iterator");
831a29403dSzoecarver           if (__other.__hold_.index() == 0)
8477a00c0dSLouis Dionne             return variant<_Iter, _Sent>{in_place_index<0>, std::__unchecked_get<0>(__other.__hold_)};
8577a00c0dSLouis Dionne           return variant<_Iter, _Sent>{in_place_index<1>, std::__unchecked_get<1>(__other.__hold_)};
861a29403dSzoecarver         }()) {}
871a29403dSzoecarver 
881a29403dSzoecarver   template <class _I2, class _S2>
891a29403dSzoecarver     requires convertible_to<const _I2&, _Iter> && convertible_to<const _S2&, _Sent> &&
901a29403dSzoecarver              assignable_from<_Iter&, const _I2&> && assignable_from<_Sent&, const _S2&>
9183ce1397SNikolas Klauser   _LIBCPP_HIDE_FROM_ABI common_iterator& operator=(const common_iterator<_I2, _S2>& __other) {
921638657dSKonstantin Varlamov     _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
939783f28cSLouis Dionne         !__other.__hold_.valueless_by_exception(), "Attempted to assign from a valueless common_iterator");
941a29403dSzoecarver 
951a29403dSzoecarver     auto __idx       = __hold_.index();
961a29403dSzoecarver     auto __other_idx = __other.__hold_.index();
971a29403dSzoecarver 
981a29403dSzoecarver     // If they're the same index, just assign.
991a29403dSzoecarver     if (__idx == 0 && __other_idx == 0)
10077a00c0dSLouis Dionne       std::__unchecked_get<0>(__hold_) = std::__unchecked_get<0>(__other.__hold_);
1011a29403dSzoecarver     else if (__idx == 1 && __other_idx == 1)
10277a00c0dSLouis Dionne       std::__unchecked_get<1>(__hold_) = std::__unchecked_get<1>(__other.__hold_);
1031a29403dSzoecarver 
1041a29403dSzoecarver     // Otherwise replace with the oposite element.
1051a29403dSzoecarver     else if (__other_idx == 1)
10677a00c0dSLouis Dionne       __hold_.template emplace<1>(std::__unchecked_get<1>(__other.__hold_));
1071a29403dSzoecarver     else if (__other_idx == 0)
10877a00c0dSLouis Dionne       __hold_.template emplace<0>(std::__unchecked_get<0>(__other.__hold_));
1091a29403dSzoecarver 
1101a29403dSzoecarver     return *this;
1111a29403dSzoecarver   }
1121a29403dSzoecarver 
1139783f28cSLouis Dionne   _LIBCPP_HIDE_FROM_ABI constexpr decltype(auto) operator*() {
1141638657dSKonstantin Varlamov     _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
1159783f28cSLouis Dionne         std::holds_alternative<_Iter>(__hold_), "Attempted to dereference a non-dereferenceable common_iterator");
11677a00c0dSLouis Dionne     return *std::__unchecked_get<_Iter>(__hold_);
1171a29403dSzoecarver   }
1181a29403dSzoecarver 
11983ce1397SNikolas Klauser   _LIBCPP_HIDE_FROM_ABI constexpr decltype(auto) operator*() const
1201a29403dSzoecarver     requires __dereferenceable<const _Iter>
1211a29403dSzoecarver   {
1221638657dSKonstantin Varlamov     _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
1239783f28cSLouis Dionne         std::holds_alternative<_Iter>(__hold_), "Attempted to dereference a non-dereferenceable common_iterator");
12477a00c0dSLouis Dionne     return *std::__unchecked_get<_Iter>(__hold_);
1251a29403dSzoecarver   }
1261a29403dSzoecarver 
1271a29403dSzoecarver   template <class _I2 = _Iter>
128f03430f5SXiaoyang Liu   _LIBCPP_HIDE_FROM_ABI auto operator->() const
1299783f28cSLouis Dionne     requires indirectly_readable<const _I2> && (requires(const _I2& __i) {
1309783f28cSLouis Dionne                __i.operator->();
1319783f28cSLouis Dionne              } || is_reference_v<iter_reference_t<_I2>> || constructible_from<iter_value_t<_I2>, iter_reference_t<_I2>>)
1321a29403dSzoecarver   {
1331638657dSKonstantin Varlamov     _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
1349783f28cSLouis Dionne         std::holds_alternative<_Iter>(__hold_), "Attempted to dereference a non-dereferenceable common_iterator");
1351a29403dSzoecarver     if constexpr (is_pointer_v<_Iter> || requires(const _Iter& __i) { __i.operator->(); }) {
13677a00c0dSLouis Dionne       return std::__unchecked_get<_Iter>(__hold_);
1371a29403dSzoecarver     } else if constexpr (is_reference_v<iter_reference_t<_Iter>>) {
13877a00c0dSLouis Dionne       auto&& __tmp = *std::__unchecked_get<_Iter>(__hold_);
13977a00c0dSLouis Dionne       return std::addressof(__tmp);
1401a29403dSzoecarver     } else {
14177a00c0dSLouis Dionne       return __proxy{*std::__unchecked_get<_Iter>(__hold_)};
1421a29403dSzoecarver     }
1431a29403dSzoecarver   }
1441a29403dSzoecarver 
14583ce1397SNikolas Klauser   _LIBCPP_HIDE_FROM_ABI common_iterator& operator++() {
1461638657dSKonstantin Varlamov     _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
1479783f28cSLouis Dionne         std::holds_alternative<_Iter>(__hold_), "Attempted to increment a non-dereferenceable common_iterator");
1489783f28cSLouis Dionne     ++std::__unchecked_get<_Iter>(__hold_);
1499783f28cSLouis Dionne     return *this;
1501a29403dSzoecarver   }
1511a29403dSzoecarver 
15283ce1397SNikolas Klauser   _LIBCPP_HIDE_FROM_ABI decltype(auto) operator++(int) {
1531638657dSKonstantin Varlamov     _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
1549783f28cSLouis Dionne         std::holds_alternative<_Iter>(__hold_), "Attempted to increment a non-dereferenceable common_iterator");
1551a29403dSzoecarver     if constexpr (forward_iterator<_Iter>) {
1561a29403dSzoecarver       auto __tmp = *this;
1571a29403dSzoecarver       ++*this;
1581a29403dSzoecarver       return __tmp;
1599783f28cSLouis Dionne     } else if constexpr (requires(_Iter& __i) {
1609783f28cSLouis Dionne                            { *__i++ } -> __can_reference;
1619783f28cSLouis Dionne                          } || !__can_use_postfix_proxy<_Iter>) {
16277a00c0dSLouis Dionne       return std::__unchecked_get<_Iter>(__hold_)++;
1631a29403dSzoecarver     } else {
164844a9c0eSArthur O'Dwyer       auto __p = __postfix_proxy{**this};
1651a29403dSzoecarver       ++*this;
1661a29403dSzoecarver       return __p;
1671a29403dSzoecarver     }
1681a29403dSzoecarver   }
1691a29403dSzoecarver 
1701a29403dSzoecarver   template <class _I2, sentinel_for<_Iter> _S2>
1711a29403dSzoecarver     requires sentinel_for<_Sent, _I2>
1729783f28cSLouis Dionne   _LIBCPP_HIDE_FROM_ABI friend constexpr bool
1739783f28cSLouis Dionne   operator==(const common_iterator& __x, const common_iterator<_I2, _S2>& __y) {
1741638657dSKonstantin Varlamov     _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
1759783f28cSLouis Dionne         !__x.__hold_.valueless_by_exception(), "Attempted to compare a valueless common_iterator");
1761638657dSKonstantin Varlamov     _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
1779783f28cSLouis Dionne         !__y.__hold_.valueless_by_exception(), "Attempted to compare a valueless common_iterator");
1781a29403dSzoecarver 
1791a29403dSzoecarver     auto __x_index = __x.__hold_.index();
1801a29403dSzoecarver     auto __y_index = __y.__hold_.index();
1811a29403dSzoecarver 
1821a29403dSzoecarver     if (__x_index == __y_index)
1831a29403dSzoecarver       return true;
1841a29403dSzoecarver 
1851a29403dSzoecarver     if (__x_index == 0)
18677a00c0dSLouis Dionne       return std::__unchecked_get<_Iter>(__x.__hold_) == std::__unchecked_get<_S2>(__y.__hold_);
1871a29403dSzoecarver 
18877a00c0dSLouis Dionne     return std::__unchecked_get<_Sent>(__x.__hold_) == std::__unchecked_get<_I2>(__y.__hold_);
1891a29403dSzoecarver   }
1901a29403dSzoecarver 
1911a29403dSzoecarver   template <class _I2, sentinel_for<_Iter> _S2>
1921a29403dSzoecarver     requires sentinel_for<_Sent, _I2> && equality_comparable_with<_Iter, _I2>
1939783f28cSLouis Dionne   _LIBCPP_HIDE_FROM_ABI friend constexpr bool
1949783f28cSLouis Dionne   operator==(const common_iterator& __x, const common_iterator<_I2, _S2>& __y) {
1951638657dSKonstantin Varlamov     _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
1969783f28cSLouis Dionne         !__x.__hold_.valueless_by_exception(), "Attempted to compare a valueless common_iterator");
1971638657dSKonstantin Varlamov     _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
1989783f28cSLouis Dionne         !__y.__hold_.valueless_by_exception(), "Attempted to compare a valueless common_iterator");
1991a29403dSzoecarver 
2001a29403dSzoecarver     auto __x_index = __x.__hold_.index();
2011a29403dSzoecarver     auto __y_index = __y.__hold_.index();
2021a29403dSzoecarver 
2031a29403dSzoecarver     if (__x_index == 1 && __y_index == 1)
2041a29403dSzoecarver       return true;
2051a29403dSzoecarver 
2061a29403dSzoecarver     if (__x_index == 0 && __y_index == 0)
20777a00c0dSLouis Dionne       return std::__unchecked_get<_Iter>(__x.__hold_) == std::__unchecked_get<_I2>(__y.__hold_);
2081a29403dSzoecarver 
2091a29403dSzoecarver     if (__x_index == 0)
21077a00c0dSLouis Dionne       return std::__unchecked_get<_Iter>(__x.__hold_) == std::__unchecked_get<_S2>(__y.__hold_);
2111a29403dSzoecarver 
21277a00c0dSLouis Dionne     return std::__unchecked_get<_Sent>(__x.__hold_) == std::__unchecked_get<_I2>(__y.__hold_);
2131a29403dSzoecarver   }
2141a29403dSzoecarver 
2151a29403dSzoecarver   template <sized_sentinel_for<_Iter> _I2, sized_sentinel_for<_Iter> _S2>
2161a29403dSzoecarver     requires sized_sentinel_for<_Sent, _I2>
2179783f28cSLouis Dionne   _LIBCPP_HIDE_FROM_ABI friend constexpr iter_difference_t<_I2>
2189783f28cSLouis Dionne   operator-(const common_iterator& __x, const common_iterator<_I2, _S2>& __y) {
2191638657dSKonstantin Varlamov     _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
2209783f28cSLouis Dionne         !__x.__hold_.valueless_by_exception(), "Attempted to subtract from a valueless common_iterator");
2211638657dSKonstantin Varlamov     _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
2229783f28cSLouis Dionne         !__y.__hold_.valueless_by_exception(), "Attempted to subtract a valueless common_iterator");
2231a29403dSzoecarver 
2241a29403dSzoecarver     auto __x_index = __x.__hold_.index();
2251a29403dSzoecarver     auto __y_index = __y.__hold_.index();
2261a29403dSzoecarver 
2271a29403dSzoecarver     if (__x_index == 1 && __y_index == 1)
2281a29403dSzoecarver       return 0;
2291a29403dSzoecarver 
2301a29403dSzoecarver     if (__x_index == 0 && __y_index == 0)
23177a00c0dSLouis Dionne       return std::__unchecked_get<_Iter>(__x.__hold_) - std::__unchecked_get<_I2>(__y.__hold_);
2321a29403dSzoecarver 
2331a29403dSzoecarver     if (__x_index == 0)
23477a00c0dSLouis Dionne       return std::__unchecked_get<_Iter>(__x.__hold_) - std::__unchecked_get<_S2>(__y.__hold_);
2351a29403dSzoecarver 
23677a00c0dSLouis Dionne     return std::__unchecked_get<_Sent>(__x.__hold_) - std::__unchecked_get<_I2>(__y.__hold_);
2371a29403dSzoecarver   }
2381a29403dSzoecarver 
2396776d65cSNoumanAmir-10xe   _LIBCPP_HIDE_FROM_ABI friend constexpr decltype(auto)
2409783f28cSLouis Dionne   iter_move(const common_iterator& __i) noexcept(noexcept(ranges::iter_move(std::declval<const _Iter&>())))
2411a29403dSzoecarver     requires input_iterator<_Iter>
2421a29403dSzoecarver   {
2431638657dSKonstantin Varlamov     _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
2449783f28cSLouis Dionne         std::holds_alternative<_Iter>(__i.__hold_), "Attempted to iter_move a non-dereferenceable common_iterator");
24577a00c0dSLouis Dionne     return ranges::iter_move(std::__unchecked_get<_Iter>(__i.__hold_));
2461a29403dSzoecarver   }
2471a29403dSzoecarver 
2481a29403dSzoecarver   template <indirectly_swappable<_Iter> _I2, class _S2>
2499783f28cSLouis Dionne   _LIBCPP_HIDE_FROM_ABI friend constexpr void
2509783f28cSLouis Dionne   iter_swap(const common_iterator& __x, const common_iterator<_I2, _S2>& __y) noexcept(
2519783f28cSLouis Dionne       noexcept(ranges::iter_swap(std::declval<const _Iter&>(), std::declval<const _I2&>()))) {
2521638657dSKonstantin Varlamov     _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
2539783f28cSLouis Dionne         std::holds_alternative<_Iter>(__x.__hold_), "Attempted to iter_swap a non-dereferenceable common_iterator");
2541638657dSKonstantin Varlamov     _LIBCPP_ASSERT_VALID_ELEMENT_ACCESS(
2559783f28cSLouis Dionne         std::holds_alternative<_I2>(__y.__hold_), "Attempted to iter_swap a non-dereferenceable common_iterator");
25677a00c0dSLouis Dionne     return ranges::iter_swap(std::__unchecked_get<_Iter>(__x.__hold_), std::__unchecked_get<_I2>(__y.__hold_));
2571a29403dSzoecarver   }
2581a29403dSzoecarver };
2591a29403dSzoecarver 
2601a29403dSzoecarver template <class _Iter, class _Sent>
2611a29403dSzoecarver struct incrementable_traits<common_iterator<_Iter, _Sent>> {
2621a29403dSzoecarver   using difference_type = iter_difference_t<_Iter>;
2631a29403dSzoecarver };
2641a29403dSzoecarver 
2651a29403dSzoecarver template <class _Iter>
2669783f28cSLouis Dionne concept __denotes_forward_iter = requires {
2679783f28cSLouis Dionne   typename iterator_traits<_Iter>::iterator_category;
2689783f28cSLouis Dionne } && derived_from<typename iterator_traits<_Iter>::iterator_category, forward_iterator_tag>;
2691a29403dSzoecarver 
2701a29403dSzoecarver template <class _Iter, class _Sent>
2719783f28cSLouis Dionne concept __common_iter_has_ptr_op = requires(const common_iterator<_Iter, _Sent>& __a) { __a.operator->(); };
2721a29403dSzoecarver 
2731a29403dSzoecarver template <class, class>
2741a29403dSzoecarver struct __arrow_type_or_void {
2751a29403dSzoecarver   using type = void;
2761a29403dSzoecarver };
2771a29403dSzoecarver 
2781a29403dSzoecarver template <class _Iter, class _Sent>
2791a29403dSzoecarver   requires __common_iter_has_ptr_op<_Iter, _Sent>
2801a29403dSzoecarver struct __arrow_type_or_void<_Iter, _Sent> {
28173e8e1baSNikolas Klauser   using type = decltype(std::declval<const common_iterator<_Iter, _Sent>&>().operator->());
2821a29403dSzoecarver };
2831a29403dSzoecarver 
2840902eb30SLouis Dionne template <input_iterator _Iter, class _Sent>
2851a29403dSzoecarver struct iterator_traits<common_iterator<_Iter, _Sent>> {
2869783f28cSLouis Dionne   using iterator_concept  = _If<forward_iterator<_Iter>, forward_iterator_tag, input_iterator_tag>;
2879783f28cSLouis Dionne   using iterator_category = _If<__denotes_forward_iter<_Iter>, forward_iterator_tag, input_iterator_tag>;
2881a29403dSzoecarver   using pointer           = typename __arrow_type_or_void<_Iter, _Sent>::type;
2891a29403dSzoecarver   using value_type        = iter_value_t<_Iter>;
2901a29403dSzoecarver   using difference_type   = iter_difference_t<_Iter>;
2911a29403dSzoecarver   using reference         = iter_reference_t<_Iter>;
2921a29403dSzoecarver };
2931a29403dSzoecarver 
2944f15267dSNikolas Klauser #endif // _LIBCPP_STD_VER >= 20
2951a29403dSzoecarver 
2961a29403dSzoecarver _LIBCPP_END_NAMESPACE_STD
2971a29403dSzoecarver 
29892e4d679SNicole Rabjohn _LIBCPP_POP_MACROS
29992e4d679SNicole Rabjohn 
3001a29403dSzoecarver #endif // _LIBCPP___ITERATOR_COMMON_ITERATOR_H
301