libstdc++
ranges_base.h
Go to the documentation of this file.
1// Core concepts and definitions for <ranges> -*- C++ -*-
2
3// Copyright (C) 2019-2026 Free Software Foundation, Inc.
4//
5// This file is part of the GNU ISO C++ Library. This library is free
6// software; you can redistribute it and/or modify it under the
7// terms of the GNU General Public License as published by the
8// Free Software Foundation; either version 3, or (at your option)
9// any later version.
10
11// This library is distributed in the hope that it will be useful,
12// but WITHOUT ANY WARRANTY; without even the implied warranty of
13// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14// GNU General Public License for more details.
15
16// Under Section 7 of GPL version 3, you are granted additional
17// permissions described in the GCC Runtime Library Exception, version
18// 3.1, as published by the Free Software Foundation.
19
20// You should have received a copy of the GNU General Public License and
21// a copy of the GCC Runtime Library Exception along with this program;
22// see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
23// <http://www.gnu.org/licenses/>.
24
25/** @file bits/ranges_base.h
26 * This is an internal header file, included by other library headers.
27 * Do not attempt to use it directly. @headername{ranges}
28 */
29
30#ifndef _GLIBCXX_RANGES_BASE_H
31#define _GLIBCXX_RANGES_BASE_H 1
32
33#ifdef _GLIBCXX_SYSHDR
34#pragma GCC system_header
35#endif
36
37#if __cplusplus > 201703L
38#include <initializer_list>
39#include <bits/stl_iterator.h>
40#include <ext/numeric_traits.h>
41#include <bits/max_size_type.h>
42#include <bits/version.h>
43
44#if __glibcxx_containers_ranges // C++ >= 23
45# include <bits/utility.h> // for tuple_element_t
46#endif
47
48#pragma GCC diagnostic push
49#pragma GCC diagnostic ignored "-Wpedantic" // __int128
50
51#if __glibcxx_algorithm_default_value_type // C++ >= 26
52# define _GLIBCXX26_RANGE_ALGO_DEF_VAL_T(_I, _P) = projected_value_t<_I, _P>
53#else
54# define _GLIBCXX26_RANGE_ALGO_DEF_VAL_T(_I, _P)
55#endif
56
57#ifdef __cpp_lib_concepts
58namespace std _GLIBCXX_VISIBILITY(default)
59{
60_GLIBCXX_BEGIN_NAMESPACE_VERSION
61namespace ranges
62{
63 template<typename>
64 inline constexpr bool disable_sized_range = false;
65
66 template<typename _Tp>
67 inline constexpr bool enable_borrowed_range = false;
68
69 namespace __detail
70 {
71 [[__gnu__::__always_inline__]]
72 constexpr __max_size_type
73 __to_unsigned_like(__max_size_type __t) noexcept
74 { return __t; }
75
76 [[__gnu__::__always_inline__]]
77 constexpr __max_size_type
78 __to_unsigned_like(__max_diff_type __t) noexcept
79 { return __max_size_type(__t); }
80
81 template<integral _Tp>
82 [[__gnu__::__always_inline__]]
83 constexpr auto
84 __to_unsigned_like(_Tp __t) noexcept
85 { return static_cast<make_unsigned_t<_Tp>>(__t); }
86
87 template<typename _Tp>
88 using __make_unsigned_like_t
89 = decltype(__detail::__to_unsigned_like(std::declval<_Tp>()));
90
91 // Part of the constraints of ranges::borrowed_range
92 template<typename _Tp>
93 concept __maybe_borrowed_range
94 = is_lvalue_reference_v<_Tp>
95 || enable_borrowed_range<remove_cvref_t<_Tp>>;
96
97 } // namespace __detail
98
99 // Namespace for helpers for the <ranges> customization points.
100 namespace __access
101 {
102 using std::ranges::__detail::__maybe_borrowed_range;
103 using std::__detail::__range_iter_t;
104
105 struct _Begin
106 {
107 private:
108 template<typename _Tp>
109 static consteval bool
110 _S_noexcept()
111 {
112 if constexpr (is_array_v<remove_reference_t<_Tp>>)
113 return true;
114 else if constexpr (__member_begin<_Tp>)
115 return noexcept(_GLIBCXX_AUTO_CAST(std::declval<_Tp&>().begin()));
116 else
117 return noexcept(_GLIBCXX_AUTO_CAST(begin(std::declval<_Tp&>())));
118 }
119
120 public:
121 template<__maybe_borrowed_range _Tp>
122 requires is_array_v<remove_reference_t<_Tp>> || __member_begin<_Tp>
123 || __adl_begin<_Tp>
124 [[nodiscard, __gnu__::__always_inline__]]
125 constexpr auto
126 operator()(_Tp&& __t) const noexcept(_S_noexcept<_Tp&>())
127 {
128 if constexpr (is_array_v<remove_reference_t<_Tp>>)
129 {
130 static_assert(is_lvalue_reference_v<_Tp>);
131 return __t + 0;
132 }
133 else if constexpr (__member_begin<_Tp>)
134 return __t.begin();
135 else
136 return begin(__t);
137 }
138 };
139
140 template<typename _Tp>
141 concept __member_end = requires(_Tp& __t)
142 {
143 { _GLIBCXX_AUTO_CAST(__t.end()) } -> sentinel_for<__range_iter_t<_Tp>>;
144 };
145
146 // Poison pill so that unqualified lookup doesn't find std::end.
147 void end() = delete;
148
149 template<typename _Tp>
150 concept __adl_end = __class_or_enum<remove_reference_t<_Tp>>
151 && requires(_Tp& __t)
152 {
153 { _GLIBCXX_AUTO_CAST(end(__t)) } -> sentinel_for<__range_iter_t<_Tp>>;
154 };
155
156 struct _End
157 {
158 private:
159 template<typename _Tp>
160 static consteval bool
161 _S_noexcept()
162 {
163 if constexpr (is_bounded_array_v<remove_reference_t<_Tp>>)
164 return true;
165 else if constexpr (__member_end<_Tp>)
166 return noexcept(_GLIBCXX_AUTO_CAST(std::declval<_Tp&>().end()));
167 else
168 return noexcept(_GLIBCXX_AUTO_CAST(end(std::declval<_Tp&>())));
169 }
170
171 public:
172 template<__maybe_borrowed_range _Tp>
173 requires is_bounded_array_v<remove_reference_t<_Tp>>
174 || __member_end<_Tp> || __adl_end<_Tp>
175 [[nodiscard, __gnu__::__always_inline__]]
176 constexpr auto
177 operator()(_Tp&& __t) const noexcept(_S_noexcept<_Tp&>())
178 {
179 if constexpr (is_bounded_array_v<remove_reference_t<_Tp>>)
180 {
181 static_assert(is_lvalue_reference_v<_Tp>);
182 return __t + extent_v<remove_reference_t<_Tp>>;
183 }
184 else if constexpr (__member_end<_Tp>)
185 return __t.end();
186 else
187 return end(__t);
188 }
189 };
190
191 template<typename _Tp>
192 concept __member_rbegin = requires(_Tp& __t)
193 {
194 { _GLIBCXX_AUTO_CAST(__t.rbegin()) } -> input_or_output_iterator;
195 };
196
197 void rbegin() = delete;
198
199 template<typename _Tp>
200 concept __adl_rbegin = __class_or_enum<remove_reference_t<_Tp>>
201 && requires(_Tp& __t)
202 {
203 { _GLIBCXX_AUTO_CAST(rbegin(__t)) } -> input_or_output_iterator;
204 };
205
206 template<typename _Tp>
207 concept __reversable = requires(_Tp& __t)
208 {
209 { _Begin{}(__t) } -> bidirectional_iterator;
210 { _End{}(__t) } -> same_as<decltype(_Begin{}(__t))>;
211 };
212
213 struct _RBegin
214 {
215 private:
216 template<typename _Tp>
217 static consteval bool
218 _S_noexcept()
219 {
220 if constexpr (__member_rbegin<_Tp>)
221 return noexcept(_GLIBCXX_AUTO_CAST(std::declval<_Tp&>().rbegin()));
222 else if constexpr (__adl_rbegin<_Tp>)
223 return noexcept(_GLIBCXX_AUTO_CAST(rbegin(std::declval<_Tp&>())));
224 else
225 {
226 if constexpr (noexcept(_End{}(std::declval<_Tp&>())))
227 {
228 using _It = decltype(_End{}(std::declval<_Tp&>()));
229 // std::reverse_iterator copy-initializes its member.
230 return is_nothrow_copy_constructible_v<_It>;
231 }
232 else
233 return false;
234 }
235 }
236
237 public:
238 template<__maybe_borrowed_range _Tp>
239 requires __member_rbegin<_Tp> || __adl_rbegin<_Tp> || __reversable<_Tp>
240 [[nodiscard, __gnu__::__always_inline__]]
241 constexpr auto
242 operator()(_Tp&& __t) const
243 noexcept(_S_noexcept<_Tp&>())
244 {
245 if constexpr (__member_rbegin<_Tp>)
246 return __t.rbegin();
247 else if constexpr (__adl_rbegin<_Tp>)
248 return rbegin(__t);
249 else
250 return std::make_reverse_iterator(_End{}(__t));
251 }
252 };
253
254 template<typename _Tp>
255 concept __member_rend = requires(_Tp& __t)
256 {
257 { _GLIBCXX_AUTO_CAST(__t.rend()) }
258 -> sentinel_for<decltype(_RBegin{}(std::forward<_Tp>(__t)))>;
259 };
260
261 void rend() = delete;
262
263 template<typename _Tp>
264 concept __adl_rend = __class_or_enum<remove_reference_t<_Tp>>
265 && requires(_Tp& __t)
266 {
267 { _GLIBCXX_AUTO_CAST(rend(__t)) }
268 -> sentinel_for<decltype(_RBegin{}(std::forward<_Tp>(__t)))>;
269 };
270
271 struct _REnd
272 {
273 private:
274 template<typename _Tp>
275 static consteval bool
276 _S_noexcept()
277 {
278 if constexpr (__member_rend<_Tp>)
279 return noexcept(_GLIBCXX_AUTO_CAST(std::declval<_Tp&>().rend()));
280 else if constexpr (__adl_rend<_Tp>)
281 return noexcept(_GLIBCXX_AUTO_CAST(rend(std::declval<_Tp&>())));
282 else
283 {
284 if constexpr (noexcept(_Begin{}(std::declval<_Tp&>())))
285 {
286 using _It = decltype(_Begin{}(std::declval<_Tp&>()));
287 // std::reverse_iterator copy-initializes its member.
288 return is_nothrow_copy_constructible_v<_It>;
289 }
290 else
291 return false;
292 }
293 }
294
295 public:
296 template<__maybe_borrowed_range _Tp>
297 requires __member_rend<_Tp> || __adl_rend<_Tp> || __reversable<_Tp>
298 [[nodiscard, __gnu__::__always_inline__]]
299 constexpr auto
300 operator()(_Tp&& __t) const
301 noexcept(_S_noexcept<_Tp&>())
302 {
303 if constexpr (__member_rend<_Tp>)
304 return __t.rend();
305 else if constexpr (__adl_rend<_Tp>)
306 return rend(__t);
307 else
308 return std::make_reverse_iterator(_Begin{}(__t));
309 }
310 };
311
312 template<typename _Tp>
313 concept __member_size = !disable_sized_range<remove_cvref_t<_Tp>>
314 && requires(_Tp& __t)
315 {
316 { _GLIBCXX_AUTO_CAST(__t.size()) } -> __detail::__is_integer_like;
317 };
318
319 void size() = delete;
320
321 template<typename _Tp>
322 concept __adl_size = __class_or_enum<remove_reference_t<_Tp>>
323 && !disable_sized_range<remove_cvref_t<_Tp>>
324 && requires(_Tp& __t)
325 {
326 { _GLIBCXX_AUTO_CAST(size(__t)) } -> __detail::__is_integer_like;
327 };
328
329 template<typename _Tp>
330 concept __sentinel_size = requires(_Tp& __t)
331 {
332 requires (!is_unbounded_array_v<remove_reference_t<_Tp>>);
333
334 { _Begin{}(__t) } -> forward_iterator;
335
336 { _End{}(__t) } -> sized_sentinel_for<decltype(_Begin{}(__t))>;
337
338 __detail::__to_unsigned_like(_End{}(__t) - _Begin{}(__t));
339 };
340
341 struct _Size
342 {
343 private:
344 template<typename _Tp>
345 static consteval bool
346 _S_noexcept()
347 {
348 if constexpr (is_bounded_array_v<remove_reference_t<_Tp>>)
349 return true;
350 else if constexpr (__member_size<_Tp>)
351 return noexcept(_GLIBCXX_AUTO_CAST(std::declval<_Tp&>().size()));
352 else if constexpr (__adl_size<_Tp>)
353 return noexcept(_GLIBCXX_AUTO_CAST(size(std::declval<_Tp&>())));
354 else if constexpr (__sentinel_size<_Tp>)
355 return noexcept(_End{}(std::declval<_Tp&>())
356 - _Begin{}(std::declval<_Tp&>()));
357 }
358
359 public:
360 template<typename _Tp>
361 requires is_bounded_array_v<remove_reference_t<_Tp>>
362 || __member_size<_Tp> || __adl_size<_Tp> || __sentinel_size<_Tp>
363 [[nodiscard, __gnu__::__always_inline__]]
364 constexpr auto
365 operator()(_Tp&& __t) const noexcept(_S_noexcept<_Tp&>())
366 {
367 if constexpr (is_bounded_array_v<remove_reference_t<_Tp>>)
368 return extent_v<remove_reference_t<_Tp>>;
369 else if constexpr (__member_size<_Tp>)
370 return __t.size();
371 else if constexpr (__adl_size<_Tp>)
372 return size(__t);
373 else if constexpr (__sentinel_size<_Tp>)
374 return __detail::__to_unsigned_like(_End{}(__t) - _Begin{}(__t));
375 }
376 };
377
378 struct _SSize
379 {
380 // _GLIBCXX_RESOLVE_LIB_DEFECTS
381 // 3403. Domain of ranges::ssize(E) doesn't match ranges::size(E)
382 template<typename _Tp>
383 requires requires (_Tp& __t) { _Size{}(__t); }
384 [[nodiscard, __gnu__::__always_inline__]]
385 constexpr auto
386 operator()(_Tp&& __t) const noexcept(noexcept(_Size{}(__t)))
387 {
388 auto __size = _Size{}(__t);
389 using __size_type = decltype(__size);
390 // Return the wider of ptrdiff_t and make-signed-like-t<__size_type>.
391 if constexpr (integral<__size_type>)
392 {
394 if constexpr (__int_traits<__size_type>::__digits
395 < __int_traits<ptrdiff_t>::__digits)
396 return static_cast<ptrdiff_t>(__size);
397 else
398 return static_cast<make_signed_t<__size_type>>(__size);
399 }
400 else // Must be one of __max_diff_type or __max_size_type.
401 return __detail::__max_diff_type(__size);
402 }
403 };
404
405 template<typename _Tp>
406 concept __member_empty = requires(_Tp& __t) { bool(__t.empty()); };
407
408 template<typename _Tp>
409 concept __size0_empty = requires(_Tp& __t) { _Size{}(__t) == 0; };
410
411 template<typename _Tp>
412 concept __eq_iter_empty = requires(_Tp& __t)
413 {
414 requires (!is_unbounded_array_v<remove_reference_t<_Tp>>);
415
416 { _Begin{}(__t) } -> forward_iterator;
417
418 bool(_Begin{}(__t) == _End{}(__t));
419 };
420
421 struct _Empty
422 {
423 private:
424 template<typename _Tp>
425 static consteval bool
426 _S_noexcept()
427 {
428 if constexpr (__member_empty<_Tp>)
429 return noexcept(bool(std::declval<_Tp&>().empty()));
430 else if constexpr (__size0_empty<_Tp>)
431 return noexcept(_Size{}(std::declval<_Tp&>()) == 0);
432 else
433 return noexcept(bool(_Begin{}(std::declval<_Tp&>())
434 == _End{}(std::declval<_Tp&>())));
435 }
436
437 public:
438 template<typename _Tp>
439 requires __member_empty<_Tp> || __size0_empty<_Tp>
440 || __eq_iter_empty<_Tp>
441 [[nodiscard, __gnu__::__always_inline__]]
442 constexpr bool
443 operator()(_Tp&& __t) const noexcept(_S_noexcept<_Tp&>())
444 {
445 if constexpr (__member_empty<_Tp>)
446 return bool(__t.empty());
447 else if constexpr (__size0_empty<_Tp>)
448 return _Size{}(__t) == 0;
449 else
450 return bool(_Begin{}(__t) == _End{}(__t));
451 }
452 };
453
454 template<typename _Tp>
455 concept __pointer_to_object = is_pointer_v<_Tp>
456 && is_object_v<remove_pointer_t<_Tp>>;
457
458 template<typename _Tp>
459 concept __member_data = requires(_Tp& __t)
460 {
461 { _GLIBCXX_AUTO_CAST(__t.data()) } -> __pointer_to_object;
462 };
463
464 template<typename _Tp>
465 concept __begin_data = contiguous_iterator<__range_iter_t<_Tp>>;
466
467 struct _Data
468 {
469 private:
470 template<typename _Tp>
471 static consteval bool
472 _S_noexcept()
473 {
474 if constexpr (__member_data<_Tp>)
475 return noexcept(_GLIBCXX_AUTO_CAST(std::declval<_Tp&>().data()));
476 else
477 return noexcept(_Begin{}(std::declval<_Tp&>()));
478 }
479
480 public:
481 template<__maybe_borrowed_range _Tp>
482 requires __member_data<_Tp> || __begin_data<_Tp>
483 [[nodiscard, __gnu__::__always_inline__]]
484 constexpr auto
485 operator()(_Tp&& __t) const noexcept(_S_noexcept<_Tp>())
486 {
487 if constexpr (__member_data<_Tp>)
488 return __t.data();
489 else
490 return std::to_address(_Begin{}(__t));
491 }
492 };
493
494 } // namespace __access
495
496 inline namespace _Cpo
497 {
498 inline constexpr ranges::__access::_Begin begin{};
499 inline constexpr ranges::__access::_End end{};
500 inline constexpr ranges::__access::_RBegin rbegin{};
501 inline constexpr ranges::__access::_REnd rend{};
502 inline constexpr ranges::__access::_Size size{};
503 inline constexpr ranges::__access::_SSize ssize{};
504 inline constexpr ranges::__access::_Empty empty{};
505 inline constexpr ranges::__access::_Data data{};
506 }
507
508 /// [range.range] The range concept.
509 template<typename _Tp>
510 concept range = requires(_Tp& __t)
511 {
512 ranges::begin(__t);
513 ranges::end(__t);
514 };
515
516 /// [range.range] The borrowed_range concept.
517 template<typename _Tp>
519 = range<_Tp> && __detail::__maybe_borrowed_range<_Tp>;
520
521 template<typename _Tp>
522 using iterator_t = std::__detail::__range_iter_t<_Tp>;
523
524 template<range _Range>
525 using sentinel_t = decltype(ranges::end(std::declval<_Range&>()));
526
527#if __glibcxx_ranges_as_const // >= C++23
528 // const_iterator_t and const_sentinel_t defined below.
529
530 template<range _Range>
531 using range_const_reference_t = iter_const_reference_t<iterator_t<_Range>>;
532#endif
533
534 template<range _Range>
535 using range_difference_t = iter_difference_t<iterator_t<_Range>>;
536
537 template<range _Range>
538 using range_value_t = iter_value_t<iterator_t<_Range>>;
539
540 template<range _Range>
541 using range_reference_t = iter_reference_t<iterator_t<_Range>>;
542
543 template<range _Range>
544 using range_rvalue_reference_t
546
547 // _GLIBCXX_RESOLVE_LIB_DEFECTS
548 // 3860. range_common_reference_t is missing
549 template<range _Range>
550 using range_common_reference_t
551 = iter_common_reference_t<iterator_t<_Range>>;
552
553 /// [range.sized] The sized_range concept.
554 template<typename _Tp>
555 concept sized_range = range<_Tp>
556 && requires(_Tp& __t) { ranges::size(__t); };
557
558 template<sized_range _Range>
559 using range_size_t = decltype(ranges::size(std::declval<_Range&>()));
560
561#if __cplusplus > 202302L
562 template<typename _Tp>
563 concept __static_sized_range = sized_range<_Tp> && requires (_Tp& __t)
564 { static_cast<char(*)[size_t(ranges::size(__t) >= 0)]>(nullptr); };
565#endif // C++26
566
567 template<typename _Derived>
568 requires is_class_v<_Derived> && same_as<_Derived, remove_cv_t<_Derived>>
569 class view_interface; // defined in <bits/ranges_util.h>
570
571 namespace __detail
572 {
573 template<typename _Tp, typename _Up>
575 void __is_derived_from_view_interface_fn(const _Tp&,
576 const view_interface<_Up>&); // not defined
577
578 // Returns true iff _Tp has exactly one public base class that's a
579 // specialization of view_interface.
580 template<typename _Tp>
581 concept __is_derived_from_view_interface
582 = requires (_Tp __t) { __is_derived_from_view_interface_fn(__t, __t); };
583 } // namespace __detail
585 /// [range.view] The ranges::view_base type.
586 struct view_base { };
587
588 /// [range.view] The ranges::enable_view boolean.
589 template<typename _Tp>
590 inline constexpr bool enable_view = derived_from<_Tp, view_base>
591 || __detail::__is_derived_from_view_interface<_Tp>;
592
593 /// [range.view] The ranges::view concept.
594 template<typename _Tp>
595 concept view
596 = range<_Tp> && movable<_Tp> && enable_view<_Tp>;
597
598 // [range.refinements]
599
600 /// A range for which ranges::begin returns an output iterator.
601 template<typename _Range, typename _Tp>
602 concept output_range
603 = range<_Range> && output_iterator<iterator_t<_Range>, _Tp>;
604
605 /// A range for which ranges::begin returns an input iterator.
606 template<typename _Tp>
607 concept input_range = range<_Tp> && input_iterator<iterator_t<_Tp>>;
608
609 /// A range for which ranges::begin returns a forward iterator.
610 template<typename _Tp>
611 concept forward_range
612 = input_range<_Tp> && forward_iterator<iterator_t<_Tp>>;
613
614 /// A range for which ranges::begin returns a bidirectional iterator.
615 template<typename _Tp>
616 concept bidirectional_range
617 = forward_range<_Tp> && bidirectional_iterator<iterator_t<_Tp>>;
618
619 /// A range for which ranges::begin returns a random access iterator.
620 template<typename _Tp>
621 concept random_access_range
622 = bidirectional_range<_Tp> && random_access_iterator<iterator_t<_Tp>>;
623
624 /// A range for which ranges::begin returns a contiguous iterator.
625 template<typename _Tp>
626 concept contiguous_range
627 = random_access_range<_Tp> && contiguous_iterator<iterator_t<_Tp>>
628 && requires(_Tp& __t)
629 {
630 { ranges::data(__t) } -> same_as<add_pointer_t<range_reference_t<_Tp>>>;
631 };
632
633 /// A range for which ranges::begin and ranges::end return the same type.
634 template<typename _Tp>
635 concept common_range
636 = range<_Tp> && same_as<iterator_t<_Tp>, sentinel_t<_Tp>>;
637
638#if __glibcxx_ranges_as_const // >= C++23
639 template<typename _Tp>
640 concept constant_range
641 = input_range<_Tp> && std::__detail::__constant_iterator<iterator_t<_Tp>>;
642#endif
643
644 namespace __access
645 {
646#if __glibcxx_ranges_as_const // >= C++23
647 template<input_range _Range>
648 [[__gnu__::__always_inline__]]
649 constexpr auto&
650 __possibly_const_range(_Range& __r) noexcept
651 {
652 // _GLIBCXX_RESOLVE_LIB_DEFECTS
653 // 4027. possibly-const-range should prefer returning const R&
654 if constexpr (input_range<const _Range>)
655 return const_cast<const _Range&>(__r);
656 else
657 return __r;
658 }
659#else
660 // If _To is an lvalue-reference, return const _Tp&, otherwise const _Tp&&.
661 template<typename _To, typename _Tp>
662 [[__gnu__::__always_inline__]]
663 constexpr decltype(auto)
664 __as_const(_Tp& __t) noexcept
665 {
666 static_assert(std::is_same_v<_To&, _Tp&>);
667
668 if constexpr (is_lvalue_reference_v<_To>)
669 return const_cast<const _Tp&>(__t);
670 else
671 return static_cast<const _Tp&&>(__t);
672 }
673#endif
674
675 struct _CBegin
676 {
677#if __glibcxx_ranges_as_const // >= C++23
678 template<__maybe_borrowed_range _Tp>
679 [[nodiscard]]
680 constexpr auto
681 operator()(_Tp&& __t) const
682 noexcept(noexcept(std::make_const_iterator
683 (ranges::begin(__access::__possibly_const_range(__t)))))
684 requires requires { std::make_const_iterator
685 (ranges::begin(__access::__possibly_const_range(__t))); }
686 {
687 auto& __r = __access::__possibly_const_range(__t);
688 return const_iterator<decltype(ranges::begin(__r))>(ranges::begin(__r));
689 }
690#else
691 template<typename _Tp>
692 [[nodiscard]]
693 constexpr auto
694 operator()(_Tp&& __e) const
695 noexcept(noexcept(_Begin{}(__access::__as_const<_Tp>(__e))))
696 requires requires { _Begin{}(__access::__as_const<_Tp>(__e)); }
697 {
698 return _Begin{}(__access::__as_const<_Tp>(__e));
699 }
700#endif
701 };
702
703 struct _CEnd final
704 {
705#if __glibcxx_ranges_as_const // >= C++23
706 template<__maybe_borrowed_range _Tp>
707 [[nodiscard]]
708 constexpr auto
709 operator()(_Tp&& __t) const
710 noexcept(noexcept(std::make_const_sentinel
711 (ranges::end(__access::__possibly_const_range(__t)))))
712 requires requires { std::make_const_sentinel
713 (ranges::end(__access::__possibly_const_range(__t))); }
714 {
715 auto& __r = __access::__possibly_const_range(__t);
716 return const_sentinel<decltype(ranges::end(__r))>(ranges::end(__r));
717 }
718#else
719 template<typename _Tp>
720 [[nodiscard]]
721 constexpr auto
722 operator()(_Tp&& __e) const
723 noexcept(noexcept(_End{}(__access::__as_const<_Tp>(__e))))
724 requires requires { _End{}(__access::__as_const<_Tp>(__e)); }
725 {
726 return _End{}(__access::__as_const<_Tp>(__e));
727 }
728#endif
729 };
730
731 struct _CRBegin
732 {
733#if __glibcxx_ranges_as_const // >= C++23
734 template<__maybe_borrowed_range _Tp>
735 [[nodiscard]]
736 constexpr auto
737 operator()(_Tp&& __t) const
738 noexcept(noexcept(std::make_const_iterator
739 (ranges::rbegin(__access::__possibly_const_range(__t)))))
740 requires requires { std::make_const_iterator
741 (ranges::rbegin(__access::__possibly_const_range(__t))); }
742 {
743 auto& __r = __access::__possibly_const_range(__t);
744 return const_iterator<decltype(ranges::rbegin(__r))>(ranges::rbegin(__r));
745 }
746#else
747 template<typename _Tp>
748 [[nodiscard]]
749 constexpr auto
750 operator()(_Tp&& __e) const
751 noexcept(noexcept(_RBegin{}(__access::__as_const<_Tp>(__e))))
752 requires requires { _RBegin{}(__access::__as_const<_Tp>(__e)); }
753 {
754 return _RBegin{}(__access::__as_const<_Tp>(__e));
755 }
756#endif
757 };
758
759 struct _CREnd
760 {
761#if __glibcxx_ranges_as_const // >= C++23
762 template<__maybe_borrowed_range _Tp>
763 [[nodiscard]]
764 constexpr auto
765 operator()(_Tp&& __t) const
766 noexcept(noexcept(std::make_const_sentinel
767 (ranges::rend(__access::__possibly_const_range(__t)))))
768 requires requires { std::make_const_sentinel
769 (ranges::rend(__access::__possibly_const_range(__t))); }
770 {
771 auto& __r = __access::__possibly_const_range(__t);
772 return const_sentinel<decltype(ranges::rend(__r))>(ranges::rend(__r));
773 }
774#else
775 template<typename _Tp>
776 [[nodiscard]]
777 constexpr auto
778 operator()(_Tp&& __e) const
779 noexcept(noexcept(_REnd{}(__access::__as_const<_Tp>(__e))))
780 requires requires { _REnd{}(__access::__as_const<_Tp>(__e)); }
781 {
782 return _REnd{}(__access::__as_const<_Tp>(__e));
783 }
784#endif
785 };
786
787 struct _CData
788 {
789#if __glibcxx_ranges_as_const // >= C++23
790 template<__maybe_borrowed_range _Tp>
791 [[nodiscard]]
792 constexpr const auto*
793 operator()(_Tp&& __t) const
794 noexcept(noexcept(ranges::data(__access::__possibly_const_range(__t))))
795 requires requires { ranges::data(__access::__possibly_const_range(__t)); }
796 { return ranges::data(__access::__possibly_const_range(__t)); }
797#else
798 template<typename _Tp>
799 [[nodiscard]]
800 constexpr auto
801 operator()(_Tp&& __e) const
802 noexcept(noexcept(_Data{}(__access::__as_const<_Tp>(__e))))
803 requires requires { _Data{}(__access::__as_const<_Tp>(__e)); }
804 {
805 return _Data{}(__access::__as_const<_Tp>(__e));
806 }
807#endif
808 };
809 } // namespace __access
810
811 inline namespace _Cpo
812 {
813 inline constexpr ranges::__access::_CBegin cbegin{};
814 inline constexpr ranges::__access::_CEnd cend{};
815 inline constexpr ranges::__access::_CRBegin crbegin{};
816 inline constexpr ranges::__access::_CREnd crend{};
817 inline constexpr ranges::__access::_CData cdata{};
818 }
819
820#if __glibcxx_ranges_as_const // >= C++23
821 // _GLIBCXX_RESOLVE_LIB_DEFECTS
822 // 3946. The definition of const_iterator_t should be reworked
823 template<range _Range>
824 using const_iterator_t = decltype(ranges::cbegin(std::declval<_Range&>()));
825
826 template<range _Range>
827 using const_sentinel_t = decltype(ranges::cend(std::declval<_Range&>()));
828#endif
829
830 namespace __detail
831 {
832 template<typename _Tp>
833 inline constexpr bool __is_initializer_list = false;
834
835 template<typename _Tp>
836 inline constexpr bool __is_initializer_list<initializer_list<_Tp>> = true;
837 } // namespace __detail
838
839 /// A range which can be safely converted to a view.
840 template<typename _Tp>
841 concept viewable_range = range<_Tp>
844 && (is_lvalue_reference_v<_Tp>
845 || (movable<remove_reference_t<_Tp>>
846 && !__detail::__is_initializer_list<remove_cvref_t<_Tp>>))));
847
848 // [range.iter.ops] range iterator operations
849
850 struct __advance_fn final
851 {
852 template<input_or_output_iterator _It>
853 constexpr void
854 operator()(_It& __it, iter_difference_t<_It> __n) const
855 {
856 if constexpr (random_access_iterator<_It>)
857 __it += __n;
858 else if constexpr (bidirectional_iterator<_It>)
859 {
860 if (__n > 0)
861 {
862 do
863 {
864 ++__it;
865 }
866 while (--__n);
867 }
868 else if (__n < 0)
869 {
870 do
871 {
872 --__it;
873 }
874 while (++__n);
875 }
876 }
877 else
878 {
879 // cannot decrement a non-bidirectional iterator
880 __glibcxx_assert(__n >= 0);
881 while (__n-- > 0)
882 ++__it;
883 }
884 }
885
886 template<input_or_output_iterator _It, sentinel_for<_It> _Sent>
887 constexpr void
888 operator()(_It& __it, _Sent __bound) const
889 {
890 if constexpr (assignable_from<_It&, _Sent>)
891 __it = std::move(__bound);
892 else if constexpr (sized_sentinel_for<_Sent, _It>)
893 (*this)(__it, __bound - __it);
894 else
895 {
896 while (__it != __bound)
897 ++__it;
898 }
899 }
900
901 template<input_or_output_iterator _It, sentinel_for<_It> _Sent>
902 constexpr iter_difference_t<_It>
903 operator()(_It& __it, iter_difference_t<_It> __n, _Sent __bound) const
904 {
905 if constexpr (sized_sentinel_for<_Sent, _It>)
906 {
907 const iter_difference_t<_It> __diff = __bound - __it;
908
909 if (__diff == 0)
910 {
911 // inline any possible side effects of advance(it, bound)
912 if constexpr (assignable_from<_It&, _Sent>)
913 __it = std::move(__bound);
914 else if constexpr (random_access_iterator<_It>)
915 __it += iter_difference_t<_It>(0);
916 return __n;
917 }
918 else if (__diff > 0 ? __n >= __diff : __n <= __diff)
919 {
920 (*this)(__it, __bound);
921 return __n - __diff;
922 }
923 else if (__n != 0) [[likely]]
924 {
925 // n and bound must not lead in opposite directions:
926 __glibcxx_assert((__n < 0) == (__diff < 0));
927
928 (*this)(__it, __n);
929 return 0;
930 }
931 else
932 {
933 // inline any possible side effects of advance(it, n)
934 if constexpr (random_access_iterator<_It>)
935 __it += iter_difference_t<_It>(0);
936 return 0;
937 }
938 }
939 else if (__n == 0 || __it == __bound)
940 return __n;
941 else if (__n > 0)
942 {
943 iter_difference_t<_It> __m = 0;
944 do
945 {
946 ++__it;
947 ++__m;
948 }
949 while (__m != __n && __it != __bound);
950 return __n - __m;
951 }
952 else if constexpr (bidirectional_iterator<_It> && same_as<_It, _Sent>)
953 {
954 iter_difference_t<_It> __m = 0;
955 do
956 {
957 --__it;
958 --__m;
959 }
960 while (__m != __n && __it != __bound);
961 return __n - __m;
962 }
963 else
964 {
965 // cannot decrement a non-bidirectional iterator
966 __glibcxx_assert(__n >= 0);
967 return __n;
968 }
969 }
970
971 void operator&() const = delete;
972 };
973
974 inline constexpr __advance_fn advance{};
975
976 struct __distance_fn final
977 {
978 template<typename _It, sentinel_for<_It> _Sent>
979 requires (!sized_sentinel_for<_Sent, _It>)
980 constexpr iter_difference_t<_It>
981 operator()[[nodiscard]](_It __first, _Sent __last) const
982 {
983 iter_difference_t<_It> __n = 0;
984 while (__first != __last)
985 {
986 ++__first;
987 ++__n;
988 }
989 return __n;
990 }
991
992 // _GLIBCXX_RESOLVE_LIB_DEFECTS
993 // 3392. cannot be used on a move-only iterator with a sized sentinel
994 // 3664. LWG 3392 broke std::ranges::distance(a, a+3)
995 // 4242. ranges::distance does not work with volatile iterators
996 template<typename _It, sized_sentinel_for<decay_t<_It>> _Sent>
997 [[nodiscard, __gnu__::__always_inline__]]
998 constexpr iter_difference_t<decay_t<_It>>
999 operator()(_It&& __first, _Sent __last) const
1000 {
1001 if constexpr (!is_array_v<remove_reference_t<_It>>)
1002 return __last - __first;
1003 else
1004 return __last - static_cast<decay_t<_It>>(__first);
1005 }
1006
1007 template<range _Range>
1008 [[nodiscard, __gnu__::__always_inline__]]
1009 constexpr range_difference_t<_Range>
1010 operator()(_Range&& __r) const
1011 {
1012 if constexpr (sized_range<_Range>)
1013 return static_cast<range_difference_t<_Range>>(ranges::size(__r));
1014 else
1015 return (*this)(ranges::begin(__r), ranges::end(__r));
1016 }
1017
1018 void operator&() const = delete;
1019 };
1020
1021 inline constexpr __distance_fn distance{};
1022
1023 struct __next_fn final
1024 {
1025 template<input_or_output_iterator _It>
1026 [[nodiscard, __gnu__::__always_inline__]]
1027 constexpr _It
1028 operator()(_It __x) const
1029 {
1030 ++__x;
1031 return __x;
1032 }
1033
1034 template<input_or_output_iterator _It>
1035 [[nodiscard, __gnu__::__always_inline__]]
1036 constexpr _It
1037 operator()(_It __x, iter_difference_t<_It> __n) const
1038 {
1039 ranges::advance(__x, __n);
1040 return __x;
1041 }
1042
1043 template<input_or_output_iterator _It, sentinel_for<_It> _Sent>
1044 [[nodiscard, __gnu__::__always_inline__]]
1045 constexpr _It
1046 operator()(_It __x, _Sent __bound) const
1047 {
1048 ranges::advance(__x, __bound);
1049 return __x;
1050 }
1051
1052 template<input_or_output_iterator _It, sentinel_for<_It> _Sent>
1053 [[nodiscard, __gnu__::__always_inline__]]
1054 constexpr _It
1055 operator()(_It __x, iter_difference_t<_It> __n, _Sent __bound) const
1056 {
1057 ranges::advance(__x, __n, __bound);
1058 return __x;
1059 }
1060
1061 void operator&() const = delete;
1062 };
1063
1064 inline constexpr __next_fn next{};
1065
1066 struct __prev_fn final
1067 {
1068 template<bidirectional_iterator _It>
1069 [[nodiscard, __gnu__::__always_inline__]]
1070 constexpr _It
1071 operator()(_It __x) const
1072 {
1073 --__x;
1074 return __x;
1075 }
1076
1077 template<bidirectional_iterator _It>
1078 [[nodiscard, __gnu__::__always_inline__]]
1079 constexpr _It
1080 operator()(_It __x, iter_difference_t<_It> __n) const
1081 {
1082 ranges::advance(__x, -__n);
1083 return __x;
1084 }
1085
1086 template<bidirectional_iterator _It>
1087 [[nodiscard, __gnu__::__always_inline__]]
1088 constexpr _It
1089 operator()(_It __x, iter_difference_t<_It> __n, _It __bound) const
1090 {
1091 ranges::advance(__x, -__n, __bound);
1092 return __x;
1093 }
1094
1095 void operator&() const = delete;
1096 };
1097
1098 inline constexpr __prev_fn prev{};
1100 /// Type returned by algorithms instead of a dangling iterator or subrange.
1101 struct dangling
1102 {
1103 constexpr dangling() noexcept = default;
1104 template<typename... _Args>
1105 constexpr dangling(_Args&&...) noexcept { }
1106 };
1107
1108 template<range _Range>
1109 using borrowed_iterator_t = __conditional_t<borrowed_range<_Range>,
1110 iterator_t<_Range>,
1111 dangling>;
1112} // namespace ranges
1113
1114#if __glibcxx_ranges_to_container // C++ >= 23
1115 struct from_range_t { explicit from_range_t() = default; };
1116 inline constexpr from_range_t from_range{};
1117#endif
1118
1119#if __glibcxx_containers_ranges // C++ >= 23
1120/// @cond undocumented
1121 template<typename _T1, typename _T2>
1122 struct pair;
1123
1124namespace __detail
1125{
1126 template<typename _Rg, typename _Tp>
1127 concept __container_compatible_range
1128 = ranges::input_range<_Rg>
1129 && convertible_to<ranges::range_reference_t<_Rg>, _Tp>;
1130
1131 // _GLIBCXX_RESOLVE_LIB_DEFECTS
1132 // 4223. Deduction guides for maps are mishandling tuples and references
1133 template<ranges::input_range _Range>
1134 using __range_key_type
1135 = remove_cvref_t<tuple_element_t<0, ranges::range_value_t<_Range>>>;
1136
1137 template<ranges::input_range _Range>
1138 using __range_mapped_type
1139 = remove_cvref_t<tuple_element_t<1, ranges::range_value_t<_Range>>>;
1140
1141 // The allocator's value_type for map-like containers.
1142 template<ranges::input_range _Range>
1143 using __range_to_alloc_type
1144 = pair<const __range_key_type<_Range>, __range_mapped_type<_Range>>;
1145}
1146/// @endcond
1147#endif
1148
1149_GLIBCXX_END_NAMESPACE_VERSION
1150} // namespace std
1151#endif // library concepts
1152#pragma GCC diagnostic pop
1153#endif // C++20
1154#endif // _GLIBCXX_RANGES_BASE_H
constexpr bool enable_view
[range.view] The ranges::enable_view boolean.
constexpr _Tp * to_address(_Tp *__ptr) noexcept
Obtain address referenced by a pointer to an object.
Definition ptr_traits.h:234
typename make_signed< _Tp >::type make_signed_t
Alias template for make_signed.
Definition type_traits:2269
typename remove_reference< _Tp >::type remove_reference_t
Alias template for remove_reference.
Definition type_traits:1913
typename make_unsigned< _Tp >::type make_unsigned_t
Alias template for make_unsigned.
Definition type_traits:2273
typename decay< _Tp >::type decay_t
Alias template for decay.
Definition type_traits:2963
pair(_T1, _T2) -> pair< _T1, _T2 >
Two pairs are equal iff their members are equal.
auto declval() noexcept -> decltype(__declval< _Tp >(0))
Definition type_traits:2741
constexpr std::remove_reference< _Tp >::type && move(_Tp &&__t) noexcept
Convert a value to an rvalue.
Definition move.h:138
constexpr _Tp && forward(typename std::remove_reference< _Tp >::type &__t) noexcept
Forward an lvalue.
Definition move.h:72
constexpr reverse_iterator< _Iterator > make_reverse_iterator(_Iterator __i)
Generator function for reverse_iterator.
ISO C++ entities toplevel namespace is std.
constexpr auto cend(const _Container &__cont) noexcept(noexcept(std::end(__cont))) -> decltype(std::end(__cont))
Return an iterator pointing to one past the last element of the const container.
constexpr auto empty(const _Container &__cont) noexcept(noexcept(__cont.empty())) -> decltype(__cont.empty())
Return whether a container is empty.
ranges::__imove::_IterMove::__type< _Tp & > iter_rvalue_reference_t
The result type of ranges::iter_move(std::declval<_Tp&>()).
constexpr auto data(_Container &__cont) noexcept(noexcept(__cont.data())) -> decltype(__cont.data())
Return the data pointer of a container.
constexpr auto cbegin(const _Container &__cont) noexcept(noexcept(std::begin(__cont))) -> decltype(std::begin(__cont))
Return an iterator pointing to the first element of the const container.
constexpr bitset< _Nb > operator&(const bitset< _Nb > &__x, const bitset< _Nb > &__y) noexcept
Global bitwise operations on bitsets.
Definition bitset:1618
Implementation details not part of the namespace std interface.
__numeric_traits_integer< _Tp > __int_traits
Convenience alias for __numeric_traits<integer-type>.
[range.view] The ranges::view_base type.
Type returned by algorithms instead of a dangling iterator or subrange.
Struct holding two objects (or references) of arbitrary type.
Definition stl_pair.h:307
[concept.same], concept same_as
Definition concepts:65
[concept.derived], concept derived_from
Definition concepts:76
[concept.assignable], concept assignable_from
Definition concepts:149
[concept.constructible], concept constructible_from
Definition concepts:162
[range.range] The range concept.
[range.range] The borrowed_range concept.
[range.sized] The sized_range concept.
[range.view] The ranges::view concept.
A range for which ranges::begin returns an output iterator.
A range for which ranges::begin returns an input iterator.
A range for which ranges::begin returns a forward iterator.
A range for which ranges::begin returns a bidirectional iterator.
A range for which ranges::begin returns a random access iterator.
A range for which ranges::begin returns a contiguous iterator.
A range for which ranges::begin and ranges::end return the same type.
A range which can be safely converted to a view.