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