176d0caaeSpatrick // -*- C++ -*- 276d0caaeSpatrick //===----------------------------------------------------------------------===// 376d0caaeSpatrick // 476d0caaeSpatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 576d0caaeSpatrick // See https://llvm.org/LICENSE.txt for license information. 676d0caaeSpatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 776d0caaeSpatrick // 876d0caaeSpatrick //===----------------------------------------------------------------------===// 976d0caaeSpatrick 1076d0caaeSpatrick #ifndef _LIBCPP___ITERATOR_COMMON_ITERATOR_H 1176d0caaeSpatrick #define _LIBCPP___ITERATOR_COMMON_ITERATOR_H 1276d0caaeSpatrick 13*4bdff4beSrobert #include <__assert> 14*4bdff4beSrobert #include <__concepts/assignable.h> 15*4bdff4beSrobert #include <__concepts/constructible.h> 16*4bdff4beSrobert #include <__concepts/convertible_to.h> 17*4bdff4beSrobert #include <__concepts/copyable.h> 18*4bdff4beSrobert #include <__concepts/derived_from.h> 19*4bdff4beSrobert #include <__concepts/equality_comparable.h> 20*4bdff4beSrobert #include <__concepts/same_as.h> 2176d0caaeSpatrick #include <__config> 2276d0caaeSpatrick #include <__iterator/concepts.h> 2376d0caaeSpatrick #include <__iterator/incrementable_traits.h> 2476d0caaeSpatrick #include <__iterator/iter_move.h> 2576d0caaeSpatrick #include <__iterator/iter_swap.h> 2676d0caaeSpatrick #include <__iterator/iterator_traits.h> 2776d0caaeSpatrick #include <__iterator/readable_traits.h> 28*4bdff4beSrobert #include <__type_traits/is_pointer.h> 29*4bdff4beSrobert #include <__utility/declval.h> 3076d0caaeSpatrick #include <variant> 3176d0caaeSpatrick 3276d0caaeSpatrick #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER) 3376d0caaeSpatrick # pragma GCC system_header 3476d0caaeSpatrick #endif 3576d0caaeSpatrick 3676d0caaeSpatrick _LIBCPP_BEGIN_NAMESPACE_STD 3776d0caaeSpatrick 38*4bdff4beSrobert #if _LIBCPP_STD_VER > 17 39*4bdff4beSrobert 40*4bdff4beSrobert template<class _Iter> 41*4bdff4beSrobert concept __can_use_postfix_proxy = 42*4bdff4beSrobert constructible_from<iter_value_t<_Iter>, iter_reference_t<_Iter>> && 43*4bdff4beSrobert move_constructible<iter_value_t<_Iter>>; 4476d0caaeSpatrick 4576d0caaeSpatrick template<input_or_output_iterator _Iter, sentinel_for<_Iter> _Sent> 4676d0caaeSpatrick requires (!same_as<_Iter, _Sent> && copyable<_Iter>) 4776d0caaeSpatrick class common_iterator { 48*4bdff4beSrobert struct __proxy { 49*4bdff4beSrobert constexpr const iter_value_t<_Iter>* operator->() const noexcept { 50*4bdff4beSrobert return _VSTD::addressof(__value_); 5176d0caaeSpatrick } 52*4bdff4beSrobert iter_value_t<_Iter> __value_; 5376d0caaeSpatrick }; 5476d0caaeSpatrick 55*4bdff4beSrobert struct __postfix_proxy { 56*4bdff4beSrobert constexpr const iter_value_t<_Iter>& operator*() const noexcept { 57*4bdff4beSrobert return __value_; 5876d0caaeSpatrick } 59*4bdff4beSrobert iter_value_t<_Iter> __value_; 6076d0caaeSpatrick }; 6176d0caaeSpatrick 6276d0caaeSpatrick public: 6376d0caaeSpatrick variant<_Iter, _Sent> __hold_; 6476d0caaeSpatrick 6576d0caaeSpatrick common_iterator() requires default_initializable<_Iter> = default; 6676d0caaeSpatrick common_iterator(_Iter __i)6776d0caaeSpatrick constexpr common_iterator(_Iter __i) : __hold_(in_place_type<_Iter>, _VSTD::move(__i)) {} common_iterator(_Sent __s)6876d0caaeSpatrick constexpr common_iterator(_Sent __s) : __hold_(in_place_type<_Sent>, _VSTD::move(__s)) {} 6976d0caaeSpatrick 7076d0caaeSpatrick template<class _I2, class _S2> 7176d0caaeSpatrick requires convertible_to<const _I2&, _Iter> && convertible_to<const _S2&, _Sent> common_iterator(const common_iterator<_I2,_S2> & __other)7276d0caaeSpatrick constexpr common_iterator(const common_iterator<_I2, _S2>& __other) 7376d0caaeSpatrick : __hold_([&]() -> variant<_Iter, _Sent> { 74*4bdff4beSrobert _LIBCPP_ASSERT(!__other.__hold_.valueless_by_exception(), "Attempted to construct from a valueless common_iterator"); 7576d0caaeSpatrick if (__other.__hold_.index() == 0) 7676d0caaeSpatrick return variant<_Iter, _Sent>{in_place_index<0>, _VSTD::__unchecked_get<0>(__other.__hold_)}; 7776d0caaeSpatrick return variant<_Iter, _Sent>{in_place_index<1>, _VSTD::__unchecked_get<1>(__other.__hold_)}; 7876d0caaeSpatrick }()) {} 7976d0caaeSpatrick 8076d0caaeSpatrick template<class _I2, class _S2> 8176d0caaeSpatrick requires convertible_to<const _I2&, _Iter> && convertible_to<const _S2&, _Sent> && 8276d0caaeSpatrick assignable_from<_Iter&, const _I2&> && assignable_from<_Sent&, const _S2&> 8376d0caaeSpatrick common_iterator& operator=(const common_iterator<_I2, _S2>& __other) { 84*4bdff4beSrobert _LIBCPP_ASSERT(!__other.__hold_.valueless_by_exception(), "Attempted to assign from a valueless common_iterator"); 8576d0caaeSpatrick 8676d0caaeSpatrick auto __idx = __hold_.index(); 8776d0caaeSpatrick auto __other_idx = __other.__hold_.index(); 8876d0caaeSpatrick 8976d0caaeSpatrick // If they're the same index, just assign. 9076d0caaeSpatrick if (__idx == 0 && __other_idx == 0) 9176d0caaeSpatrick _VSTD::__unchecked_get<0>(__hold_) = _VSTD::__unchecked_get<0>(__other.__hold_); 9276d0caaeSpatrick else if (__idx == 1 && __other_idx == 1) 9376d0caaeSpatrick _VSTD::__unchecked_get<1>(__hold_) = _VSTD::__unchecked_get<1>(__other.__hold_); 9476d0caaeSpatrick 9576d0caaeSpatrick // Otherwise replace with the oposite element. 9676d0caaeSpatrick else if (__other_idx == 1) 9776d0caaeSpatrick __hold_.template emplace<1>(_VSTD::__unchecked_get<1>(__other.__hold_)); 9876d0caaeSpatrick else if (__other_idx == 0) 9976d0caaeSpatrick __hold_.template emplace<0>(_VSTD::__unchecked_get<0>(__other.__hold_)); 10076d0caaeSpatrick 10176d0caaeSpatrick return *this; 10276d0caaeSpatrick } 10376d0caaeSpatrick decltype(auto)104*4bdff4beSrobert constexpr decltype(auto) operator*() 10576d0caaeSpatrick { 106*4bdff4beSrobert _LIBCPP_ASSERT(std::holds_alternative<_Iter>(__hold_), "Attempted to dereference a non-dereferenceable common_iterator"); 10776d0caaeSpatrick return *_VSTD::__unchecked_get<_Iter>(__hold_); 10876d0caaeSpatrick } 10976d0caaeSpatrick decltype(auto)110*4bdff4beSrobert constexpr decltype(auto) operator*() const 11176d0caaeSpatrick requires __dereferenceable<const _Iter> 11276d0caaeSpatrick { 113*4bdff4beSrobert _LIBCPP_ASSERT(std::holds_alternative<_Iter>(__hold_), "Attempted to dereference a non-dereferenceable common_iterator"); 11476d0caaeSpatrick return *_VSTD::__unchecked_get<_Iter>(__hold_); 11576d0caaeSpatrick } 11676d0caaeSpatrick 11776d0caaeSpatrick template<class _I2 = _Iter> 11876d0caaeSpatrick decltype(auto) operator->() const 11976d0caaeSpatrick requires indirectly_readable<const _I2> && 12076d0caaeSpatrick (requires(const _I2& __i) { __i.operator->(); } || 12176d0caaeSpatrick is_reference_v<iter_reference_t<_I2>> || 12276d0caaeSpatrick constructible_from<iter_value_t<_I2>, iter_reference_t<_I2>>) 12376d0caaeSpatrick { 124*4bdff4beSrobert _LIBCPP_ASSERT(std::holds_alternative<_Iter>(__hold_), "Attempted to dereference a non-dereferenceable common_iterator"); 12576d0caaeSpatrick if constexpr (is_pointer_v<_Iter> || requires(const _Iter& __i) { __i.operator->(); }) { 12676d0caaeSpatrick return _VSTD::__unchecked_get<_Iter>(__hold_); 12776d0caaeSpatrick } else if constexpr (is_reference_v<iter_reference_t<_Iter>>) { 12876d0caaeSpatrick auto&& __tmp = *_VSTD::__unchecked_get<_Iter>(__hold_); 12976d0caaeSpatrick return _VSTD::addressof(__tmp); 13076d0caaeSpatrick } else { 131*4bdff4beSrobert return __proxy{*_VSTD::__unchecked_get<_Iter>(__hold_)}; 13276d0caaeSpatrick } 13376d0caaeSpatrick } 13476d0caaeSpatrick 13576d0caaeSpatrick common_iterator& operator++() { 136*4bdff4beSrobert _LIBCPP_ASSERT(std::holds_alternative<_Iter>(__hold_), "Attempted to increment a non-dereferenceable common_iterator"); 13776d0caaeSpatrick ++_VSTD::__unchecked_get<_Iter>(__hold_); return *this; 13876d0caaeSpatrick } 13976d0caaeSpatrick 14076d0caaeSpatrick decltype(auto) operator++(int) { 141*4bdff4beSrobert _LIBCPP_ASSERT(std::holds_alternative<_Iter>(__hold_), "Attempted to increment a non-dereferenceable common_iterator"); 14276d0caaeSpatrick if constexpr (forward_iterator<_Iter>) { 14376d0caaeSpatrick auto __tmp = *this; 14476d0caaeSpatrick ++*this; 14576d0caaeSpatrick return __tmp; 146*4bdff4beSrobert } else if constexpr (requires (_Iter& __i) { { *__i++ } -> __can_reference; } || 147*4bdff4beSrobert !__can_use_postfix_proxy<_Iter>) { 14876d0caaeSpatrick return _VSTD::__unchecked_get<_Iter>(__hold_)++; 14976d0caaeSpatrick } else { 150*4bdff4beSrobert auto __p = __postfix_proxy{**this}; 15176d0caaeSpatrick ++*this; 15276d0caaeSpatrick return __p; 15376d0caaeSpatrick } 15476d0caaeSpatrick } 15576d0caaeSpatrick 15676d0caaeSpatrick template<class _I2, sentinel_for<_Iter> _S2> 15776d0caaeSpatrick requires sentinel_for<_Sent, _I2> 158*4bdff4beSrobert _LIBCPP_HIDE_FROM_ABI 159*4bdff4beSrobert friend constexpr bool operator==(const common_iterator& __x, const common_iterator<_I2, _S2>& __y) { 160*4bdff4beSrobert _LIBCPP_ASSERT(!__x.__hold_.valueless_by_exception(), "Attempted to compare a valueless common_iterator"); 161*4bdff4beSrobert _LIBCPP_ASSERT(!__y.__hold_.valueless_by_exception(), "Attempted to compare a valueless common_iterator"); 16276d0caaeSpatrick 16376d0caaeSpatrick auto __x_index = __x.__hold_.index(); 16476d0caaeSpatrick auto __y_index = __y.__hold_.index(); 16576d0caaeSpatrick 16676d0caaeSpatrick if (__x_index == __y_index) 16776d0caaeSpatrick return true; 16876d0caaeSpatrick 16976d0caaeSpatrick if (__x_index == 0) 17076d0caaeSpatrick return _VSTD::__unchecked_get<_Iter>(__x.__hold_) == _VSTD::__unchecked_get<_S2>(__y.__hold_); 17176d0caaeSpatrick 17276d0caaeSpatrick return _VSTD::__unchecked_get<_Sent>(__x.__hold_) == _VSTD::__unchecked_get<_I2>(__y.__hold_); 17376d0caaeSpatrick } 17476d0caaeSpatrick 17576d0caaeSpatrick template<class _I2, sentinel_for<_Iter> _S2> 17676d0caaeSpatrick requires sentinel_for<_Sent, _I2> && equality_comparable_with<_Iter, _I2> 177*4bdff4beSrobert _LIBCPP_HIDE_FROM_ABI 178*4bdff4beSrobert friend constexpr bool operator==(const common_iterator& __x, const common_iterator<_I2, _S2>& __y) { 179*4bdff4beSrobert _LIBCPP_ASSERT(!__x.__hold_.valueless_by_exception(), "Attempted to compare a valueless common_iterator"); 180*4bdff4beSrobert _LIBCPP_ASSERT(!__y.__hold_.valueless_by_exception(), "Attempted to compare a valueless common_iterator"); 18176d0caaeSpatrick 18276d0caaeSpatrick auto __x_index = __x.__hold_.index(); 18376d0caaeSpatrick auto __y_index = __y.__hold_.index(); 18476d0caaeSpatrick 18576d0caaeSpatrick if (__x_index == 1 && __y_index == 1) 18676d0caaeSpatrick return true; 18776d0caaeSpatrick 18876d0caaeSpatrick if (__x_index == 0 && __y_index == 0) 18976d0caaeSpatrick return _VSTD::__unchecked_get<_Iter>(__x.__hold_) == _VSTD::__unchecked_get<_I2>(__y.__hold_); 19076d0caaeSpatrick 19176d0caaeSpatrick if (__x_index == 0) 19276d0caaeSpatrick return _VSTD::__unchecked_get<_Iter>(__x.__hold_) == _VSTD::__unchecked_get<_S2>(__y.__hold_); 19376d0caaeSpatrick 19476d0caaeSpatrick return _VSTD::__unchecked_get<_Sent>(__x.__hold_) == _VSTD::__unchecked_get<_I2>(__y.__hold_); 19576d0caaeSpatrick } 19676d0caaeSpatrick 19776d0caaeSpatrick template<sized_sentinel_for<_Iter> _I2, sized_sentinel_for<_Iter> _S2> 19876d0caaeSpatrick requires sized_sentinel_for<_Sent, _I2> 199*4bdff4beSrobert _LIBCPP_HIDE_FROM_ABI 200*4bdff4beSrobert friend constexpr iter_difference_t<_I2> operator-(const common_iterator& __x, const common_iterator<_I2, _S2>& __y) { 201*4bdff4beSrobert _LIBCPP_ASSERT(!__x.__hold_.valueless_by_exception(), "Attempted to subtract from a valueless common_iterator"); 202*4bdff4beSrobert _LIBCPP_ASSERT(!__y.__hold_.valueless_by_exception(), "Attempted to subtract a valueless common_iterator"); 20376d0caaeSpatrick 20476d0caaeSpatrick auto __x_index = __x.__hold_.index(); 20576d0caaeSpatrick auto __y_index = __y.__hold_.index(); 20676d0caaeSpatrick 20776d0caaeSpatrick if (__x_index == 1 && __y_index == 1) 20876d0caaeSpatrick return 0; 20976d0caaeSpatrick 21076d0caaeSpatrick if (__x_index == 0 && __y_index == 0) 21176d0caaeSpatrick return _VSTD::__unchecked_get<_Iter>(__x.__hold_) - _VSTD::__unchecked_get<_I2>(__y.__hold_); 21276d0caaeSpatrick 21376d0caaeSpatrick if (__x_index == 0) 21476d0caaeSpatrick return _VSTD::__unchecked_get<_Iter>(__x.__hold_) - _VSTD::__unchecked_get<_S2>(__y.__hold_); 21576d0caaeSpatrick 21676d0caaeSpatrick return _VSTD::__unchecked_get<_Sent>(__x.__hold_) - _VSTD::__unchecked_get<_I2>(__y.__hold_); 21776d0caaeSpatrick } 21876d0caaeSpatrick iter_move(const common_iterator & __i)219*4bdff4beSrobert _LIBCPP_HIDE_FROM_ABI friend constexpr iter_rvalue_reference_t<_Iter> iter_move(const common_iterator& __i) 220*4bdff4beSrobert noexcept(noexcept(ranges::iter_move(std::declval<const _Iter&>()))) 22176d0caaeSpatrick requires input_iterator<_Iter> 22276d0caaeSpatrick { 223*4bdff4beSrobert _LIBCPP_ASSERT(std::holds_alternative<_Iter>(__i.__hold_), "Attempted to iter_move a non-dereferenceable common_iterator"); 22476d0caaeSpatrick return ranges::iter_move( _VSTD::__unchecked_get<_Iter>(__i.__hold_)); 22576d0caaeSpatrick } 22676d0caaeSpatrick 22776d0caaeSpatrick template<indirectly_swappable<_Iter> _I2, class _S2> iter_swap(const common_iterator & __x,const common_iterator<_I2,_S2> & __y)228*4bdff4beSrobert _LIBCPP_HIDE_FROM_ABI friend constexpr void iter_swap(const common_iterator& __x, const common_iterator<_I2, _S2>& __y) 229*4bdff4beSrobert noexcept(noexcept(ranges::iter_swap(std::declval<const _Iter&>(), std::declval<const _I2&>()))) 23076d0caaeSpatrick { 231*4bdff4beSrobert _LIBCPP_ASSERT(std::holds_alternative<_Iter>(__x.__hold_), "Attempted to iter_swap a non-dereferenceable common_iterator"); 232*4bdff4beSrobert _LIBCPP_ASSERT(std::holds_alternative<_I2>(__y.__hold_), "Attempted to iter_swap a non-dereferenceable common_iterator"); 233*4bdff4beSrobert return ranges::iter_swap(_VSTD::__unchecked_get<_Iter>(__x.__hold_), _VSTD::__unchecked_get<_I2>(__y.__hold_)); 23476d0caaeSpatrick } 23576d0caaeSpatrick }; 23676d0caaeSpatrick 23776d0caaeSpatrick template<class _Iter, class _Sent> 23876d0caaeSpatrick struct incrementable_traits<common_iterator<_Iter, _Sent>> { 23976d0caaeSpatrick using difference_type = iter_difference_t<_Iter>; 24076d0caaeSpatrick }; 24176d0caaeSpatrick 24276d0caaeSpatrick template<class _Iter> 24376d0caaeSpatrick concept __denotes_forward_iter = 24476d0caaeSpatrick requires { typename iterator_traits<_Iter>::iterator_category; } && 24576d0caaeSpatrick derived_from<typename iterator_traits<_Iter>::iterator_category, forward_iterator_tag>; 24676d0caaeSpatrick 24776d0caaeSpatrick template<class _Iter, class _Sent> 24876d0caaeSpatrick concept __common_iter_has_ptr_op = requires(const common_iterator<_Iter, _Sent>& __a) { 24976d0caaeSpatrick __a.operator->(); 25076d0caaeSpatrick }; 25176d0caaeSpatrick 25276d0caaeSpatrick template<class, class> 25376d0caaeSpatrick struct __arrow_type_or_void { 25476d0caaeSpatrick using type = void; 25576d0caaeSpatrick }; 25676d0caaeSpatrick 25776d0caaeSpatrick template<class _Iter, class _Sent> 25876d0caaeSpatrick requires __common_iter_has_ptr_op<_Iter, _Sent> 25976d0caaeSpatrick struct __arrow_type_or_void<_Iter, _Sent> { 260*4bdff4beSrobert using type = decltype(std::declval<const common_iterator<_Iter, _Sent>&>().operator->()); 26176d0caaeSpatrick }; 26276d0caaeSpatrick 263*4bdff4beSrobert template<input_iterator _Iter, class _Sent> 26476d0caaeSpatrick struct iterator_traits<common_iterator<_Iter, _Sent>> { 26576d0caaeSpatrick using iterator_concept = _If<forward_iterator<_Iter>, 26676d0caaeSpatrick forward_iterator_tag, 26776d0caaeSpatrick input_iterator_tag>; 26876d0caaeSpatrick using iterator_category = _If<__denotes_forward_iter<_Iter>, 26976d0caaeSpatrick forward_iterator_tag, 27076d0caaeSpatrick input_iterator_tag>; 27176d0caaeSpatrick using pointer = typename __arrow_type_or_void<_Iter, _Sent>::type; 27276d0caaeSpatrick using value_type = iter_value_t<_Iter>; 27376d0caaeSpatrick using difference_type = iter_difference_t<_Iter>; 27476d0caaeSpatrick using reference = iter_reference_t<_Iter>; 27576d0caaeSpatrick }; 27676d0caaeSpatrick 277*4bdff4beSrobert #endif // _LIBCPP_STD_VER > 17 27876d0caaeSpatrick 27976d0caaeSpatrick _LIBCPP_END_NAMESPACE_STD 28076d0caaeSpatrick 28176d0caaeSpatrick #endif // _LIBCPP___ITERATOR_COMMON_ITERATOR_H 282