xref: /freebsd-src/contrib/llvm-project/libcxx/include/__ranges/subrange.h (revision 81ad626541db97eb356e2c1d4a20eb2a26a766ab)
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 
12*81ad6265SDimitry Andric #include <__assert>
13349cc55cSDimitry Andric #include <__concepts/constructible.h>
14349cc55cSDimitry Andric #include <__concepts/convertible_to.h>
15349cc55cSDimitry Andric #include <__concepts/copyable.h>
16349cc55cSDimitry Andric #include <__concepts/derived_from.h>
17349cc55cSDimitry Andric #include <__concepts/different_from.h>
18fe6060f1SDimitry Andric #include <__config>
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 
39*81ad6265SDimitry Andric #if _LIBCPP_STD_VER > 17 && !defined(_LIBCPP_HAS_NO_INCOMPLETE_RANGES)
40fe6060f1SDimitry Andric 
41fe6060f1SDimitry Andric namespace ranges {
42fe6060f1SDimitry Andric   template<class _From, class _To>
4304eeddc0SDimitry Andric   concept __uses_nonqualification_pointer_conversion =
4404eeddc0SDimitry Andric     is_pointer_v<_From> && is_pointer_v<_To> &&
4504eeddc0SDimitry Andric     !convertible_to<remove_pointer_t<_From>(*)[], remove_pointer_t<_To>(*)[]>;
4604eeddc0SDimitry Andric 
4704eeddc0SDimitry Andric   template<class _From, class _To>
48fe6060f1SDimitry Andric   concept __convertible_to_non_slicing =
49fe6060f1SDimitry Andric     convertible_to<_From, _To> &&
5004eeddc0SDimitry 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>>;
59*81ad6265SDimitry Andric       { std::get<0>(__t) } -> convertible_to<const tuple_element_t<0, _Tp>&>;
60*81ad6265SDimitry Andric       { std::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   {
80*81ad6265SDimitry Andric   public:
81*81ad6265SDimitry Andric     // Note: this is an internal implementation detail that is public only for internal usage.
82349cc55cSDimitry Andric     static constexpr bool _StoreSize = (_Kind == subrange_kind::sized && !sized_sentinel_for<_Sent, _Iter>);
83*81ad6265SDimitry Andric 
84*81ad6265SDimitry Andric   private:
85349cc55cSDimitry Andric     static constexpr bool _MustProvideSizeAtConstruction = !_StoreSize; // just to improve compiler diagnostics
86349cc55cSDimitry Andric     struct _Empty { constexpr _Empty(auto) noexcept { } };
87349cc55cSDimitry Andric     using _Size = conditional_t<_StoreSize, make_unsigned_t<iter_difference_t<_Iter>>, _Empty>;
88*81ad6265SDimitry Andric     _LIBCPP_NO_UNIQUE_ADDRESS _Iter __begin_ = _Iter();
89*81ad6265SDimitry Andric     _LIBCPP_NO_UNIQUE_ADDRESS _Sent __end_ = _Sent();
90*81ad6265SDimitry Andric     _LIBCPP_NO_UNIQUE_ADDRESS _Size __size_ = 0;
91fe6060f1SDimitry Andric 
92349cc55cSDimitry Andric   public:
93fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
94fe6060f1SDimitry Andric     subrange() requires default_initializable<_Iter> = default;
95fe6060f1SDimitry Andric 
96fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
97fe6060f1SDimitry Andric     constexpr subrange(__convertible_to_non_slicing<_Iter> auto __iter, _Sent __sent)
98349cc55cSDimitry Andric       requires _MustProvideSizeAtConstruction
99*81ad6265SDimitry Andric       : __begin_(std::move(__iter)), __end_(std::move(__sent))
100349cc55cSDimitry Andric     { }
101fe6060f1SDimitry Andric 
102fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
103fe6060f1SDimitry Andric     constexpr subrange(__convertible_to_non_slicing<_Iter> auto __iter, _Sent __sent,
104fe6060f1SDimitry Andric                        make_unsigned_t<iter_difference_t<_Iter>> __n)
105fe6060f1SDimitry Andric       requires (_Kind == subrange_kind::sized)
106*81ad6265SDimitry Andric       : __begin_(std::move(__iter)), __end_(std::move(__sent)), __size_(__n)
107349cc55cSDimitry Andric     {
108349cc55cSDimitry Andric       if constexpr (sized_sentinel_for<_Sent, _Iter>)
109349cc55cSDimitry Andric         _LIBCPP_ASSERT((__end_ - __begin_) == static_cast<iter_difference_t<_Iter>>(__n),
110349cc55cSDimitry Andric           "std::ranges::subrange was passed an invalid size hint");
111349cc55cSDimitry Andric     }
112fe6060f1SDimitry Andric 
113fe6060f1SDimitry Andric     template<__different_from<subrange> _Range>
114fe6060f1SDimitry Andric       requires borrowed_range<_Range> &&
115fe6060f1SDimitry Andric                __convertible_to_non_slicing<iterator_t<_Range>, _Iter> &&
116fe6060f1SDimitry Andric                convertible_to<sentinel_t<_Range>, _Sent>
117fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
118fe6060f1SDimitry Andric     constexpr subrange(_Range&& __range)
119349cc55cSDimitry Andric       requires (!_StoreSize)
120349cc55cSDimitry Andric       : subrange(ranges::begin(__range), ranges::end(__range))
121349cc55cSDimitry Andric     { }
122fe6060f1SDimitry Andric 
123fe6060f1SDimitry Andric     template<__different_from<subrange> _Range>
124fe6060f1SDimitry Andric       requires borrowed_range<_Range> &&
125fe6060f1SDimitry Andric                __convertible_to_non_slicing<iterator_t<_Range>, _Iter> &&
126fe6060f1SDimitry Andric                convertible_to<sentinel_t<_Range>, _Sent>
127fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
128fe6060f1SDimitry Andric     constexpr subrange(_Range&& __range)
129349cc55cSDimitry Andric       requires _StoreSize && sized_range<_Range>
130349cc55cSDimitry Andric       : subrange(__range, ranges::size(__range))
131349cc55cSDimitry Andric     { }
132fe6060f1SDimitry Andric 
133fe6060f1SDimitry Andric     template<borrowed_range _Range>
134fe6060f1SDimitry Andric       requires __convertible_to_non_slicing<iterator_t<_Range>, _Iter> &&
135fe6060f1SDimitry Andric                convertible_to<sentinel_t<_Range>, _Sent>
136fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
137fe6060f1SDimitry Andric     constexpr subrange(_Range&& __range, make_unsigned_t<iter_difference_t<_Iter>> __n)
138fe6060f1SDimitry Andric       requires (_Kind == subrange_kind::sized)
139349cc55cSDimitry Andric       : subrange(ranges::begin(__range), ranges::end(__range), __n)
140349cc55cSDimitry Andric     { }
141fe6060f1SDimitry Andric 
142fe6060f1SDimitry Andric     template<__different_from<subrange> _Pair>
143fe6060f1SDimitry Andric       requires __pair_like_convertible_from<_Pair, const _Iter&, const _Sent&>
144fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
145349cc55cSDimitry Andric     constexpr operator _Pair() const {
146349cc55cSDimitry Andric       return _Pair(__begin_, __end_);
147349cc55cSDimitry Andric     }
148fe6060f1SDimitry Andric 
149fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
150fe6060f1SDimitry Andric     constexpr _Iter begin() const requires copyable<_Iter> {
151349cc55cSDimitry Andric       return __begin_;
152fe6060f1SDimitry Andric     }
153fe6060f1SDimitry Andric 
154fe6060f1SDimitry Andric     [[nodiscard]] _LIBCPP_HIDE_FROM_ABI constexpr _Iter begin() requires (!copyable<_Iter>) {
155*81ad6265SDimitry Andric       return std::move(__begin_);
156fe6060f1SDimitry Andric     }
157fe6060f1SDimitry Andric 
158fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
159349cc55cSDimitry Andric     constexpr _Sent end() const {
160349cc55cSDimitry Andric       return __end_;
161349cc55cSDimitry Andric     }
162fe6060f1SDimitry Andric 
163349cc55cSDimitry Andric     [[nodiscard]] _LIBCPP_HIDE_FROM_ABI constexpr bool empty() const {
164349cc55cSDimitry Andric       return __begin_ == __end_;
165349cc55cSDimitry Andric     }
166fe6060f1SDimitry Andric 
167fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
168fe6060f1SDimitry Andric     constexpr make_unsigned_t<iter_difference_t<_Iter>> size() const
169fe6060f1SDimitry Andric       requires (_Kind == subrange_kind::sized)
170fe6060f1SDimitry Andric     {
171349cc55cSDimitry Andric       if constexpr (_StoreSize)
172349cc55cSDimitry Andric         return __size_;
173fe6060f1SDimitry Andric       else
174*81ad6265SDimitry Andric         return std::__to_unsigned_like(__end_ - __begin_);
175fe6060f1SDimitry Andric     }
176fe6060f1SDimitry Andric 
177fe6060f1SDimitry Andric     [[nodiscard]] _LIBCPP_HIDE_FROM_ABI constexpr subrange next(iter_difference_t<_Iter> __n = 1) const&
178349cc55cSDimitry Andric       requires forward_iterator<_Iter>
179349cc55cSDimitry Andric     {
180fe6060f1SDimitry Andric       auto __tmp = *this;
181fe6060f1SDimitry Andric       __tmp.advance(__n);
182fe6060f1SDimitry Andric       return __tmp;
183fe6060f1SDimitry Andric     }
184fe6060f1SDimitry Andric 
185fe6060f1SDimitry Andric     [[nodiscard]] _LIBCPP_HIDE_FROM_ABI constexpr subrange next(iter_difference_t<_Iter> __n = 1) && {
186fe6060f1SDimitry Andric       advance(__n);
187*81ad6265SDimitry Andric       return std::move(*this);
188fe6060f1SDimitry Andric     }
189fe6060f1SDimitry Andric 
190fe6060f1SDimitry Andric     [[nodiscard]] _LIBCPP_HIDE_FROM_ABI constexpr subrange prev(iter_difference_t<_Iter> __n = 1) const
191349cc55cSDimitry Andric       requires bidirectional_iterator<_Iter>
192349cc55cSDimitry Andric     {
193fe6060f1SDimitry Andric       auto __tmp = *this;
194fe6060f1SDimitry Andric       __tmp.advance(-__n);
195fe6060f1SDimitry Andric       return __tmp;
196fe6060f1SDimitry Andric     }
197fe6060f1SDimitry Andric 
198fe6060f1SDimitry Andric     _LIBCPP_HIDE_FROM_ABI
199fe6060f1SDimitry Andric     constexpr subrange& advance(iter_difference_t<_Iter> __n) {
200fe6060f1SDimitry Andric       if constexpr (bidirectional_iterator<_Iter>) {
201fe6060f1SDimitry Andric         if (__n < 0) {
202349cc55cSDimitry Andric           ranges::advance(__begin_, __n);
203349cc55cSDimitry Andric           if constexpr (_StoreSize)
204*81ad6265SDimitry Andric             __size_ += std::__to_unsigned_like(-__n);
205fe6060f1SDimitry Andric           return *this;
206fe6060f1SDimitry Andric         }
207fe6060f1SDimitry Andric       }
208fe6060f1SDimitry Andric 
209349cc55cSDimitry Andric       auto __d = __n - ranges::advance(__begin_, __n, __end_);
210349cc55cSDimitry Andric       if constexpr (_StoreSize)
211*81ad6265SDimitry Andric         __size_ -= std::__to_unsigned_like(__d);
212fe6060f1SDimitry Andric       return *this;
213fe6060f1SDimitry Andric     }
214fe6060f1SDimitry Andric   };
215fe6060f1SDimitry Andric 
216fe6060f1SDimitry Andric   template<input_or_output_iterator _Iter, sentinel_for<_Iter> _Sent>
217fe6060f1SDimitry Andric   subrange(_Iter, _Sent) -> subrange<_Iter, _Sent>;
218fe6060f1SDimitry Andric 
219fe6060f1SDimitry Andric   template<input_or_output_iterator _Iter, sentinel_for<_Iter> _Sent>
220fe6060f1SDimitry Andric   subrange(_Iter, _Sent, make_unsigned_t<iter_difference_t<_Iter>>)
221fe6060f1SDimitry Andric     -> subrange<_Iter, _Sent, subrange_kind::sized>;
222fe6060f1SDimitry Andric 
223fe6060f1SDimitry Andric   template<borrowed_range _Range>
224fe6060f1SDimitry Andric   subrange(_Range&&) -> subrange<iterator_t<_Range>, sentinel_t<_Range>,
225fe6060f1SDimitry Andric                                  (sized_range<_Range> || sized_sentinel_for<sentinel_t<_Range>, iterator_t<_Range>>)
226fe6060f1SDimitry Andric                                    ? subrange_kind::sized : subrange_kind::unsized>;
227fe6060f1SDimitry Andric 
228fe6060f1SDimitry Andric   template<borrowed_range _Range>
229fe6060f1SDimitry Andric   subrange(_Range&&, make_unsigned_t<range_difference_t<_Range>>)
230fe6060f1SDimitry Andric     -> subrange<iterator_t<_Range>, sentinel_t<_Range>, subrange_kind::sized>;
231fe6060f1SDimitry Andric 
232fe6060f1SDimitry Andric   template<size_t _Index, class _Iter, class _Sent, subrange_kind _Kind>
2331fd87a68SDimitry Andric     requires ((_Index == 0 && copyable<_Iter>) || _Index == 1)
234fe6060f1SDimitry Andric   _LIBCPP_HIDE_FROM_ABI
235fe6060f1SDimitry Andric   constexpr auto get(const subrange<_Iter, _Sent, _Kind>& __subrange) {
236fe6060f1SDimitry Andric     if constexpr (_Index == 0)
237fe6060f1SDimitry Andric       return __subrange.begin();
238fe6060f1SDimitry Andric     else
239fe6060f1SDimitry Andric       return __subrange.end();
240fe6060f1SDimitry Andric   }
241fe6060f1SDimitry Andric 
242fe6060f1SDimitry Andric   template<size_t _Index, class _Iter, class _Sent, subrange_kind _Kind>
243fe6060f1SDimitry Andric     requires (_Index < 2)
244fe6060f1SDimitry Andric   _LIBCPP_HIDE_FROM_ABI
245fe6060f1SDimitry Andric   constexpr auto get(subrange<_Iter, _Sent, _Kind>&& __subrange) {
246fe6060f1SDimitry Andric     if constexpr (_Index == 0)
247fe6060f1SDimitry Andric       return __subrange.begin();
248fe6060f1SDimitry Andric     else
249fe6060f1SDimitry Andric       return __subrange.end();
250fe6060f1SDimitry Andric   }
251fe6060f1SDimitry Andric 
252fe6060f1SDimitry Andric   template<class _Ip, class _Sp, subrange_kind _Kp>
253fe6060f1SDimitry Andric   inline constexpr bool enable_borrowed_range<subrange<_Ip, _Sp, _Kp>> = true;
254fe6060f1SDimitry Andric 
255fe6060f1SDimitry Andric   template<range _Rp>
256fe6060f1SDimitry Andric   using borrowed_subrange_t = _If<borrowed_range<_Rp>, subrange<iterator_t<_Rp>>, dangling>;
257fe6060f1SDimitry Andric } // namespace ranges
258fe6060f1SDimitry Andric 
259349cc55cSDimitry Andric // [range.subrange.general]
260349cc55cSDimitry Andric 
261fe6060f1SDimitry Andric using ranges::get;
262fe6060f1SDimitry Andric 
263349cc55cSDimitry Andric // [ranges.syn]
264349cc55cSDimitry Andric 
265349cc55cSDimitry Andric template<class _Ip, class _Sp, ranges::subrange_kind _Kp>
266349cc55cSDimitry Andric struct tuple_size<ranges::subrange<_Ip, _Sp, _Kp>> : integral_constant<size_t, 2> {};
267349cc55cSDimitry Andric 
268349cc55cSDimitry Andric template<class _Ip, class _Sp, ranges::subrange_kind _Kp>
269349cc55cSDimitry Andric struct tuple_element<0, ranges::subrange<_Ip, _Sp, _Kp>> {
270349cc55cSDimitry Andric   using type = _Ip;
271349cc55cSDimitry Andric };
272349cc55cSDimitry Andric 
273349cc55cSDimitry Andric template<class _Ip, class _Sp, ranges::subrange_kind _Kp>
274349cc55cSDimitry Andric struct tuple_element<1, ranges::subrange<_Ip, _Sp, _Kp>> {
275349cc55cSDimitry Andric   using type = _Sp;
276349cc55cSDimitry Andric };
277349cc55cSDimitry Andric 
278349cc55cSDimitry Andric template<class _Ip, class _Sp, ranges::subrange_kind _Kp>
279349cc55cSDimitry Andric struct tuple_element<0, const ranges::subrange<_Ip, _Sp, _Kp>> {
280349cc55cSDimitry Andric   using type = _Ip;
281349cc55cSDimitry Andric };
282349cc55cSDimitry Andric 
283349cc55cSDimitry Andric template<class _Ip, class _Sp, ranges::subrange_kind _Kp>
284349cc55cSDimitry Andric struct tuple_element<1, const ranges::subrange<_Ip, _Sp, _Kp>> {
285349cc55cSDimitry Andric   using type = _Sp;
286349cc55cSDimitry Andric };
287fe6060f1SDimitry Andric 
288*81ad6265SDimitry Andric #endif // _LIBCPP_STD_VER > 17 && !defined(_LIBCPP_HAS_NO_INCOMPLETE_RANGES)
289fe6060f1SDimitry Andric 
290fe6060f1SDimitry Andric _LIBCPP_END_NAMESPACE_STD
291fe6060f1SDimitry Andric 
292fe6060f1SDimitry Andric #endif // _LIBCPP___RANGES_SUBRANGE_H
293