xref: /openbsd-src/gnu/llvm/libcxx/include/__iterator/common_iterator.h (revision 4bdff4bed0e3d54e55670334c7d0077db4170f86)
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