xref: /freebsd-src/contrib/llvm-project/libcxx/include/__ranges/subrange.h (revision 04eeddc0aa8e0a417a16eaf9d7d095207f4a8623)
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>
43*04eeddc0SDimitry Andric   concept __uses_nonqualification_pointer_conversion =
44*04eeddc0SDimitry Andric     is_pointer_v<_From> && is_pointer_v<_To> &&
45*04eeddc0SDimitry Andric     !convertible_to<remove_pointer_t<_From>(*)[], remove_pointer_t<_To>(*)[]>;
46*04eeddc0SDimitry Andric 
47*04eeddc0SDimitry Andric   template<class _From, class _To>
48fe6060f1SDimitry Andric   concept __convertible_to_non_slicing =
49fe6060f1SDimitry Andric     convertible_to<_From, _To> &&
50*04eeddc0SDimitry Andric     !__uses_nonqualification_pointer_conversion<decay_t<_From>, decay_t<_To>>;
51fe6060f1SDimitry Andric 
52fe6060f1SDimitry Andric   template<class _Tp>
53fe6060f1SDimitry Andric   concept __pair_like =
54fe6060f1SDimitry Andric     !is_reference_v<_Tp> && requires(_Tp __t) {
55fe6060f1SDimitry Andric       typename tuple_size<_Tp>::type; // Ensures `tuple_size<T>` is complete.
56fe6060f1SDimitry Andric       requires derived_from<tuple_size<_Tp>, integral_constant<size_t, 2>>;
57fe6060f1SDimitry Andric       typename tuple_element_t<0, remove_const_t<_Tp>>;
58fe6060f1SDimitry Andric       typename tuple_element_t<1, remove_const_t<_Tp>>;
59fe6060f1SDimitry Andric       { _VSTD::get<0>(__t) } -> convertible_to<const tuple_element_t<0, _Tp>&>;
60fe6060f1SDimitry Andric       { _VSTD::get<1>(__t) } -> convertible_to<const tuple_element_t<1, _Tp>&>;
61fe6060f1SDimitry Andric     };
62fe6060f1SDimitry Andric 
63fe6060f1SDimitry Andric   template<class _Pair, class _Iter, class _Sent>
64fe6060f1SDimitry Andric   concept __pair_like_convertible_from =
65fe6060f1SDimitry Andric     !range<_Pair> && __pair_like<_Pair> &&
66fe6060f1SDimitry Andric     constructible_from<_Pair, _Iter, _Sent> &&
67fe6060f1SDimitry Andric     __convertible_to_non_slicing<_Iter, tuple_element_t<0, _Pair>> &&
68fe6060f1SDimitry Andric     convertible_to<_Sent, tuple_element_t<1, _Pair>>;
69fe6060f1SDimitry Andric 
70fe6060f1SDimitry Andric   enum class _LIBCPP_ENUM_VIS subrange_kind : bool { unsized, sized };
71fe6060f1SDimitry Andric 
72fe6060f1SDimitry Andric   template<input_or_output_iterator _Iter, sentinel_for<_Iter> _Sent = _Iter,
73fe6060f1SDimitry Andric            subrange_kind _Kind = sized_sentinel_for<_Sent, _Iter>
74fe6060f1SDimitry Andric              ? subrange_kind::sized
75fe6060f1SDimitry Andric              : subrange_kind::unsized>
76fe6060f1SDimitry Andric     requires (_Kind == subrange_kind::sized || !sized_sentinel_for<_Sent, _Iter>)
77349cc55cSDimitry Andric   class _LIBCPP_TEMPLATE_VIS subrange
78349cc55cSDimitry Andric     : public view_interface<subrange<_Iter, _Sent, _Kind>>
79349cc55cSDimitry Andric   {
80349cc55cSDimitry Andric   private:
81349cc55cSDimitry Andric     static constexpr bool _StoreSize = (_Kind == subrange_kind::sized && !sized_sentinel_for<_Sent, _Iter>);
82349cc55cSDimitry Andric     static constexpr bool _MustProvideSizeAtConstruction = !_StoreSize; // just to improve compiler diagnostics
83349cc55cSDimitry Andric     struct _Empty { constexpr _Empty(auto) noexcept { } };
84349cc55cSDimitry Andric     using _Size = conditional_t<_StoreSize, make_unsigned_t<iter_difference_t<_Iter>>, _Empty>;
85349cc55cSDimitry Andric     [[no_unique_address]] _Iter __begin_ = _Iter();
86349cc55cSDimitry Andric     [[no_unique_address]] _Sent __end_ = _Sent();
87349cc55cSDimitry Andric     [[no_unique_address]] _Size __size_ = 0;
88fe6060f1SDimitry Andric 
89349cc55cSDimitry Andric   public:
90fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
91fe6060f1SDimitry Andric     subrange() requires default_initializable<_Iter> = default;
92fe6060f1SDimitry Andric 
93fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
94fe6060f1SDimitry Andric     constexpr subrange(__convertible_to_non_slicing<_Iter> auto __iter, _Sent __sent)
95349cc55cSDimitry Andric       requires _MustProvideSizeAtConstruction
960eae32dcSDimitry Andric       : __begin_(_VSTD::move(__iter)), __end_(_VSTD::move(__sent))
97349cc55cSDimitry Andric     { }
98fe6060f1SDimitry Andric 
99fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
100fe6060f1SDimitry Andric     constexpr subrange(__convertible_to_non_slicing<_Iter> auto __iter, _Sent __sent,
101fe6060f1SDimitry Andric                        make_unsigned_t<iter_difference_t<_Iter>> __n)
102fe6060f1SDimitry Andric       requires (_Kind == subrange_kind::sized)
1030eae32dcSDimitry Andric       : __begin_(_VSTD::move(__iter)), __end_(_VSTD::move(__sent)), __size_(__n)
104349cc55cSDimitry Andric     {
105349cc55cSDimitry Andric       if constexpr (sized_sentinel_for<_Sent, _Iter>)
106349cc55cSDimitry Andric         _LIBCPP_ASSERT((__end_ - __begin_) == static_cast<iter_difference_t<_Iter>>(__n),
107349cc55cSDimitry Andric           "std::ranges::subrange was passed an invalid size hint");
108349cc55cSDimitry Andric     }
109fe6060f1SDimitry Andric 
110fe6060f1SDimitry Andric     template<__different_from<subrange> _Range>
111fe6060f1SDimitry Andric       requires borrowed_range<_Range> &&
112fe6060f1SDimitry Andric                __convertible_to_non_slicing<iterator_t<_Range>, _Iter> &&
113fe6060f1SDimitry Andric                convertible_to<sentinel_t<_Range>, _Sent>
114fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
115fe6060f1SDimitry Andric     constexpr subrange(_Range&& __range)
116349cc55cSDimitry Andric       requires (!_StoreSize)
117349cc55cSDimitry Andric       : subrange(ranges::begin(__range), ranges::end(__range))
118349cc55cSDimitry Andric     { }
119fe6060f1SDimitry Andric 
120fe6060f1SDimitry Andric     template<__different_from<subrange> _Range>
121fe6060f1SDimitry Andric       requires borrowed_range<_Range> &&
122fe6060f1SDimitry Andric                __convertible_to_non_slicing<iterator_t<_Range>, _Iter> &&
123fe6060f1SDimitry Andric                convertible_to<sentinel_t<_Range>, _Sent>
124fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
125fe6060f1SDimitry Andric     constexpr subrange(_Range&& __range)
126349cc55cSDimitry Andric       requires _StoreSize && sized_range<_Range>
127349cc55cSDimitry Andric       : subrange(__range, ranges::size(__range))
128349cc55cSDimitry Andric     { }
129fe6060f1SDimitry Andric 
130fe6060f1SDimitry Andric     template<borrowed_range _Range>
131fe6060f1SDimitry Andric       requires __convertible_to_non_slicing<iterator_t<_Range>, _Iter> &&
132fe6060f1SDimitry Andric                convertible_to<sentinel_t<_Range>, _Sent>
133fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
134fe6060f1SDimitry Andric     constexpr subrange(_Range&& __range, make_unsigned_t<iter_difference_t<_Iter>> __n)
135fe6060f1SDimitry Andric       requires (_Kind == subrange_kind::sized)
136349cc55cSDimitry Andric       : subrange(ranges::begin(__range), ranges::end(__range), __n)
137349cc55cSDimitry Andric     { }
138fe6060f1SDimitry Andric 
139fe6060f1SDimitry Andric     template<__different_from<subrange> _Pair>
140fe6060f1SDimitry Andric       requires __pair_like_convertible_from<_Pair, const _Iter&, const _Sent&>
141fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
142349cc55cSDimitry Andric     constexpr operator _Pair() const {
143349cc55cSDimitry Andric       return _Pair(__begin_, __end_);
144349cc55cSDimitry Andric     }
145fe6060f1SDimitry Andric 
146fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
147fe6060f1SDimitry Andric     constexpr _Iter begin() const requires copyable<_Iter> {
148349cc55cSDimitry Andric       return __begin_;
149fe6060f1SDimitry Andric     }
150fe6060f1SDimitry Andric 
151fe6060f1SDimitry Andric     [[nodiscard]] _LIBCPP_HIDE_FROM_ABI constexpr _Iter begin() requires (!copyable<_Iter>) {
152349cc55cSDimitry Andric       return _VSTD::move(__begin_);
153fe6060f1SDimitry Andric     }
154fe6060f1SDimitry Andric 
155fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
156349cc55cSDimitry Andric     constexpr _Sent end() const {
157349cc55cSDimitry Andric       return __end_;
158349cc55cSDimitry Andric     }
159fe6060f1SDimitry Andric 
160349cc55cSDimitry Andric     [[nodiscard]] _LIBCPP_HIDE_FROM_ABI constexpr bool empty() const {
161349cc55cSDimitry Andric       return __begin_ == __end_;
162349cc55cSDimitry Andric     }
163fe6060f1SDimitry Andric 
164fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
165fe6060f1SDimitry Andric     constexpr make_unsigned_t<iter_difference_t<_Iter>> size() const
166fe6060f1SDimitry Andric       requires (_Kind == subrange_kind::sized)
167fe6060f1SDimitry Andric     {
168349cc55cSDimitry Andric       if constexpr (_StoreSize)
169349cc55cSDimitry Andric         return __size_;
170fe6060f1SDimitry Andric       else
171349cc55cSDimitry Andric         return _VSTD::__to_unsigned_like(__end_ - __begin_);
172fe6060f1SDimitry Andric     }
173fe6060f1SDimitry Andric 
174fe6060f1SDimitry Andric     [[nodiscard]] _LIBCPP_HIDE_FROM_ABI constexpr subrange next(iter_difference_t<_Iter> __n = 1) const&
175349cc55cSDimitry Andric       requires forward_iterator<_Iter>
176349cc55cSDimitry Andric     {
177fe6060f1SDimitry Andric       auto __tmp = *this;
178fe6060f1SDimitry Andric       __tmp.advance(__n);
179fe6060f1SDimitry Andric       return __tmp;
180fe6060f1SDimitry Andric     }
181fe6060f1SDimitry Andric 
182fe6060f1SDimitry Andric     [[nodiscard]] _LIBCPP_HIDE_FROM_ABI constexpr subrange next(iter_difference_t<_Iter> __n = 1) && {
183fe6060f1SDimitry Andric       advance(__n);
184fe6060f1SDimitry Andric       return _VSTD::move(*this);
185fe6060f1SDimitry Andric     }
186fe6060f1SDimitry Andric 
187fe6060f1SDimitry Andric     [[nodiscard]] _LIBCPP_HIDE_FROM_ABI constexpr subrange prev(iter_difference_t<_Iter> __n = 1) const
188349cc55cSDimitry Andric       requires bidirectional_iterator<_Iter>
189349cc55cSDimitry Andric     {
190fe6060f1SDimitry Andric       auto __tmp = *this;
191fe6060f1SDimitry Andric       __tmp.advance(-__n);
192fe6060f1SDimitry Andric       return __tmp;
193fe6060f1SDimitry Andric     }
194fe6060f1SDimitry Andric 
195fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
196fe6060f1SDimitry Andric     constexpr subrange& advance(iter_difference_t<_Iter> __n) {
197fe6060f1SDimitry Andric       if constexpr (bidirectional_iterator<_Iter>) {
198fe6060f1SDimitry Andric         if (__n < 0) {
199349cc55cSDimitry Andric           ranges::advance(__begin_, __n);
200349cc55cSDimitry Andric           if constexpr (_StoreSize)
201349cc55cSDimitry Andric             __size_ += _VSTD::__to_unsigned_like(-__n);
202fe6060f1SDimitry Andric           return *this;
203fe6060f1SDimitry Andric         }
204fe6060f1SDimitry Andric       }
205fe6060f1SDimitry Andric 
206349cc55cSDimitry Andric       auto __d = __n - ranges::advance(__begin_, __n, __end_);
207349cc55cSDimitry Andric       if constexpr (_StoreSize)
208349cc55cSDimitry Andric         __size_ -= _VSTD::__to_unsigned_like(__d);
209fe6060f1SDimitry Andric       return *this;
210fe6060f1SDimitry Andric     }
211fe6060f1SDimitry Andric   };
212fe6060f1SDimitry Andric 
213fe6060f1SDimitry Andric   template<input_or_output_iterator _Iter, sentinel_for<_Iter> _Sent>
214fe6060f1SDimitry Andric   subrange(_Iter, _Sent) -> subrange<_Iter, _Sent>;
215fe6060f1SDimitry Andric 
216fe6060f1SDimitry Andric   template<input_or_output_iterator _Iter, sentinel_for<_Iter> _Sent>
217fe6060f1SDimitry Andric   subrange(_Iter, _Sent, make_unsigned_t<iter_difference_t<_Iter>>)
218fe6060f1SDimitry Andric     -> subrange<_Iter, _Sent, subrange_kind::sized>;
219fe6060f1SDimitry Andric 
220fe6060f1SDimitry Andric   template<borrowed_range _Range>
221fe6060f1SDimitry Andric   subrange(_Range&&) -> subrange<iterator_t<_Range>, sentinel_t<_Range>,
222fe6060f1SDimitry Andric                                  (sized_range<_Range> || sized_sentinel_for<sentinel_t<_Range>, iterator_t<_Range>>)
223fe6060f1SDimitry Andric                                    ? subrange_kind::sized : subrange_kind::unsized>;
224fe6060f1SDimitry Andric 
225fe6060f1SDimitry Andric   template<borrowed_range _Range>
226fe6060f1SDimitry Andric   subrange(_Range&&, make_unsigned_t<range_difference_t<_Range>>)
227fe6060f1SDimitry Andric     -> subrange<iterator_t<_Range>, sentinel_t<_Range>, subrange_kind::sized>;
228fe6060f1SDimitry Andric 
229fe6060f1SDimitry Andric   template<size_t _Index, class _Iter, class _Sent, subrange_kind _Kind>
230fe6060f1SDimitry Andric     requires (_Index < 2)
231fe6060f1SDimitry Andric   _LIBCPP_HIDE_FROM_ABI
232fe6060f1SDimitry Andric   constexpr auto get(const subrange<_Iter, _Sent, _Kind>& __subrange) {
233fe6060f1SDimitry Andric     if constexpr (_Index == 0)
234fe6060f1SDimitry Andric       return __subrange.begin();
235fe6060f1SDimitry Andric     else
236fe6060f1SDimitry Andric       return __subrange.end();
237fe6060f1SDimitry Andric   }
238fe6060f1SDimitry Andric 
239fe6060f1SDimitry Andric   template<size_t _Index, class _Iter, class _Sent, subrange_kind _Kind>
240fe6060f1SDimitry Andric     requires (_Index < 2)
241fe6060f1SDimitry Andric   _LIBCPP_HIDE_FROM_ABI
242fe6060f1SDimitry Andric   constexpr auto get(subrange<_Iter, _Sent, _Kind>&& __subrange) {
243fe6060f1SDimitry Andric     if constexpr (_Index == 0)
244fe6060f1SDimitry Andric       return __subrange.begin();
245fe6060f1SDimitry Andric     else
246fe6060f1SDimitry Andric       return __subrange.end();
247fe6060f1SDimitry Andric   }
248fe6060f1SDimitry Andric 
249fe6060f1SDimitry Andric   template<class _Ip, class _Sp, subrange_kind _Kp>
250fe6060f1SDimitry Andric   inline constexpr bool enable_borrowed_range<subrange<_Ip, _Sp, _Kp>> = true;
251fe6060f1SDimitry Andric 
252fe6060f1SDimitry Andric   template<range _Rp>
253fe6060f1SDimitry Andric   using borrowed_subrange_t = _If<borrowed_range<_Rp>, subrange<iterator_t<_Rp>>, dangling>;
254fe6060f1SDimitry Andric } // namespace ranges
255fe6060f1SDimitry Andric 
256349cc55cSDimitry Andric // [range.subrange.general]
257349cc55cSDimitry Andric 
258fe6060f1SDimitry Andric using ranges::get;
259fe6060f1SDimitry Andric 
260349cc55cSDimitry Andric // [ranges.syn]
261349cc55cSDimitry Andric 
262349cc55cSDimitry Andric template<class _Ip, class _Sp, ranges::subrange_kind _Kp>
263349cc55cSDimitry Andric struct tuple_size<ranges::subrange<_Ip, _Sp, _Kp>> : integral_constant<size_t, 2> {};
264349cc55cSDimitry Andric 
265349cc55cSDimitry Andric template<class _Ip, class _Sp, ranges::subrange_kind _Kp>
266349cc55cSDimitry Andric struct tuple_element<0, ranges::subrange<_Ip, _Sp, _Kp>> {
267349cc55cSDimitry Andric   using type = _Ip;
268349cc55cSDimitry Andric };
269349cc55cSDimitry Andric 
270349cc55cSDimitry Andric template<class _Ip, class _Sp, ranges::subrange_kind _Kp>
271349cc55cSDimitry Andric struct tuple_element<1, ranges::subrange<_Ip, _Sp, _Kp>> {
272349cc55cSDimitry Andric   using type = _Sp;
273349cc55cSDimitry Andric };
274349cc55cSDimitry Andric 
275349cc55cSDimitry Andric template<class _Ip, class _Sp, ranges::subrange_kind _Kp>
276349cc55cSDimitry Andric struct tuple_element<0, const ranges::subrange<_Ip, _Sp, _Kp>> {
277349cc55cSDimitry Andric   using type = _Ip;
278349cc55cSDimitry Andric };
279349cc55cSDimitry Andric 
280349cc55cSDimitry Andric template<class _Ip, class _Sp, ranges::subrange_kind _Kp>
281349cc55cSDimitry Andric struct tuple_element<1, const ranges::subrange<_Ip, _Sp, _Kp>> {
282349cc55cSDimitry Andric   using type = _Sp;
283349cc55cSDimitry Andric };
284fe6060f1SDimitry Andric 
285fe6060f1SDimitry Andric #endif // !defined(_LIBCPP_HAS_NO_RANGES)
286fe6060f1SDimitry Andric 
287fe6060f1SDimitry Andric _LIBCPP_END_NAMESPACE_STD
288fe6060f1SDimitry Andric 
289fe6060f1SDimitry Andric #endif // _LIBCPP___RANGES_SUBRANGE_H
290