xref: /freebsd-src/contrib/llvm-project/libcxx/include/__ranges/subrange.h (revision 0eae32dcef82f6f06de6419a0d623d7def0cc8f6)
1fe6060f1SDimitry Andric // -*- C++ -*-
2fe6060f1SDimitry Andric //===----------------------------------------------------------------------===//
3fe6060f1SDimitry Andric //
4fe6060f1SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
5fe6060f1SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
6fe6060f1SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7fe6060f1SDimitry Andric //
8fe6060f1SDimitry Andric //===----------------------------------------------------------------------===//
9fe6060f1SDimitry Andric #ifndef _LIBCPP___RANGES_SUBRANGE_H
10fe6060f1SDimitry Andric #define _LIBCPP___RANGES_SUBRANGE_H
11fe6060f1SDimitry Andric 
12349cc55cSDimitry Andric #include <__concepts/constructible.h>
13349cc55cSDimitry Andric #include <__concepts/convertible_to.h>
14349cc55cSDimitry Andric #include <__concepts/copyable.h>
15349cc55cSDimitry Andric #include <__concepts/derived_from.h>
16349cc55cSDimitry Andric #include <__concepts/different_from.h>
17fe6060f1SDimitry Andric #include <__config>
18349cc55cSDimitry Andric #include <__debug>
19349cc55cSDimitry Andric #include <__iterator/advance.h>
20fe6060f1SDimitry Andric #include <__iterator/concepts.h>
21fe6060f1SDimitry Andric #include <__iterator/incrementable_traits.h>
22fe6060f1SDimitry Andric #include <__iterator/iterator_traits.h>
23fe6060f1SDimitry Andric #include <__ranges/access.h>
24fe6060f1SDimitry Andric #include <__ranges/concepts.h>
25fe6060f1SDimitry Andric #include <__ranges/dangling.h>
26fe6060f1SDimitry Andric #include <__ranges/enable_borrowed_range.h>
27fe6060f1SDimitry Andric #include <__ranges/size.h>
28fe6060f1SDimitry Andric #include <__ranges/view_interface.h>
29349cc55cSDimitry Andric #include <__tuple>
30349cc55cSDimitry Andric #include <__utility/move.h>
31fe6060f1SDimitry Andric #include <type_traits>
32fe6060f1SDimitry Andric 
33fe6060f1SDimitry Andric #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
34fe6060f1SDimitry Andric #pragma GCC system_header
35fe6060f1SDimitry Andric #endif
36fe6060f1SDimitry Andric 
37fe6060f1SDimitry Andric _LIBCPP_BEGIN_NAMESPACE_STD
38fe6060f1SDimitry Andric 
39fe6060f1SDimitry Andric #if !defined(_LIBCPP_HAS_NO_RANGES)
40fe6060f1SDimitry Andric 
41fe6060f1SDimitry Andric namespace ranges {
42fe6060f1SDimitry Andric   template<class _From, class _To>
43fe6060f1SDimitry Andric   concept __convertible_to_non_slicing =
44fe6060f1SDimitry Andric     convertible_to<_From, _To> &&
45fe6060f1SDimitry Andric     // If they're both pointers, they must have the same element type.
46fe6060f1SDimitry Andric     !(is_pointer_v<decay_t<_From>> &&
47fe6060f1SDimitry Andric       is_pointer_v<decay_t<_To>> &&
48fe6060f1SDimitry Andric       __different_from<remove_pointer_t<decay_t<_From>>, remove_pointer_t<decay_t<_To>>>);
49fe6060f1SDimitry Andric 
50fe6060f1SDimitry Andric   template<class _Tp>
51fe6060f1SDimitry Andric   concept __pair_like =
52fe6060f1SDimitry Andric     !is_reference_v<_Tp> && requires(_Tp __t) {
53fe6060f1SDimitry Andric       typename tuple_size<_Tp>::type; // Ensures `tuple_size<T>` is complete.
54fe6060f1SDimitry Andric       requires derived_from<tuple_size<_Tp>, integral_constant<size_t, 2>>;
55fe6060f1SDimitry Andric       typename tuple_element_t<0, remove_const_t<_Tp>>;
56fe6060f1SDimitry Andric       typename tuple_element_t<1, remove_const_t<_Tp>>;
57fe6060f1SDimitry Andric       { _VSTD::get<0>(__t) } -> convertible_to<const tuple_element_t<0, _Tp>&>;
58fe6060f1SDimitry Andric       { _VSTD::get<1>(__t) } -> convertible_to<const tuple_element_t<1, _Tp>&>;
59fe6060f1SDimitry Andric     };
60fe6060f1SDimitry Andric 
61fe6060f1SDimitry Andric   template<class _Pair, class _Iter, class _Sent>
62fe6060f1SDimitry Andric   concept __pair_like_convertible_from =
63fe6060f1SDimitry Andric     !range<_Pair> && __pair_like<_Pair> &&
64fe6060f1SDimitry Andric     constructible_from<_Pair, _Iter, _Sent> &&
65fe6060f1SDimitry Andric     __convertible_to_non_slicing<_Iter, tuple_element_t<0, _Pair>> &&
66fe6060f1SDimitry Andric     convertible_to<_Sent, tuple_element_t<1, _Pair>>;
67fe6060f1SDimitry Andric 
68fe6060f1SDimitry Andric   enum class _LIBCPP_ENUM_VIS subrange_kind : bool { unsized, sized };
69fe6060f1SDimitry Andric 
70fe6060f1SDimitry Andric   template<input_or_output_iterator _Iter, sentinel_for<_Iter> _Sent = _Iter,
71fe6060f1SDimitry Andric            subrange_kind _Kind = sized_sentinel_for<_Sent, _Iter>
72fe6060f1SDimitry Andric              ? subrange_kind::sized
73fe6060f1SDimitry Andric              : subrange_kind::unsized>
74fe6060f1SDimitry Andric     requires (_Kind == subrange_kind::sized || !sized_sentinel_for<_Sent, _Iter>)
75349cc55cSDimitry Andric   class _LIBCPP_TEMPLATE_VIS subrange
76349cc55cSDimitry Andric     : public view_interface<subrange<_Iter, _Sent, _Kind>>
77349cc55cSDimitry Andric   {
78349cc55cSDimitry Andric   private:
79349cc55cSDimitry Andric     static constexpr bool _StoreSize = (_Kind == subrange_kind::sized && !sized_sentinel_for<_Sent, _Iter>);
80349cc55cSDimitry Andric     static constexpr bool _MustProvideSizeAtConstruction = !_StoreSize; // just to improve compiler diagnostics
81349cc55cSDimitry Andric     struct _Empty { constexpr _Empty(auto) noexcept { } };
82349cc55cSDimitry Andric     using _Size = conditional_t<_StoreSize, make_unsigned_t<iter_difference_t<_Iter>>, _Empty>;
83349cc55cSDimitry Andric     [[no_unique_address]] _Iter __begin_ = _Iter();
84349cc55cSDimitry Andric     [[no_unique_address]] _Sent __end_ = _Sent();
85349cc55cSDimitry Andric     [[no_unique_address]] _Size __size_ = 0;
86fe6060f1SDimitry Andric 
87349cc55cSDimitry Andric   public:
88fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
89fe6060f1SDimitry Andric     subrange() requires default_initializable<_Iter> = default;
90fe6060f1SDimitry Andric 
91fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
92fe6060f1SDimitry Andric     constexpr subrange(__convertible_to_non_slicing<_Iter> auto __iter, _Sent __sent)
93349cc55cSDimitry Andric       requires _MustProvideSizeAtConstruction
94*0eae32dcSDimitry Andric       : __begin_(_VSTD::move(__iter)), __end_(_VSTD::move(__sent))
95349cc55cSDimitry Andric     { }
96fe6060f1SDimitry Andric 
97fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
98fe6060f1SDimitry Andric     constexpr subrange(__convertible_to_non_slicing<_Iter> auto __iter, _Sent __sent,
99fe6060f1SDimitry Andric                        make_unsigned_t<iter_difference_t<_Iter>> __n)
100fe6060f1SDimitry Andric       requires (_Kind == subrange_kind::sized)
101*0eae32dcSDimitry Andric       : __begin_(_VSTD::move(__iter)), __end_(_VSTD::move(__sent)), __size_(__n)
102349cc55cSDimitry Andric     {
103349cc55cSDimitry Andric       if constexpr (sized_sentinel_for<_Sent, _Iter>)
104349cc55cSDimitry Andric         _LIBCPP_ASSERT((__end_ - __begin_) == static_cast<iter_difference_t<_Iter>>(__n),
105349cc55cSDimitry Andric           "std::ranges::subrange was passed an invalid size hint");
106349cc55cSDimitry Andric     }
107fe6060f1SDimitry Andric 
108fe6060f1SDimitry Andric     template<__different_from<subrange> _Range>
109fe6060f1SDimitry Andric       requires borrowed_range<_Range> &&
110fe6060f1SDimitry Andric                __convertible_to_non_slicing<iterator_t<_Range>, _Iter> &&
111fe6060f1SDimitry Andric                convertible_to<sentinel_t<_Range>, _Sent>
112fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
113fe6060f1SDimitry Andric     constexpr subrange(_Range&& __range)
114349cc55cSDimitry Andric       requires (!_StoreSize)
115349cc55cSDimitry Andric       : subrange(ranges::begin(__range), ranges::end(__range))
116349cc55cSDimitry Andric     { }
117fe6060f1SDimitry Andric 
118fe6060f1SDimitry Andric     template<__different_from<subrange> _Range>
119fe6060f1SDimitry Andric       requires borrowed_range<_Range> &&
120fe6060f1SDimitry Andric                __convertible_to_non_slicing<iterator_t<_Range>, _Iter> &&
121fe6060f1SDimitry Andric                convertible_to<sentinel_t<_Range>, _Sent>
122fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
123fe6060f1SDimitry Andric     constexpr subrange(_Range&& __range)
124349cc55cSDimitry Andric       requires _StoreSize && sized_range<_Range>
125349cc55cSDimitry Andric       : subrange(__range, ranges::size(__range))
126349cc55cSDimitry Andric     { }
127fe6060f1SDimitry Andric 
128fe6060f1SDimitry Andric     template<borrowed_range _Range>
129fe6060f1SDimitry Andric       requires __convertible_to_non_slicing<iterator_t<_Range>, _Iter> &&
130fe6060f1SDimitry Andric                convertible_to<sentinel_t<_Range>, _Sent>
131fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
132fe6060f1SDimitry Andric     constexpr subrange(_Range&& __range, make_unsigned_t<iter_difference_t<_Iter>> __n)
133fe6060f1SDimitry Andric       requires (_Kind == subrange_kind::sized)
134349cc55cSDimitry Andric       : subrange(ranges::begin(__range), ranges::end(__range), __n)
135349cc55cSDimitry Andric     { }
136fe6060f1SDimitry Andric 
137fe6060f1SDimitry Andric     template<__different_from<subrange> _Pair>
138fe6060f1SDimitry Andric       requires __pair_like_convertible_from<_Pair, const _Iter&, const _Sent&>
139fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
140349cc55cSDimitry Andric     constexpr operator _Pair() const {
141349cc55cSDimitry Andric       return _Pair(__begin_, __end_);
142349cc55cSDimitry Andric     }
143fe6060f1SDimitry Andric 
144fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
145fe6060f1SDimitry Andric     constexpr _Iter begin() const requires copyable<_Iter> {
146349cc55cSDimitry Andric       return __begin_;
147fe6060f1SDimitry Andric     }
148fe6060f1SDimitry Andric 
149fe6060f1SDimitry Andric     [[nodiscard]] _LIBCPP_HIDE_FROM_ABI constexpr _Iter begin() requires (!copyable<_Iter>) {
150349cc55cSDimitry Andric       return _VSTD::move(__begin_);
151fe6060f1SDimitry Andric     }
152fe6060f1SDimitry Andric 
153fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
154349cc55cSDimitry Andric     constexpr _Sent end() const {
155349cc55cSDimitry Andric       return __end_;
156349cc55cSDimitry Andric     }
157fe6060f1SDimitry Andric 
158349cc55cSDimitry Andric     [[nodiscard]] _LIBCPP_HIDE_FROM_ABI constexpr bool empty() const {
159349cc55cSDimitry Andric       return __begin_ == __end_;
160349cc55cSDimitry Andric     }
161fe6060f1SDimitry Andric 
162fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
163fe6060f1SDimitry Andric     constexpr make_unsigned_t<iter_difference_t<_Iter>> size() const
164fe6060f1SDimitry Andric       requires (_Kind == subrange_kind::sized)
165fe6060f1SDimitry Andric     {
166349cc55cSDimitry Andric       if constexpr (_StoreSize)
167349cc55cSDimitry Andric         return __size_;
168fe6060f1SDimitry Andric       else
169349cc55cSDimitry Andric         return _VSTD::__to_unsigned_like(__end_ - __begin_);
170fe6060f1SDimitry Andric     }
171fe6060f1SDimitry Andric 
172fe6060f1SDimitry Andric     [[nodiscard]] _LIBCPP_HIDE_FROM_ABI constexpr subrange next(iter_difference_t<_Iter> __n = 1) const&
173349cc55cSDimitry Andric       requires forward_iterator<_Iter>
174349cc55cSDimitry Andric     {
175fe6060f1SDimitry Andric       auto __tmp = *this;
176fe6060f1SDimitry Andric       __tmp.advance(__n);
177fe6060f1SDimitry Andric       return __tmp;
178fe6060f1SDimitry Andric     }
179fe6060f1SDimitry Andric 
180fe6060f1SDimitry Andric     [[nodiscard]] _LIBCPP_HIDE_FROM_ABI constexpr subrange next(iter_difference_t<_Iter> __n = 1) && {
181fe6060f1SDimitry Andric       advance(__n);
182fe6060f1SDimitry Andric       return _VSTD::move(*this);
183fe6060f1SDimitry Andric     }
184fe6060f1SDimitry Andric 
185fe6060f1SDimitry Andric     [[nodiscard]] _LIBCPP_HIDE_FROM_ABI constexpr subrange prev(iter_difference_t<_Iter> __n = 1) const
186349cc55cSDimitry Andric       requires bidirectional_iterator<_Iter>
187349cc55cSDimitry Andric     {
188fe6060f1SDimitry Andric       auto __tmp = *this;
189fe6060f1SDimitry Andric       __tmp.advance(-__n);
190fe6060f1SDimitry Andric       return __tmp;
191fe6060f1SDimitry Andric     }
192fe6060f1SDimitry Andric 
193fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
194fe6060f1SDimitry Andric     constexpr subrange& advance(iter_difference_t<_Iter> __n) {
195fe6060f1SDimitry Andric       if constexpr (bidirectional_iterator<_Iter>) {
196fe6060f1SDimitry Andric         if (__n < 0) {
197349cc55cSDimitry Andric           ranges::advance(__begin_, __n);
198349cc55cSDimitry Andric           if constexpr (_StoreSize)
199349cc55cSDimitry Andric             __size_ += _VSTD::__to_unsigned_like(-__n);
200fe6060f1SDimitry Andric           return *this;
201fe6060f1SDimitry Andric         }
202fe6060f1SDimitry Andric       }
203fe6060f1SDimitry Andric 
204349cc55cSDimitry Andric       auto __d = __n - ranges::advance(__begin_, __n, __end_);
205349cc55cSDimitry Andric       if constexpr (_StoreSize)
206349cc55cSDimitry Andric         __size_ -= _VSTD::__to_unsigned_like(__d);
207fe6060f1SDimitry Andric       return *this;
208fe6060f1SDimitry Andric     }
209fe6060f1SDimitry Andric   };
210fe6060f1SDimitry Andric 
211fe6060f1SDimitry Andric   template<input_or_output_iterator _Iter, sentinel_for<_Iter> _Sent>
212fe6060f1SDimitry Andric   subrange(_Iter, _Sent) -> subrange<_Iter, _Sent>;
213fe6060f1SDimitry Andric 
214fe6060f1SDimitry Andric   template<input_or_output_iterator _Iter, sentinel_for<_Iter> _Sent>
215fe6060f1SDimitry Andric   subrange(_Iter, _Sent, make_unsigned_t<iter_difference_t<_Iter>>)
216fe6060f1SDimitry Andric     -> subrange<_Iter, _Sent, subrange_kind::sized>;
217fe6060f1SDimitry Andric 
218fe6060f1SDimitry Andric   template<borrowed_range _Range>
219fe6060f1SDimitry Andric   subrange(_Range&&) -> subrange<iterator_t<_Range>, sentinel_t<_Range>,
220fe6060f1SDimitry Andric                                  (sized_range<_Range> || sized_sentinel_for<sentinel_t<_Range>, iterator_t<_Range>>)
221fe6060f1SDimitry Andric                                    ? subrange_kind::sized : subrange_kind::unsized>;
222fe6060f1SDimitry Andric 
223fe6060f1SDimitry Andric   template<borrowed_range _Range>
224fe6060f1SDimitry Andric   subrange(_Range&&, make_unsigned_t<range_difference_t<_Range>>)
225fe6060f1SDimitry Andric     -> subrange<iterator_t<_Range>, sentinel_t<_Range>, subrange_kind::sized>;
226fe6060f1SDimitry Andric 
227fe6060f1SDimitry Andric   template<size_t _Index, class _Iter, class _Sent, subrange_kind _Kind>
228fe6060f1SDimitry Andric     requires (_Index < 2)
229fe6060f1SDimitry Andric   _LIBCPP_HIDE_FROM_ABI
230fe6060f1SDimitry Andric   constexpr auto get(const subrange<_Iter, _Sent, _Kind>& __subrange) {
231fe6060f1SDimitry Andric     if constexpr (_Index == 0)
232fe6060f1SDimitry Andric       return __subrange.begin();
233fe6060f1SDimitry Andric     else
234fe6060f1SDimitry Andric       return __subrange.end();
235fe6060f1SDimitry Andric   }
236fe6060f1SDimitry Andric 
237fe6060f1SDimitry Andric   template<size_t _Index, class _Iter, class _Sent, subrange_kind _Kind>
238fe6060f1SDimitry Andric     requires (_Index < 2)
239fe6060f1SDimitry Andric   _LIBCPP_HIDE_FROM_ABI
240fe6060f1SDimitry Andric   constexpr auto get(subrange<_Iter, _Sent, _Kind>&& __subrange) {
241fe6060f1SDimitry Andric     if constexpr (_Index == 0)
242fe6060f1SDimitry Andric       return __subrange.begin();
243fe6060f1SDimitry Andric     else
244fe6060f1SDimitry Andric       return __subrange.end();
245fe6060f1SDimitry Andric   }
246fe6060f1SDimitry Andric 
247fe6060f1SDimitry Andric   template<class _Ip, class _Sp, subrange_kind _Kp>
248fe6060f1SDimitry Andric   inline constexpr bool enable_borrowed_range<subrange<_Ip, _Sp, _Kp>> = true;
249fe6060f1SDimitry Andric 
250fe6060f1SDimitry Andric   template<range _Rp>
251fe6060f1SDimitry Andric   using borrowed_subrange_t = _If<borrowed_range<_Rp>, subrange<iterator_t<_Rp>>, dangling>;
252fe6060f1SDimitry Andric } // namespace ranges
253fe6060f1SDimitry Andric 
254349cc55cSDimitry Andric // [range.subrange.general]
255349cc55cSDimitry Andric 
256fe6060f1SDimitry Andric using ranges::get;
257fe6060f1SDimitry Andric 
258349cc55cSDimitry Andric // [ranges.syn]
259349cc55cSDimitry Andric 
260349cc55cSDimitry Andric template<class _Ip, class _Sp, ranges::subrange_kind _Kp>
261349cc55cSDimitry Andric struct tuple_size<ranges::subrange<_Ip, _Sp, _Kp>> : integral_constant<size_t, 2> {};
262349cc55cSDimitry Andric 
263349cc55cSDimitry Andric template<class _Ip, class _Sp, ranges::subrange_kind _Kp>
264349cc55cSDimitry Andric struct tuple_element<0, ranges::subrange<_Ip, _Sp, _Kp>> {
265349cc55cSDimitry Andric   using type = _Ip;
266349cc55cSDimitry Andric };
267349cc55cSDimitry Andric 
268349cc55cSDimitry Andric template<class _Ip, class _Sp, ranges::subrange_kind _Kp>
269349cc55cSDimitry Andric struct tuple_element<1, ranges::subrange<_Ip, _Sp, _Kp>> {
270349cc55cSDimitry Andric   using type = _Sp;
271349cc55cSDimitry Andric };
272349cc55cSDimitry Andric 
273349cc55cSDimitry Andric template<class _Ip, class _Sp, ranges::subrange_kind _Kp>
274349cc55cSDimitry Andric struct tuple_element<0, const ranges::subrange<_Ip, _Sp, _Kp>> {
275349cc55cSDimitry Andric   using type = _Ip;
276349cc55cSDimitry Andric };
277349cc55cSDimitry Andric 
278349cc55cSDimitry Andric template<class _Ip, class _Sp, ranges::subrange_kind _Kp>
279349cc55cSDimitry Andric struct tuple_element<1, const ranges::subrange<_Ip, _Sp, _Kp>> {
280349cc55cSDimitry Andric   using type = _Sp;
281349cc55cSDimitry Andric };
282fe6060f1SDimitry Andric 
283fe6060f1SDimitry Andric #endif // !defined(_LIBCPP_HAS_NO_RANGES)
284fe6060f1SDimitry Andric 
285fe6060f1SDimitry Andric _LIBCPP_END_NAMESPACE_STD
286fe6060f1SDimitry Andric 
287fe6060f1SDimitry Andric #endif // _LIBCPP___RANGES_SUBRANGE_H
288