libstdc++
mdspan
Go to the documentation of this file.
1// <mdspan> -*- C++ -*-
2
3// Copyright The GNU Toolchain Authors.
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 mdspan
26 * This is a Standard C++ Library header.
27 */
28
29#ifndef _GLIBCXX_MDSPAN
30#define _GLIBCXX_MDSPAN 1
31
32#ifdef _GLIBCXX_SYSHDR
33#pragma GCC system_header
34#endif
35
36#include <span>
37#include <array>
38#include <type_traits>
39#include <utility>
40
41#define __glibcxx_want_mdspan
42#define __glibcxx_want_aligned_accessor
43#define __glibcxx_want_hardened_mdspan
44#define __glibcxx_want_submdspan
45#include <bits/version.h>
46
47#if __glibcxx_aligned_accessor
48#include <bits/align.h>
49#endif
50
51#if __glibcxx_submdspan
52#include <tuple>
53#endif
54
55#if __cplusplus > 202302L
57#endif
58
59#ifdef __glibcxx_mdspan
60
61namespace std _GLIBCXX_VISIBILITY(default)
62{
63_GLIBCXX_BEGIN_NAMESPACE_VERSION
64 namespace __mdspan
65 {
66 consteval bool
67 __all_static(std::span<const size_t> __extents)
68 {
69 for(auto __ext : __extents)
70 if (__ext == dynamic_extent)
71 return false;
72 return true;
73 }
74
75 consteval bool
76 __all_dynamic(std::span<const size_t> __extents)
77 {
78 for(auto __ext : __extents)
79 if (__ext != dynamic_extent)
80 return false;
81 return true;
82 }
83
84 template<typename _IndexType, typename _OIndexTypeRef>
85 constexpr _IndexType
86 __index_type_cast(_OIndexTypeRef&& __other)
87 {
88 // _GLIBCXX_RESOLVE_LIB_DEFECTS
89 // 4020. extents::index-cast weirdness
90 using _OIndexType = std::remove_cvref_t<_OIndexTypeRef>;
91 if constexpr (std::is_integral_v<_OIndexType>)
92 {
93 constexpr _IndexType __index_type_max
94 = __gnu_cxx::__int_traits<_IndexType>::__max;
95 constexpr _OIndexType __oindex_type_max
96 = __gnu_cxx::__int_traits<_OIndexType>::__max;
97
98 if constexpr (__index_type_max < __oindex_type_max)
99 __glibcxx_assert(cmp_less_equal(__other, __index_type_max));
100
101 if constexpr (std::is_signed_v<_OIndexType>)
102 __glibcxx_assert(__other >= 0);
103 return static_cast<_IndexType>(__other);
104 }
105 else
106 {
107 // _GLIBCXX_RESOLVE_LIB_DEFECTS
108 // 4314. Missing move in mdspan layout mapping::operator()
109 auto __ret = static_cast<_IndexType>(std::forward<_OIndexTypeRef>(__other));
110 if constexpr (std::is_signed_v<_IndexType>)
111 __glibcxx_assert(__ret >= 0);
112 return __ret;
113 }
114 }
115
116 template<array _Extents>
117 class _StaticExtents
118 {
119 public:
120 static constexpr size_t _S_rank = _Extents.size();
121
122 // For __r in [0, _S_rank], _S_dynamic_index(__r) is the number
123 // of dynamic extents up to (and not including) __r.
124 //
125 // If __r is the index of a dynamic extent, then
126 // _S_dynamic_index[__r] is the index of that extent in
127 // _M_dyn_exts.
128 static constexpr size_t
129 _S_dynamic_index(size_t __r) noexcept
130 { return _S_dynamic_index_data[__r]; }
131
132 static constexpr auto _S_dynamic_index_data = [] consteval
133 {
134 array<size_t, _S_rank+1> __ret;
135 size_t __dyn = 0;
136 for (size_t __i = 0; __i < _S_rank; ++__i)
137 {
138 __ret[__i] = __dyn;
139 __dyn += (_Extents[__i] == dynamic_extent);
140 }
141 __ret[_S_rank] = __dyn;
142 return __ret;
143 }();
144
145 static constexpr size_t _S_rank_dynamic = _S_dynamic_index(_S_rank);
146
147 // For __r in [0, _S_rank_dynamic), _S_dynamic_index_inv(__r) is the
148 // index of the __r-th dynamic extent in _Extents.
149 static constexpr size_t
150 _S_dynamic_index_inv(size_t __r) noexcept
151 { return _S_dynamic_index_inv_data[__r]; }
152
153 static constexpr auto _S_dynamic_index_inv_data = [] consteval
154 {
155 array<size_t, _S_rank_dynamic> __ret;
156 for (size_t __i = 0, __r = 0; __i < _S_rank; ++__i)
157 if (_Extents[__i] == dynamic_extent)
158 __ret[__r++] = __i;
159 return __ret;
160 }();
161
162 static constexpr size_t
163 _S_static_extent(size_t __r) noexcept
164 { return _Extents[__r]; }
165 };
166
167 template<array _Extents>
168 requires (__all_dynamic<_Extents>())
169 class _StaticExtents<_Extents>
170 {
171 public:
172 static constexpr size_t _S_rank = _Extents.size();
173
174 static constexpr size_t
175 _S_dynamic_index(size_t __r) noexcept
176 { return __r; }
177
178 static constexpr size_t _S_rank_dynamic = _S_rank;
179
180 static constexpr size_t
181 _S_dynamic_index_inv(size_t __k) noexcept
182 { return __k; }
183
184 static constexpr size_t
185 _S_static_extent(size_t) noexcept
186 { return dynamic_extent; }
187 };
188
189 template<typename _IndexType, array _Extents>
190 class _ExtentsStorage : public _StaticExtents<_Extents>
191 {
192 private:
193 using _Base = _StaticExtents<_Extents>;
194
195 public:
196 using _Base::_S_rank;
197 using _Base::_S_rank_dynamic;
198 using _Base::_S_dynamic_index;
199 using _Base::_S_dynamic_index_inv;
200 using _Base::_S_static_extent;
201
202 static constexpr bool
203 _S_is_dynamic(size_t __r) noexcept
204 {
205 if constexpr (__all_static(_Extents))
206 return false;
207 else if constexpr (__all_dynamic(_Extents))
208 return true;
209 else
210 return _Extents[__r] == dynamic_extent;
211 }
212
213 template<typename _OIndexType>
214 static constexpr _IndexType
215 _S_int_cast(const _OIndexType& __other) noexcept
216 { return _IndexType(__other); }
217
218 constexpr _IndexType
219 _M_extent(size_t __r) const noexcept
220 {
221 if (_S_is_dynamic(__r))
222 return _M_dyn_exts[_S_dynamic_index(__r)];
223 else
224 return _S_static_extent(__r);
225 }
226
227 template<size_t _OtherRank, typename _GetOtherExtent>
228 static constexpr bool
229 _S_is_compatible_extents(_GetOtherExtent __get_extent) noexcept
230 {
231 if constexpr (_OtherRank == _S_rank)
232 for (size_t __i = 0; __i < _S_rank; ++__i)
233 if (!_S_is_dynamic(__i)
234 && !cmp_equal(_Extents[__i], _S_int_cast(__get_extent(__i))))
235 return false;
236 return true;
237 }
238
239 template<size_t _OtherRank, typename _GetOtherExtent>
240 constexpr void
241 _M_init_dynamic_extents(_GetOtherExtent __get_extent) noexcept
242 {
243 __glibcxx_assert(_S_is_compatible_extents<_OtherRank>(__get_extent));
244 for (size_t __i = 0; __i < _S_rank_dynamic; ++__i)
245 {
246 size_t __di = __i;
247 if constexpr (_OtherRank != _S_rank_dynamic)
248 __di = _S_dynamic_index_inv(__i);
249 _M_dyn_exts[__i] = _S_int_cast(__get_extent(__di));
250 }
251 }
252
253 constexpr
254 _ExtentsStorage() noexcept = default;
255
256 template<typename _OIndexType, array _OExtents>
257 constexpr
258 _ExtentsStorage(const _ExtentsStorage<_OIndexType, _OExtents>&
259 __other) noexcept
260 {
261 _M_init_dynamic_extents<_S_rank>([&__other](size_t __i)
262 { return __other._M_extent(__i); });
263 }
264
265 template<typename _OIndexType, size_t _Nm>
266 constexpr
267 _ExtentsStorage(span<const _OIndexType, _Nm> __exts) noexcept
268 {
269 _M_init_dynamic_extents<_Nm>(
270 [&__exts](size_t __i) -> const _OIndexType&
271 { return __exts[__i]; });
272 }
273
274 static constexpr const array<size_t, _S_rank>&
275 _S_static_extents() noexcept
276 { return _Extents; }
277
278 constexpr span<const _IndexType>
279 _M_dynamic_extents(size_t __begin, size_t __end) const noexcept
280 requires (_Extents.size() > 0)
281 {
282 return {_M_dyn_exts + _S_dynamic_index(__begin),
283 _S_dynamic_index(__end) - _S_dynamic_index(__begin)};
284 }
285
286 private:
287 using _Storage = __array_traits<_IndexType, _S_rank_dynamic>::_Type;
288 [[no_unique_address]] _Storage _M_dyn_exts{};
289 };
290
291 template<typename _OIndexType, typename _SIndexType>
292 concept __valid_index_type =
293 is_convertible_v<_OIndexType, _SIndexType> &&
294 is_nothrow_constructible_v<_SIndexType, _OIndexType>;
295
296 template<size_t _Extent, typename _IndexType>
297 concept
298 __valid_static_extent = _Extent == dynamic_extent
299 || _Extent <= __gnu_cxx::__int_traits<_IndexType>::__max;
300
301 template<typename _Extents>
302 constexpr const array<size_t, _Extents::rank()>&
303 __static_extents() noexcept
304 { return _Extents::_Storage::_S_static_extents(); }
305
306 template<typename _Extents>
307 constexpr span<const size_t>
308 __static_extents(size_t __begin, size_t __end) noexcept
309 {
310 const auto& __sta_exts = __static_extents<_Extents>();
311 return span<const size_t>(__sta_exts.data() + __begin, __end - __begin);
312 }
313
314 // Pre-compute: \prod_{i = 0}^r _Extents[i], for r = 0,..., n (exclusive)
315 template<array _Extents>
316 constexpr auto __fwd_partial_prods = [] consteval
317 {
318 constexpr size_t __rank = _Extents.size();
319 std::array<size_t, __rank> __ret;
320 size_t __prod = 1;
321 for (size_t __r = 0; __r < __rank; ++__r)
322 {
323 __ret[__r] = __prod;
324 if (size_t __ext = _Extents[__r]; __ext != dynamic_extent)
325 __prod *= __ext;
326 }
327 return __ret;
328 }();
329
330 // Pre-compute: \prod_{i = r+1}^{n-1} _Extents[i]
331 template<array _Extents>
332 constexpr auto __rev_partial_prods = [] consteval
333 {
334 constexpr size_t __rank = _Extents.size();
335 std::array<size_t, __rank> __ret;
336 size_t __prod = 1;
337 for (size_t __r = __rank; __r > 0; --__r)
338 {
339 __ret[__r - 1] = __prod;
340 if (size_t __ext = _Extents[__r - 1]; __ext != dynamic_extent)
341 __prod *= __ext;
342 }
343 return __ret;
344 }();
345
346 template<typename _Extents>
347 constexpr span<const typename _Extents::index_type>
348 __dynamic_extents(const _Extents& __exts, size_t __begin = 0,
349 size_t __end = _Extents::rank()) noexcept
350 { return __exts._M_exts._M_dynamic_extents(__begin, __end); }
351 }
352
353#if __glibcxx_submdspan
354 struct full_extent_t
355 {
356 explicit full_extent_t() = default;
357 };
358
359 inline constexpr full_extent_t full_extent{};
360
361 template<typename _OffsetType, typename _ExtentType, typename _StrideType>
362 struct extent_slice
363 {
364 static_assert(__is_signed_or_unsigned_integer<_OffsetType>::value
365 || __detail::__integral_constant_like<_OffsetType>);
366 static_assert(__is_signed_or_unsigned_integer<_ExtentType>::value
367 || __detail::__integral_constant_like<_ExtentType>);
368 static_assert(__is_signed_or_unsigned_integer<_StrideType>::value
369 || __detail::__integral_constant_like<_StrideType>);
370
371 using offset_type = _OffsetType;
372 using extent_type = _ExtentType;
373 using stride_type = _StrideType;
374
375 [[no_unique_address]] offset_type offset{};
376 [[no_unique_address]] extent_type extent{};
377 [[no_unique_address]] stride_type stride{};
378 };
379
380 template<typename _FirstType, typename _LastType, typename _StrideType = constant_wrapper<1zu>>
381 struct range_slice
382 {
383 static_assert(__is_signed_or_unsigned_integer<_FirstType>::value
384 || __detail::__integral_constant_like<_FirstType>);
385 static_assert(__is_signed_or_unsigned_integer<_LastType>::value
386 || __detail::__integral_constant_like<_LastType>);
387 static_assert(__is_signed_or_unsigned_integer<_StrideType>::value
388 || __detail::__integral_constant_like<_StrideType>);
389
390 [[no_unique_address]] _FirstType first{};
391 [[no_unique_address]] _LastType last{};
392 [[no_unique_address]] _StrideType stride{};
393 };
394
395 template<typename _Mapping>
396 struct submdspan_mapping_result
397 {
398 [[no_unique_address]] _Mapping mapping = _Mapping();
399 size_t offset{};
400 };
401
402 template<typename _Tp>
403 constexpr bool __is_submdspan_mapping_result = false;
404
405 template<typename _Mapping>
406 constexpr bool __is_submdspan_mapping_result<submdspan_mapping_result<_Mapping>> = true;
407
408 template<typename _Mapping>
409 concept __submdspan_mapping_result = __is_submdspan_mapping_result<_Mapping>;
410
411#endif // __glibcxx_submdspan
412
413 template<typename _IndexType, size_t... _Extents>
414 class extents
415 {
416 static_assert(__is_signed_or_unsigned_integer<_IndexType>::value,
417 "IndexType must be a signed or unsigned integer type");
418 static_assert(
419 (__mdspan::__valid_static_extent<_Extents, _IndexType> && ...),
420 "Extents must either be dynamic or representable as IndexType");
421
422 using _Storage = __mdspan::_ExtentsStorage<
423 _IndexType, array<size_t, sizeof...(_Extents)>{_Extents...}>;
424 [[no_unique_address]] _Storage _M_exts;
425
426 public:
427 using index_type = _IndexType;
428 using size_type = make_unsigned_t<index_type>;
429 using rank_type = size_t;
430
431 static constexpr rank_type
432 rank() noexcept { return _Storage::_S_rank; }
433
434 static constexpr rank_type
435 rank_dynamic() noexcept { return _Storage::_S_rank_dynamic; }
436
437 static constexpr size_t
438 static_extent(rank_type __r) noexcept
439 {
440 __glibcxx_assert(__r < rank());
441 if constexpr (rank() == 0)
442 __builtin_trap();
443 else
444 return _Storage::_S_static_extent(__r);
445 }
446
447 constexpr index_type
448 extent(rank_type __r) const noexcept
449 {
450 __glibcxx_assert(__r < rank());
451 if constexpr (rank() == 0)
452 __builtin_trap();
453 else
454 return _M_exts._M_extent(__r);
455 }
456
457 constexpr
458 extents() noexcept = default;
459
460 private:
461 static consteval bool
462 _S_is_less_dynamic(size_t __ext, size_t __oext)
463 { return (__ext != dynamic_extent) && (__oext == dynamic_extent); }
464
465 template<typename _OIndexType, size_t... _OExtents>
466 static consteval bool
467 _S_ctor_explicit()
468 {
469 return (_S_is_less_dynamic(_Extents, _OExtents) || ...)
470 || (__gnu_cxx::__int_traits<index_type>::__max
471 < __gnu_cxx::__int_traits<_OIndexType>::__max);
472 }
473
474 template<size_t... _OExtents>
475 static consteval bool
476 _S_is_compatible_extents()
477 {
478 if constexpr (sizeof...(_OExtents) != rank())
479 return false;
480 else
481 return ((_OExtents == dynamic_extent || _Extents == dynamic_extent
482 || _OExtents == _Extents) && ...);
483 }
484
485 public:
486 template<typename _OIndexType, size_t... _OExtents>
487 requires (_S_is_compatible_extents<_OExtents...>())
488 constexpr explicit(_S_ctor_explicit<_OIndexType, _OExtents...>())
489 extents(const extents<_OIndexType, _OExtents...>& __other) noexcept
490 : _M_exts(__other._M_exts)
491 { }
492
493 template<__mdspan::__valid_index_type<index_type>... _OIndexTypes>
494 requires (sizeof...(_OIndexTypes) == rank()
495 || sizeof...(_OIndexTypes) == rank_dynamic())
496 constexpr explicit extents(_OIndexTypes... __exts) noexcept
497 : _M_exts(span<const _IndexType, sizeof...(_OIndexTypes)>(
498 initializer_list{static_cast<_IndexType>(std::move(__exts))...}))
499 { }
500
501 template<typename _OIndexType, size_t _Nm>
502 requires __mdspan::__valid_index_type<const _OIndexType&, index_type>
503 && (_Nm == rank() || _Nm == rank_dynamic())
504 constexpr explicit(_Nm != rank_dynamic())
505 extents(span<_OIndexType, _Nm> __exts) noexcept
506 : _M_exts(span<const _OIndexType, _Nm>(__exts))
507 { }
508
509 template<typename _OIndexType, size_t _Nm>
510 requires __mdspan::__valid_index_type<const _OIndexType&, index_type>
511 && (_Nm == rank() || _Nm == rank_dynamic())
512 constexpr explicit(_Nm != rank_dynamic())
513 extents(const array<_OIndexType, _Nm>& __exts) noexcept
514 : _M_exts(span<const _OIndexType, _Nm>(__exts))
515 { }
516
517 template<typename _OIndexType, size_t... _OExtents>
518 friend constexpr bool
519 operator==(const extents& __self,
520 const extents<_OIndexType, _OExtents...>& __other) noexcept
521 {
522 if constexpr (!_S_is_compatible_extents<_OExtents...>())
523 return false;
524 else
525 {
526 auto __impl = [&__self, &__other]<size_t... _Counts>(
527 index_sequence<_Counts...>)
528 { return (cmp_equal(__self.extent(_Counts),
529 __other.extent(_Counts)) && ...); };
530 return __impl(make_index_sequence<__self.rank()>());
531 }
532 }
533
534 private:
535 friend constexpr const array<size_t, rank()>&
536 __mdspan::__static_extents<extents>() noexcept;
537
538 friend constexpr span<const index_type>
539 __mdspan::__dynamic_extents<extents>(const extents&, size_t, size_t)
540 noexcept;
541
542 template<typename _OIndexType, size_t... _OExtents>
543 friend class extents;
544 };
545
546 namespace __mdspan
547 {
548 template<typename _Tp, size_t _Nm>
549 constexpr bool
550 __contains_zero(span<_Tp, _Nm> __exts) noexcept
551 {
552 for (size_t __i = 0; __i < __exts.size(); ++__i)
553 if (__exts[__i] == 0)
554 return true;
555 return false;
556 }
557
558 template<typename _Tp, size_t _Nm>
559 consteval bool
560 __contains_zero(const array<_Tp, _Nm>& __exts) noexcept
561 { return __contains_zero(span<const _Tp>(__exts)); }
562
563 template<typename _Extents>
564 constexpr bool
565 __empty(const _Extents& __exts) noexcept
566 {
567 if constexpr (__contains_zero(__static_extents<_Extents>()))
568 return true;
569 else if constexpr (_Extents::rank_dynamic() > 0)
570 return __contains_zero(__dynamic_extents(__exts));
571 else
572 return false;
573 }
574
575 template<typename _Extents>
576 constexpr typename _Extents::index_type
577 __extents_prod(const _Extents& __exts, size_t __sta_prod, size_t __begin,
578 size_t __end) noexcept
579 {
580 if (__sta_prod == 0)
581 return 0;
582
583 size_t __ret = __sta_prod;
584 if constexpr (_Extents::rank_dynamic() > 0)
585 for (auto __factor : __dynamic_extents(__exts, __begin, __end))
586 __ret *= size_t(__factor);
587 return static_cast<typename _Extents::index_type>(__ret);
588 }
589
590 // Preconditions: _r < _Extents::rank()
591 template<typename _Extents>
592 constexpr typename _Extents::index_type
593 __fwd_prod(const _Extents& __exts, size_t __begin, size_t __end) noexcept
594 {
595 size_t __sta_prod = [__begin, __end] {
596 span<const size_t> __sta_exts
597 = __static_extents<_Extents>(__begin, __end);
598 size_t __ret = 1;
599 for(auto __ext : __sta_exts)
600 if (__ext != dynamic_extent)
601 __ret *= __ext;
602 return __ret;
603 }();
604 return __extents_prod(__exts, __sta_prod, __begin, __end);
605 }
606
607 template<typename _Extents>
608 constexpr typename _Extents::index_type
609 __fwd_prod(const _Extents& __exts, size_t __r) noexcept
610 {
611 constexpr size_t __rank = _Extents::rank();
612 constexpr auto& __sta_exts = __static_extents<_Extents>();
613 if constexpr (__rank == 1)
614 return 1;
615 else if constexpr (__rank == 2)
616 return __r == 0 ? 1 : __exts.extent(0);
617 else if constexpr (__all_dynamic(std::span(__sta_exts).first(__rank-1)))
618 return __extents_prod(__exts, 1, 0, __r);
619 else
620 {
621 size_t __sta_prod = __fwd_partial_prods<__sta_exts>[__r];
622 return __extents_prod(__exts, __sta_prod, 0, __r);
623 }
624 }
625
626 template<typename _IndexType, size_t _Nm>
627 consteval _IndexType
628 __fwd_prod(span<const _IndexType, _Nm> __values)
629 {
630 _IndexType __ret = 1;
631 for(auto __value : __values)
632 __ret *= __value;
633 return __ret;
634 }
635
636 // Preconditions: _r < _Extents::rank()
637 template<typename _Extents>
638 constexpr typename _Extents::index_type
639 __rev_prod(const _Extents& __exts, size_t __r) noexcept
640 {
641 constexpr size_t __rank = _Extents::rank();
642 constexpr auto& __sta_exts = __static_extents<_Extents>();
643 if constexpr (__rank == 1)
644 return 1;
645 else if constexpr (__rank == 2)
646 return __r == 0 ? __exts.extent(1) : 1;
647 else if constexpr (__all_dynamic(std::span(__sta_exts).last(__rank-1)))
648 return __extents_prod(__exts, 1, __r + 1, __rank);
649 else
650 {
651 size_t __sta_prod = __rev_partial_prods<__sta_exts>[__r];
652 return __extents_prod(__exts, __sta_prod, __r + 1, __rank);
653 }
654 }
655
656 template<typename _Extents>
657 constexpr typename _Extents::index_type
658 __size(const _Extents& __exts) noexcept
659 {
660 constexpr size_t __sta_prod = [] {
661 span<const size_t> __sta_exts = __static_extents<_Extents>();
662 size_t __ret = 1;
663 for(auto __ext : __sta_exts)
664 if (__ext != dynamic_extent)
665 __ret *= __ext;
666 return __ret;
667 }();
668 return __extents_prod(__exts, __sta_prod, 0, _Extents::rank());
669 }
670
671 template<typename _IndexType, size_t... _Counts>
672 auto __build_dextents_type(integer_sequence<size_t, _Counts...>)
673 -> extents<_IndexType, ((void) _Counts, dynamic_extent)...>;
674 }
675
676 template<typename _IndexType, size_t _Rank>
677 using dextents = decltype(__mdspan::__build_dextents_type<_IndexType>(
679
680#if __glibcxx_mdspan >= 202406L
681 template<size_t _Rank, typename _IndexType = size_t>
682 using dims = dextents<_IndexType, _Rank>;
683#endif
684
685 template<typename... _Integrals>
686 requires (is_convertible_v<_Integrals, size_t> && ...)
687 explicit extents(_Integrals...) ->
688 extents<size_t, __detail::__maybe_static_ext<_Integrals>...>;
689
690 struct layout_left
691 {
692 template<typename _Extents>
693 class mapping;
694 };
695
696 struct layout_right
697 {
698 template<typename _Extents>
699 class mapping;
700 };
701
702 struct layout_stride
703 {
704 template<typename _Extents>
705 class mapping;
706 };
707
708#ifdef __glibcxx_padded_layouts
709 template<size_t _PaddingValue>
710 struct layout_left_padded
711 {
712 template<typename _Extents>
713 class mapping;
714 };
715
716 template<size_t _PaddingValue>
717 struct layout_right_padded
718 {
719 template<typename _Extents>
720 class mapping;
721 };
722#endif
723
724 namespace __mdspan
725 {
726 template<typename _Tp>
727 constexpr bool __is_extents = false;
728
729 template<typename _IndexType, size_t... _Extents>
730 constexpr bool __is_extents<extents<_IndexType, _Extents...>> = true;
731
732 template<typename _Extents, typename... _Indices>
733 constexpr typename _Extents::index_type
734 __linear_index_left(const _Extents& __exts, _Indices... __indices)
735 noexcept
736 {
737 using _IndexType = typename _Extents::index_type;
738 _IndexType __res = 0;
739 if constexpr (sizeof...(__indices) > 0)
740 {
741 _IndexType __mult = 1;
742 auto __update = [&, __pos = 0u](_IndexType __idx) mutable
743 {
744 _GLIBCXX_DEBUG_ASSERT(cmp_less(__idx, __exts.extent(__pos)));
745 __res += __idx * __mult;
746 __mult *= __exts.extent(__pos);
747 ++__pos;
748 };
749 (__update(__indices), ...);
750 }
751 return __res;
752 }
753
754 template<typename _IndexType>
755 consteval _IndexType
756 __static_quotient(std::span<const size_t> __sta_exts,
757 _IndexType __nom = __gnu_cxx::__int_traits<_IndexType>::__max)
758 {
759 for (auto __factor : __sta_exts)
760 {
761 if (__factor != dynamic_extent)
762 __nom /= _IndexType(__factor);
763 if (__nom == 0)
764 break;
765 }
766 return __nom;
767 }
768
769 template<typename _Extents,
770 typename _IndexType = typename _Extents::index_type>
771 requires __is_extents<_Extents>
772 consteval _IndexType
773 __static_quotient(_IndexType __nom
774 = __gnu_cxx::__int_traits<_IndexType>::__max)
775 {
776 std::span<const size_t> __sta_exts = __static_extents<_Extents>();
777 return __static_quotient<_IndexType>(__sta_exts, __nom);
778 }
779
780 template<typename _Extents>
781 constexpr bool
782 __is_representable_extents(const _Extents& __exts) noexcept
783 {
784 using _IndexType = _Extents::index_type;
785
786 if constexpr (__contains_zero(__static_extents<_Extents>()))
787 return true;
788 else
789 {
790 constexpr auto __sta_quo = __static_quotient<_Extents>();
791 if constexpr (_Extents::rank_dynamic() == 0)
792 return __sta_quo != 0;
793 else
794 {
795 auto __dyn_exts = __dynamic_extents(__exts);
796 if (__contains_zero(__dyn_exts))
797 return true;
798
799 if constexpr (__sta_quo == 0)
800 return false;
801 else
802 {
803 auto __dyn_quo = _IndexType(__sta_quo);
804 for (auto __factor : __dyn_exts)
805 {
806 __dyn_quo /= __factor;
807 if (__dyn_quo == 0)
808 return false;
809 }
810 return true;
811 }
812 }
813 }
814 }
815
816 template<typename _Extents, typename _IndexType>
817 concept __representable_size = _Extents::rank_dynamic() != 0
818 || __contains_zero(__static_extents<_Extents>())
819 || (__static_quotient<_Extents, _IndexType>() != 0);
820
821 template<typename _Layout, typename _Mapping>
822 concept __mapping_of =
823 is_same_v<typename _Layout::template mapping<
824 typename _Mapping::extents_type>,
825 _Mapping>;
826
827 template<template<size_t> typename _Layout, typename _Mapping>
828 concept __padded_mapping_of = __mapping_of<
829 _Layout<_Mapping::padding_value>, _Mapping>;
830
831#ifdef __glibcxx_padded_layouts
832 template<typename _Mapping>
833 constexpr bool __is_left_padded_mapping = __padded_mapping_of<
834 layout_left_padded, _Mapping>;
835
836 template<typename _Mapping>
837 constexpr bool __is_right_padded_mapping = __padded_mapping_of<
838 layout_right_padded, _Mapping>;
839
840 template<typename _Mapping>
841 constexpr bool __is_padded_mapping = __is_left_padded_mapping<_Mapping>
842 || __is_right_padded_mapping<_Mapping>;
843#endif
844
845 template<typename _PaddedMapping>
846 consteval size_t
847 __get_static_stride()
848 { return _PaddedMapping::_PaddedStorage::_S_static_stride; }
849
850 template<typename _Mapping>
851 concept __standardized_mapping = __mapping_of<layout_left, _Mapping>
852 || __mapping_of<layout_right, _Mapping>
853 || __mapping_of<layout_stride, _Mapping>
854#ifdef __glibcxx_padded_layouts
855 || __is_left_padded_mapping<_Mapping>
856 || __is_right_padded_mapping<_Mapping>
857#endif
858 ;
859
860 // A tag type to create internal ctors.
861 class __internal_ctor
862 { };
863
864 template<typename _Mapping>
865 constexpr typename _Mapping::index_type
866 __offset(const _Mapping& __m) noexcept
867 {
868 using _IndexType = typename _Mapping::index_type;
869 constexpr auto __rank = _Mapping::extents_type::rank();
870
871 if constexpr (__standardized_mapping<_Mapping>)
872 return 0;
873 else if (__empty(__m.extents()))
874 return 0;
875 else
876 {
877 auto __impl = [&__m]<size_t... _Counts>(index_sequence<_Counts...>)
878 { return __m(((void) _Counts, _IndexType(0))...); };
879 return __impl(make_index_sequence<__rank>());
880 }
881 }
882
883#ifdef __glibcxx_submdspan
884 template<typename _Tp>
885 constexpr bool __is_extent_slice = false;
886
887 template<typename _OffsetType, typename _ExtentType, typename _StrideType>
888 constexpr bool __is_extent_slice<extent_slice<_OffsetType,
889 _ExtentType, _StrideType>> = true;
890
891 template<typename _Tp>
892 constexpr bool __is_range_slice = false;
893
894 template<typename _FirstType, typename _LastType, typename _StrideType>
895 constexpr bool __is_range_slice<range_slice<_FirstType,
896 _LastType, _StrideType>> = true;
897
898 template<typename _IndexType, typename _OIndexType>
899 consteval bool
900 __is_representable_integer(_OIndexType __value)
901 {
902 constexpr auto __min = __gnu_cxx::__int_traits<_IndexType>::__min;
903 constexpr auto __max = __gnu_cxx::__int_traits<_IndexType>::__max;
904 return std::cmp_less_equal(__min, __value)
905 && std::cmp_less_equal(__value, __max);
906 }
907
908 template<size_t _Index, typename _Extents>
909 constexpr auto
910 __extract_extent(const _Extents& __exts)
911 {
912 using _IndexType = typename _Extents::index_type;
913 return extents<_IndexType, _Extents::static_extent(_Index)>{
914 __exts.extent(_Index)};
915 }
916
917 template<typename _Slice, typename _IndexType>
918 concept __acceptable_slice_type = same_as<_Slice, full_extent_t>
919 || same_as<_Slice, _IndexType> || __is_constant_wrapper_v<_Slice>
920 || __is_extent_slice<_Slice>;
921
922 template<typename _IndexType, typename... _Slices>
923 consteval auto
924 __subrank()
925 {
926 return (static_cast<size_t>(!convertible_to<_Slices, _IndexType>)
927 + ... + 0);
928 }
929
930 template<typename _IndexType, typename... _Slices>
931 consteval auto
932 __inv_map_rank()
933 {
934 constexpr auto __rank = sizeof...(_Slices);
935 constexpr auto __sub_rank = __subrank<_IndexType, _Slices...>();
936 auto __map = std::array<size_t, __sub_rank>{};
937 auto __is_int_like = std::array<bool, __rank>{
938 convertible_to<_Slices, _IndexType>...};
939
940 size_t __i = 0;
941 for (size_t __k = 0; __k < __rank; ++__k)
942 if (!__is_int_like[__k])
943 __map[__i++] = __k;
944 return __map;
945 }
946
947 template<typename _Slice>
948 constexpr auto
949 __slice_begin(_Slice __slice)
950 {
951 if constexpr (same_as<_Slice, full_extent_t>)
952 return 0;
953 else if constexpr (__is_extent_slice<_Slice>)
954 return __slice.offset;
955 else
956 return __slice; // collapsing slice
957 }
958
959 template<typename _Mapping, typename... _Slices>
960 constexpr size_t
961 __suboffset(const _Mapping& __mapping, const _Slices&... __slices)
962 {
963 using _IndexType = typename _Mapping::index_type;
964 auto __any_past_the_end = [&]<size_t... _Is>(index_sequence<_Is...>)
965 {
966 auto __is_past_the_end = [](const auto& __slice, const auto& __ext)
967 {
968 using _Slice = remove_cvref_t<decltype(__slice)>;
969 if constexpr (is_convertible_v<_Slice, _IndexType>)
970 return false;
971 else if constexpr (same_as<_Slice, full_extent_t>
972 && __ext.static_extent(0) != 0
973 && __ext.static_extent(0) != dynamic_extent)
974 return false;
975 else
976 return __mdspan::__slice_begin(__slice) == __ext.extent(0);
977 };
978
979 const auto& __exts = __mapping.extents();
980 return ((__is_past_the_end(__slices...[_Is],
981 __mdspan::__extract_extent<_Is>(__exts))) || ...);
982 };
983
984 if constexpr ((same_as<_Slices, full_extent_t> && ...))
985 return __mdspan::__offset(__mapping);
986
987 if (__any_past_the_end(std::make_index_sequence<sizeof...(__slices)>()))
988 return __mapping.required_span_size();
989 return __mapping(__mdspan::__slice_begin(__slices)...);
990 }
991
992 template<typename _IndexType, size_t _Extent, typename _Slice>
993 consteval size_t
994 __static_slice_extent()
995 {
996 if constexpr (same_as<_Slice, full_extent_t>)
997 return _Extent;
998 else if constexpr (same_as<_Slice, constant_wrapper<_IndexType(0)>>)
999 return 0;
1000 else if constexpr (__is_constant_wrapper_v<typename _Slice::extent_type>)
1001 return _Slice::extent_type::value;
1002 else
1003 return dynamic_extent;
1004 }
1005
1006 template<size_t _K, typename _Extents, typename _Slice>
1007 constexpr typename _Extents::index_type
1008 __dynamic_slice_extent(const _Extents& __exts, _Slice __slice)
1009 {
1010 if constexpr (__is_extent_slice<_Slice>)
1011 return __slice.extent;
1012 else
1013 return __exts.extent(_K);
1014 }
1015
1016 template<typename _IndexType, size_t... _Extents, typename... _Slices>
1017 requires (sizeof...(_Slices) == sizeof...(_Extents))
1018 constexpr auto
1019 __subextents(const extents<_IndexType, _Extents...>& __exts,
1020 _Slices... __slices)
1021 {
1022 constexpr auto __inv_map = __mdspan::__inv_map_rank<_IndexType, _Slices...>();
1023 auto __impl = [&]<size_t... _Indices>(std::index_sequence<_Indices...>)
1024 {
1025 using _SubExts = extents<_IndexType,
1026 __mdspan::__static_slice_extent<_IndexType,
1027 _Extents...[__inv_map[_Indices]],
1028 _Slices...[__inv_map[_Indices]]>()...>;
1029 if constexpr (_SubExts::rank_dynamic() == 0)
1030 return _SubExts{};
1031 else
1032 {
1033 using _StaticSubExtents = __mdspan::_StaticExtents<
1034 __mdspan::__static_extents<_SubExts>()>;
1035 auto __create = [&]<size_t... _Is>(std::index_sequence<_Is...>)
1036 {
1037 constexpr auto __slice_idx = [__inv_map](size_t __i) consteval
1038 {
1039 return __inv_map[_StaticSubExtents::_S_dynamic_index_inv(__i)];
1040 };
1041
1042 return _SubExts{__mdspan::__dynamic_slice_extent<__slice_idx(_Is)>(
1043 __exts, __slices...[__slice_idx(_Is)])...};
1044 };
1045 constexpr auto __dyn_subrank = _SubExts::rank_dynamic();
1046 return __create(std::make_index_sequence<__dyn_subrank>());
1047 }
1048 };
1049
1050 return __impl(std::make_index_sequence<__inv_map.size()>());
1051 }
1052
1053 enum class _LayoutSide
1054 {
1055 __left,
1056 __right,
1057 __unknown
1058 };
1059
1060 template<typename _Mapping>
1061 consteval _LayoutSide
1062 __mapping_side()
1063 {
1064 if constexpr (__is_left_padded_mapping<_Mapping>
1065 || __mapping_of<layout_left, _Mapping>)
1066 return _LayoutSide::__left;
1067 if constexpr (__is_right_padded_mapping<_Mapping>
1068 || __mapping_of<layout_right, _Mapping>)
1069 return _LayoutSide::__right;
1070 else
1071 return _LayoutSide::__unknown;
1072 }
1073
1074 template<_LayoutSide _Side, size_t _Rank>
1075 struct _StridesTrait
1076 {
1077 static constexpr const _LayoutSide _S_side = _Side;
1078
1079 static constexpr size_t
1080 _S_idx(size_t __k) noexcept
1081 {
1082 if constexpr (_Side == _LayoutSide::__left)
1083 return __k;
1084 else
1085 return _Rank - 1 - __k;
1086 }
1087
1088 // Unifies the formulas for computing strides for padded and unpadded
1089 // layouts.
1090 template<typename _Mapping>
1091 static constexpr typename _Mapping::index_type
1092 _S_padded_extent(const _Mapping& __mapping, size_t __k)
1093 {
1094 if (__k == 0)
1095 return __mapping.stride(_S_idx(1));
1096 else
1097 return __mapping.extents().extent(_S_idx(__k));
1098 }
1099
1100 template<typename _IndexType, typename... _Slices>
1101 static consteval auto
1102 _S_inv_map()
1103 {
1104 static_assert(_Side != _LayoutSide::__unknown);
1105 auto __impl = [&]<size_t... _Is>(std::index_sequence<_Is...>)
1106 {
1107 return __mdspan::__inv_map_rank<_IndexType, _Slices...[_S_idx(_Is)]...>();
1108 };
1109 return __impl(std::make_index_sequence<_Rank>());
1110 }
1111 };
1112
1113 template<typename _SubExts, typename _Mapping, typename... _Slices>
1114 constexpr auto
1115 __substrides_generic(const _Mapping& __mapping, const _Slices&... __slices)
1116 {
1117 using _IndexType = typename _Mapping::index_type;
1118 if constexpr (_SubExts::rank() == 0)
1119 return array<_IndexType, _SubExts::rank()>{};
1120 else
1121 {
1122 auto __stride = [&__mapping](size_t __k, auto __slice) -> _IndexType
1123 {
1124 if constexpr (__is_extent_slice<decltype(__slice)>)
1125 if (__slice.extent > 1)
1126 return __mapping.stride(__k) * __slice.stride;
1127 return __mapping.stride(__k);
1128 };
1129
1130 auto __impl = [&]<size_t... _Is>(std::index_sequence<_Is...>)
1131 {
1132 constexpr auto __inv_map
1133 = __mdspan::__inv_map_rank<_IndexType, _Slices...>();
1134 return array<_IndexType, _SubExts::rank()>{
1135 __stride(__inv_map[_Is], __slices...[__inv_map[_Is]])...};
1136 };
1137 return __impl(std::make_index_sequence<_SubExts::rank()>());
1138 }
1139 };
1140
1141 template<typename _SubExts, typename _Mapping, typename... _Slices>
1142 constexpr auto
1143 __substrides_standardized(const _Mapping& __mapping,
1144 const _Slices&... __slices)
1145 {
1146 using _IndexType = typename _Mapping::index_type;
1147 using _Trait = _StridesTrait<__mapping_side<_Mapping>(),
1148 _Mapping::extents_type::rank()>;
1149 using _SubTrait = _StridesTrait<__mapping_side<_Mapping>(), _SubExts::rank()>;
1150
1151 constexpr size_t __sub_rank = _SubExts::rank();
1152
1153 std::array<_IndexType, __sub_rank> __ret;
1154 if constexpr (__sub_rank > 0)
1155 {
1156 constexpr auto __inv_map
1157 = _Trait::template _S_inv_map<_IndexType, _Slices...>();
1158 auto __loop = [&]<size_t... _Ks>(std::index_sequence<_Ks...>)
1159 {
1160 size_t __i0 = 0;
1161 size_t __stride = 1;
1162 auto __body = [&](size_t __k, auto __slice)
1163 {
1164 for (size_t __i = __i0; __i < __inv_map[__k]; ++__i)
1165 __stride *= _Trait::_S_padded_extent(__mapping, __i);
1166
1167 size_t __krev = _SubTrait::_S_idx(__k);
1168 if constexpr (__is_extent_slice<decltype(__slice)>)
1169 {
1170 if (__slice.extent > 1)
1171 __ret[__krev] = __stride * __slice.stride;
1172 else
1173 __ret[__krev] = __stride;
1174 }
1175 else
1176 __ret[__krev] = __stride;
1177
1178 __i0 = __inv_map[__k];
1179 };
1180
1181 ((__body(_Ks, __slices...[_Trait::_S_idx(__inv_map[_Ks])])),...);
1182 };
1184 }
1185 return __ret;
1186 }
1187
1188
1189 template<typename _SubExts, typename _Mapping, typename... _Slices>
1190 constexpr auto
1191 __substrides(const _Mapping& __mapping, const _Slices&... __slices)
1192 {
1193 if constexpr (__mdspan::__mapping_side<_Mapping>() == _LayoutSide::__unknown)
1194 return __mdspan::__substrides_generic<_SubExts>(__mapping, __slices...);
1195 else
1196 return __mdspan::__substrides_standardized<_SubExts>(__mapping, __slices...);
1197 }
1198
1199 template<typename _Slice>
1200 concept __is_unit_stride_slice = (__mdspan::__is_extent_slice<_Slice>
1201 && __is_constant_wrapper_v<typename _Slice::stride_type>
1202 && _Slice::stride_type::value == 1)
1203 || std::same_as<_Slice, full_extent_t>;
1204
1205 // These are (forced) exclusive categories:
1206 // - full & collapsing: obvious,
1207 // - unit_extent_slice: extent_slice{a, b, cw<1>}, but not `full`,
1208 // - extent_slice: extent_slice{a, b, c} with c != cw<1>.
1209 enum class _SliceKind
1210 {
1211 __extent_slice,
1212 __unit_stride_slice,
1213 __full,
1214 __collapsing
1215 };
1216
1217 template<typename _Slice>
1218 consteval _SliceKind
1219 __make_slice_kind()
1220 {
1221 if constexpr (std::same_as<_Slice, full_extent_t>)
1222 return _SliceKind::__full;
1223 else if constexpr (__mdspan::__is_extent_slice<_Slice>)
1224 {
1225 if constexpr (__mdspan::__is_unit_stride_slice<_Slice>)
1226 return _SliceKind::__unit_stride_slice;
1227 else
1228 return _SliceKind::__extent_slice;
1229 }
1230 else
1231 return _SliceKind::__collapsing;
1232 }
1233
1234 template<typename... _Slices>
1235 consteval array<_SliceKind, sizeof...(_Slices)>
1236 __make_slice_kind_array()
1237 {
1238 return array<_SliceKind, sizeof...(_Slices)>{
1239 __mdspan::__make_slice_kind<_Slices>()...};
1240 }
1241
1242 // __block_size - 1
1243 // [full, ..., full, unit_slice , *]
1244 consteval bool
1245 __is_block(span<const _SliceKind> __slice_kinds, size_t __block_size)
1246 {
1247 if (__block_size == 0)
1248 return false;
1249
1250 if (__block_size > __slice_kinds.size())
1251 return false;
1252
1253 for (size_t __i = 0; __i < __block_size - 1; ++__i)
1254 if (__slice_kinds[__i] != _SliceKind::__full)
1255 return false;
1256
1257 auto __last = __slice_kinds[__block_size - 1];
1258 return __last == _SliceKind::__full
1259 || __last == _SliceKind::__unit_stride_slice;
1260 }
1261
1262 // __u __u + __sub_rank-2
1263 // [unit_slice, i, ..., k, full, ..., full, unit_slice, *]
1264 static consteval size_t
1265 __padded_block_begin_generic(span<const _SliceKind> __slice_kinds,
1266 size_t __sub_rank)
1267 {
1268 if (__slice_kinds[0] != _SliceKind::__full
1269 && __slice_kinds[0] != _SliceKind::__unit_stride_slice)
1270 return dynamic_extent;
1271 else if (__slice_kinds.size() == 1)
1272 return dynamic_extent;
1273 else
1274 {
1275 size_t __u = 1;
1276 while(__u < __slice_kinds.size()
1277 && __slice_kinds[__u] == _SliceKind::__collapsing)
1278 ++__u;
1279
1280 if (__mdspan::__is_block(__slice_kinds.subspan(__u), __sub_rank -1))
1281 return __u;
1282 return dynamic_extent;
1283 }
1284 }
1285
1286 template<_LayoutSide _Side, size_t _Nm>
1287 static consteval size_t
1288 __padded_block_begin(span<const _SliceKind, _Nm> __slice_kinds, size_t __sub_rank)
1289 {
1290 if constexpr (_Side == _LayoutSide::__left)
1291 return __mdspan::__padded_block_begin_generic(__slice_kinds, __sub_rank);
1292 else
1293 {
1294 std::array<_SliceKind, _Nm> __rev_slices;
1295 for(size_t __i = 0; __i < _Nm; ++__i)
1296 __rev_slices[__i] = __slice_kinds[_Nm - 1 - __i];
1297 auto __rev_slice_kinds = span<const _SliceKind>(__rev_slices);
1298
1299 auto __u = __mdspan::__padded_block_begin_generic(__rev_slice_kinds,
1300 __sub_rank);
1301 return __u == dynamic_extent ? dynamic_extent : _Nm - 1 - __u;
1302 }
1303 }
1304
1305 template<_LayoutSide _Side, bool _Padded>
1306 struct _SubMdspanMapping;
1307
1308 template<>
1309 struct _SubMdspanMapping<_LayoutSide::__left, false>
1310 {
1311 using _Layout = layout_left;
1312 template<size_t _Pad> using _PaddedLayout = layout_left_padded<_Pad>;
1313
1314 template<typename _Mapping, size_t _Us>
1315 static consteval size_t
1316 _S_pad()
1317 {
1318 using _Extents = typename _Mapping::extents_type;
1319 constexpr auto __sta_exts = __mdspan::__static_extents<_Extents>(0, _Us);
1320 if constexpr (!__mdspan::__all_static(__sta_exts))
1321 return dynamic_extent;
1322 else
1323 return __mdspan::__fwd_prod(__sta_exts);
1324 }
1325
1326 template<size_t _Nm>
1327 static consteval bool
1328 _S_is_unpadded_submdspan(span<const _SliceKind, _Nm> __slice_kinds, size_t __sub_rank)
1329 { return __mdspan::__is_block(__slice_kinds, __sub_rank); }
1330 };
1331
1332 template<>
1333 struct _SubMdspanMapping<_LayoutSide::__left, true>
1334 {
1335 using _Layout = layout_left;
1336 template<size_t _Pad> using _PaddedLayout = layout_left_padded<_Pad>;
1337
1338 template<typename _Mapping, size_t _Us>
1339 static consteval size_t
1340 _S_pad()
1341 {
1342 using _Extents = typename _Mapping::extents_type;
1343 constexpr auto __sta_exts
1344 = __mdspan::__static_extents<_Extents>(1, _Us);
1345 constexpr auto __sta_padstride
1346 = __mdspan::__get_static_stride<_Mapping>();
1347 if constexpr (__sta_padstride == dynamic_extent
1348 || !__mdspan::__all_static(__sta_exts))
1349 return dynamic_extent;
1350 else
1351 return __sta_padstride * __mdspan::__fwd_prod(__sta_exts);
1352 }
1353
1354 template<size_t _Nm>
1355 static consteval bool
1356 _S_is_unpadded_submdspan(span<const _SliceKind, _Nm> __slice_kinds,
1357 size_t __sub_rank)
1358 {
1359 if (__sub_rank == 1)
1360 return __slice_kinds[0] == _SliceKind::__unit_stride_slice
1361 || __slice_kinds[0] == _SliceKind::__full;
1362 else
1363 return false;
1364 }
1365 };
1366
1367 template<>
1368 struct _SubMdspanMapping<_LayoutSide::__right, false>
1369 {
1370 using _Layout = layout_right;
1371 template<size_t _Pad> using _PaddedLayout = layout_right_padded<_Pad>;
1372
1373 template<typename _Mapping, size_t _Us>
1374 static consteval size_t
1375 _S_pad()
1376 {
1377 using _Extents = typename _Mapping::extents_type;
1378 constexpr auto __rank = _Extents::rank();
1379 constexpr auto __sta_exts
1380 = __mdspan::__static_extents<_Extents>(_Us + 1, __rank);
1381 if constexpr (!__mdspan::__all_static(__sta_exts))
1382 return dynamic_extent;
1383 else
1384 return __fwd_prod(__sta_exts);
1385 }
1386
1387 template<size_t _Nm>
1388 static consteval bool
1389 _S_is_unpadded_submdspan(span<const _SliceKind, _Nm> __slice_kinds,
1390 size_t __sub_rank)
1391 {
1392 auto __rev_slice_kinds = array<_SliceKind, _Nm>{};
1393 for(size_t __i = 0; __i < _Nm; ++__i)
1394 __rev_slice_kinds[__i] = __slice_kinds[_Nm - 1 - __i];
1395 return __mdspan::__is_block(span(__rev_slice_kinds), __sub_rank);
1396 }
1397 };
1398
1399 template<>
1400 struct _SubMdspanMapping<_LayoutSide::__right, true>
1401 {
1402 using _Layout = layout_right;
1403 template<size_t _Pad> using _PaddedLayout = layout_right_padded<_Pad>;
1404
1405 template<typename _Mapping, size_t _Us>
1406 static consteval size_t
1407 _S_pad()
1408 {
1409 using _Extents = typename _Mapping::extents_type;
1410 constexpr auto __rank = _Extents::rank();
1411 constexpr auto __sta_exts
1412 = __mdspan::__static_extents<_Extents>(_Us + 1, __rank - 1);
1413 constexpr auto __sta_padstride
1414 = __mdspan::__get_static_stride<_Mapping>();
1415 if constexpr (__sta_padstride == dynamic_extent
1416 || !__mdspan::__all_static(__sta_exts))
1417 return dynamic_extent;
1418 else
1419 return __sta_padstride * __mdspan::__fwd_prod(__sta_exts);
1420 }
1421
1422 template<size_t _Nm>
1423 static consteval bool
1424 _S_is_unpadded_submdspan(span<const _SliceKind, _Nm> __slice_kinds,
1425 size_t __sub_rank)
1426 {
1427 if (__sub_rank == 1)
1428 return __slice_kinds[_Nm - 1] == _SliceKind::__unit_stride_slice
1429 || __slice_kinds[_Nm - 1] == _SliceKind::__full;
1430 else
1431 return false;
1432 }
1433 };
1434
1435
1436 template<typename _Mapping>
1437 constexpr auto
1438 __submdspan_mapping_impl(const _Mapping& __mapping)
1439 { return submdspan_mapping_result{__mapping, 0}; }
1440
1441 template<typename _Mapping, typename... _Slices>
1442 requires (sizeof...(_Slices) > 0)
1443 constexpr auto
1444 __submdspan_mapping_impl(const _Mapping& __mapping, _Slices... __slices)
1445 {
1446 using _IndexType = typename _Mapping::index_type;
1447 static_assert((__acceptable_slice_type<_Slices, _IndexType> && ...));
1448
1449 constexpr auto __side = __mdspan::__mapping_side<_Mapping>();
1450 constexpr auto __rank = sizeof...(_Slices);
1451 using _Trait = _SubMdspanMapping<__side, __is_padded_mapping<_Mapping>>;
1452 using _SliceView = span<const _SliceKind, __rank>;
1453
1454 constexpr auto __slice_kinds = __mdspan::__make_slice_kind_array<_Slices...>();
1455 auto __offset = __mdspan::__suboffset(__mapping, __slices...);
1456 auto __sub_exts = __mdspan::__subextents(__mapping.extents(), __slices...);
1457 using _SubExts = decltype(__sub_exts);
1458 constexpr auto __sub_rank = _SubExts::rank();
1459 if constexpr (__sub_rank == 0)
1460 return submdspan_mapping_result{
1461 typename _Trait::_Layout::mapping(__sub_exts), __offset};
1462 else if constexpr (_Trait::_S_is_unpadded_submdspan(
1463 _SliceView(__slice_kinds), __sub_rank))
1464 return submdspan_mapping_result{
1465 typename _Trait::_Layout::mapping(__sub_exts), __offset};
1466 else if constexpr (
1467 constexpr auto __u = __padded_block_begin<__side>(
1468 _SliceView(__slice_kinds), __sub_rank);
1469 __u != dynamic_extent)
1470 {
1471 constexpr auto __pad = _Trait::template _S_pad<_Mapping, __u>();
1472 using _Layout = typename _Trait::template _PaddedLayout<__pad>;
1473 return submdspan_mapping_result{
1474 typename _Layout::mapping(__sub_exts, __mapping.stride(__u)),
1475 __offset};
1476 }
1477 else
1478 {
1479 auto __sub_strides
1480 = __mdspan::__substrides<_SubExts>(__mapping, __slices...);
1481 return submdspan_mapping_result{
1482 layout_stride::mapping(__sub_exts, __sub_strides), __offset};
1483 }
1484 }
1485#endif // __glibcxx_submdspan
1486 }
1487
1488 template<typename _Extents>
1489 class layout_left::mapping
1490 {
1491 public:
1492 using extents_type = _Extents;
1493 using index_type = typename extents_type::index_type;
1494 using size_type = typename extents_type::size_type;
1495 using rank_type = typename extents_type::rank_type;
1496 using layout_type = layout_left;
1497
1498 static_assert(__mdspan::__representable_size<extents_type, index_type>,
1499 "The size of extents_type must be representable as index_type");
1500
1501 constexpr
1502 mapping() noexcept = default;
1503
1504 constexpr
1505 mapping(const mapping&) noexcept = default;
1506
1507 constexpr
1508 mapping(const extents_type& __extents) noexcept
1509 : _M_extents(__extents)
1510 { __glibcxx_assert(__mdspan::__is_representable_extents(_M_extents)); }
1511
1512 template<typename _OExtents>
1513 requires is_constructible_v<extents_type, _OExtents>
1514 constexpr explicit(!is_convertible_v<_OExtents, extents_type>)
1515 mapping(const mapping<_OExtents>& __other) noexcept
1516 : mapping(__other.extents(), __mdspan::__internal_ctor{})
1517 { }
1518
1519 template<typename _OExtents>
1520 requires (extents_type::rank() <= 1)
1521 && is_constructible_v<extents_type, _OExtents>
1522 constexpr explicit(!is_convertible_v<_OExtents, extents_type>)
1523 mapping(const layout_right::mapping<_OExtents>& __other) noexcept
1524 : mapping(__other.extents(), __mdspan::__internal_ctor{})
1525 { }
1526
1527 // noexcept for consistency with other layouts.
1528 template<typename _OExtents>
1529 requires is_constructible_v<extents_type, _OExtents>
1530 constexpr explicit(!(extents_type::rank() == 0
1531 && is_convertible_v<_OExtents, extents_type>))
1532 mapping(const layout_stride::mapping<_OExtents>& __other) noexcept
1533 : mapping(__other.extents(), __mdspan::__internal_ctor{})
1534 { __glibcxx_assert(*this == __other); }
1535
1536#if __glibcxx_padded_layouts
1537 template<typename _LeftpadMapping>
1538 requires __mdspan::__is_left_padded_mapping<_LeftpadMapping>
1539 && is_constructible_v<extents_type,
1540 typename _LeftpadMapping::extents_type>
1541 constexpr
1542 explicit(!is_convertible_v<typename _LeftpadMapping::extents_type,
1543 extents_type>)
1544 mapping(const _LeftpadMapping& __other) noexcept
1545 : mapping(__other.extents(), __mdspan::__internal_ctor{})
1546 {
1547 constexpr size_t __ostride_sta
1548 = __mdspan::__get_static_stride<_LeftpadMapping>();
1549
1550 if constexpr (extents_type::rank() > 1)
1551 {
1552 if constexpr (extents_type::static_extent(0) != dynamic_extent
1553 && __ostride_sta != dynamic_extent)
1554 static_assert(extents_type::static_extent(0) == __ostride_sta);
1555 else
1556 __glibcxx_assert(__other.stride(1)
1557 == __other.extents().extent(0));
1558 }
1559 }
1560#endif // __glibcxx_padded_layouts
1561
1562 constexpr mapping&
1563 operator=(const mapping&) noexcept = default;
1564
1565 constexpr const extents_type&
1566 extents() const noexcept { return _M_extents; }
1567
1568 constexpr index_type
1569 required_span_size() const noexcept
1570 { return __mdspan::__size(_M_extents); }
1571
1572 // _GLIBCXX_RESOLVE_LIB_DEFECTS
1573 // 4314. Missing move in mdspan layout mapping::operator()
1574 template<__mdspan::__valid_index_type<index_type>... _Indices>
1575 requires (sizeof...(_Indices) == extents_type::rank())
1576 constexpr index_type
1577 operator()(_Indices... __indices) const noexcept
1578 {
1579 return __mdspan::__linear_index_left(_M_extents,
1580 static_cast<index_type>(std::move(__indices))...);
1581 }
1582
1583 static constexpr bool
1584 is_always_unique() noexcept { return true; }
1585
1586 static constexpr bool
1587 is_always_exhaustive() noexcept { return true; }
1588
1589 static constexpr bool
1590 is_always_strided() noexcept { return true; }
1591
1592 static constexpr bool
1593 is_unique() noexcept { return true; }
1594
1595 static constexpr bool
1596 is_exhaustive() noexcept { return true; }
1597
1598 static constexpr bool
1599 is_strided() noexcept { return true; }
1600
1601 constexpr index_type
1602 stride(rank_type __i) const noexcept
1603 requires (extents_type::rank() > 0)
1604 {
1605 __glibcxx_assert(__i < extents_type::rank());
1606 return __mdspan::__fwd_prod(_M_extents, __i);
1607 }
1608
1609 template<typename _OExtents>
1610 requires (extents_type::rank() == _OExtents::rank())
1611 friend constexpr bool
1612 operator==(const mapping& __self, const mapping<_OExtents>& __other)
1613 noexcept
1614 { return __self.extents() == __other.extents(); }
1615
1616 private:
1617 template<typename _OExtents>
1618 constexpr explicit
1619 mapping(const _OExtents& __oexts, __mdspan::__internal_ctor) noexcept
1620 : _M_extents(__oexts)
1621 {
1622 static_assert(__mdspan::__representable_size<_OExtents, index_type>,
1623 "The size of OtherExtents must be representable as index_type");
1624 __glibcxx_assert(__mdspan::__is_representable_extents(_M_extents));
1625 }
1626
1627#if __glibcxx_submdspan
1628 template<typename... _Slices>
1629 requires (extents_type::rank() == sizeof...(_Slices))
1630 friend constexpr auto
1631 submdspan_mapping(const mapping& __mapping, _Slices... __slices)
1632 { return __mdspan::__submdspan_mapping_impl(__mapping, __slices...); }
1633#endif // __glibcxx_submdspan
1634
1635 [[no_unique_address]] extents_type _M_extents{};
1636 };
1637
1638 namespace __mdspan
1639 {
1640 template<typename _Extents, typename... _Indices>
1641 constexpr typename _Extents::index_type
1642 __linear_index_right(const _Extents& __exts, _Indices... __indices)
1643 noexcept
1644 {
1645 using _IndexType = typename _Extents::index_type;
1646 array<_IndexType, sizeof...(__indices)> __ind_arr{__indices...};
1647 _IndexType __res = 0;
1648 if constexpr (sizeof...(__indices) > 0)
1649 {
1650 _IndexType __mult = 1;
1651 auto __update = [&, __pos = __exts.rank()](_IndexType) mutable
1652 {
1653 --__pos;
1654 _GLIBCXX_DEBUG_ASSERT(cmp_less(__ind_arr[__pos],
1655 __exts.extent(__pos)));
1656 __res += __ind_arr[__pos] * __mult;
1657 __mult *= __exts.extent(__pos);
1658 };
1659 (__update(__indices), ...);
1660 }
1661 return __res;
1662 }
1663 }
1664
1665 template<typename _Extents>
1666 class layout_right::mapping
1667 {
1668 public:
1669 using extents_type = _Extents;
1670 using index_type = typename extents_type::index_type;
1671 using size_type = typename extents_type::size_type;
1672 using rank_type = typename extents_type::rank_type;
1673 using layout_type = layout_right;
1674
1675 static_assert(__mdspan::__representable_size<extents_type, index_type>,
1676 "The size of extents_type must be representable as index_type");
1677
1678 constexpr
1679 mapping() noexcept = default;
1680
1681 constexpr
1682 mapping(const mapping&) noexcept = default;
1683
1684 constexpr
1685 mapping(const extents_type& __extents) noexcept
1686 : _M_extents(__extents)
1687 { __glibcxx_assert(__mdspan::__is_representable_extents(_M_extents)); }
1688
1689 template<typename _OExtents>
1690 requires is_constructible_v<extents_type, _OExtents>
1691 constexpr explicit(!is_convertible_v<_OExtents, extents_type>)
1692 mapping(const mapping<_OExtents>& __other) noexcept
1693 : mapping(__other.extents(), __mdspan::__internal_ctor{})
1694 { }
1695
1696 template<typename _OExtents>
1697 requires (extents_type::rank() <= 1)
1698 && is_constructible_v<extents_type, _OExtents>
1699 constexpr explicit(!is_convertible_v<_OExtents, extents_type>)
1700 mapping(const layout_left::mapping<_OExtents>& __other) noexcept
1701 : mapping(__other.extents(), __mdspan::__internal_ctor{})
1702 { }
1703
1704 template<typename _OExtents>
1705 requires is_constructible_v<extents_type, _OExtents>
1706 constexpr explicit(!(extents_type::rank() == 0
1707 && is_convertible_v<_OExtents, extents_type>))
1708 mapping(const layout_stride::mapping<_OExtents>& __other) noexcept
1709 : mapping(__other.extents(), __mdspan::__internal_ctor{})
1710 { __glibcxx_assert(*this == __other); }
1711
1712#if __glibcxx_padded_layouts
1713 template<typename _RightPaddedMapping>
1714 requires __mdspan::__is_right_padded_mapping<_RightPaddedMapping>
1715 && is_constructible_v<extents_type,
1716 typename _RightPaddedMapping::extents_type>
1717 constexpr
1718 explicit(!is_convertible_v<typename _RightPaddedMapping::extents_type,
1719 extents_type>)
1720 mapping(const _RightPaddedMapping& __other) noexcept
1721 : mapping(__other.extents(), __mdspan::__internal_ctor{})
1722 {
1723 constexpr size_t __rank = extents_type::rank();
1724 constexpr size_t __ostride_sta
1725 = __mdspan::__get_static_stride<_RightPaddedMapping>();
1726
1727 if constexpr (__rank > 1)
1728 {
1729 if constexpr (extents_type::static_extent(__rank - 1) != dynamic_extent
1730 && __ostride_sta != dynamic_extent)
1731 static_assert(extents_type::static_extent(__rank - 1)
1732 == __ostride_sta);
1733 else
1734 __glibcxx_assert(__other.stride(__rank - 2)
1735 == __other.extents().extent(__rank - 1));
1736 }
1737 }
1738#endif
1739
1740 constexpr mapping&
1741 operator=(const mapping&) noexcept = default;
1742
1743 constexpr const extents_type&
1744 extents() const noexcept { return _M_extents; }
1745
1746 constexpr index_type
1747 required_span_size() const noexcept
1748 { return __mdspan::__size(_M_extents); }
1749
1750 // _GLIBCXX_RESOLVE_LIB_DEFECTS
1751 // 4314. Missing move in mdspan layout mapping::operator()
1752 template<__mdspan::__valid_index_type<index_type>... _Indices>
1753 requires (sizeof...(_Indices) == extents_type::rank())
1754 constexpr index_type
1755 operator()(_Indices... __indices) const noexcept
1756 {
1757 return __mdspan::__linear_index_right(
1758 _M_extents, static_cast<index_type>(std::move(__indices))...);
1759 }
1760
1761 static constexpr bool
1762 is_always_unique() noexcept
1763 { return true; }
1764
1765 static constexpr bool
1766 is_always_exhaustive() noexcept
1767 { return true; }
1768
1769 static constexpr bool
1770 is_always_strided() noexcept
1771 { return true; }
1772
1773 static constexpr bool
1774 is_unique() noexcept
1775 { return true; }
1776
1777 static constexpr bool
1778 is_exhaustive() noexcept
1779 { return true; }
1780
1781 static constexpr bool
1782 is_strided() noexcept
1783 { return true; }
1784
1785 constexpr index_type
1786 stride(rank_type __i) const noexcept
1787 requires (extents_type::rank() > 0)
1788 {
1789 __glibcxx_assert(__i < extents_type::rank());
1790 return __mdspan::__rev_prod(_M_extents, __i);
1791 }
1792
1793 template<typename _OExtents>
1794 requires (extents_type::rank() == _OExtents::rank())
1795 friend constexpr bool
1796 operator==(const mapping& __self, const mapping<_OExtents>& __other)
1797 noexcept
1798 { return __self.extents() == __other.extents(); }
1799
1800 private:
1801 template<typename _OExtents>
1802 constexpr explicit
1803 mapping(const _OExtents& __oexts, __mdspan::__internal_ctor) noexcept
1804 : _M_extents(__oexts)
1805 {
1806 static_assert(__mdspan::__representable_size<_OExtents, index_type>,
1807 "The size of OtherExtents must be representable as index_type");
1808 __glibcxx_assert(__mdspan::__is_representable_extents(_M_extents));
1809 }
1810
1811#if __glibcxx_submdspan
1812 template<typename... _Slices>
1813 requires (extents_type::rank() == sizeof...(_Slices))
1814 friend constexpr auto
1815 submdspan_mapping(const mapping& __mapping, _Slices... __slices)
1816 { return __mdspan::__submdspan_mapping_impl(__mapping, __slices...); }
1817#endif // __glibcxx_submdspan
1818
1819 [[no_unique_address]] extents_type _M_extents{};
1820 };
1821
1822 namespace __mdspan
1823 {
1824 template<typename _Mp>
1825 concept __mapping_alike = requires
1826 {
1827 requires __is_extents<typename _Mp::extents_type>;
1828 { _Mp::is_always_strided() } -> same_as<bool>;
1829 { _Mp::is_always_exhaustive() } -> same_as<bool>;
1830 { _Mp::is_always_unique() } -> same_as<bool>;
1831 bool_constant<_Mp::is_always_strided()>::value;
1832 bool_constant<_Mp::is_always_exhaustive()>::value;
1833 bool_constant<_Mp::is_always_unique()>::value;
1834 };
1835
1836 template<typename _Mapping, typename... _Indices>
1837 constexpr typename _Mapping::index_type
1838 __linear_index_strides(const _Mapping& __m, _Indices... __indices)
1839 noexcept
1840 {
1841 using _IndexType = typename _Mapping::index_type;
1842 _IndexType __res = 0;
1843 if constexpr (sizeof...(__indices) > 0)
1844 {
1845 auto __update = [&, __pos = 0u](_IndexType __idx) mutable
1846 {
1847 _GLIBCXX_DEBUG_ASSERT(cmp_less(__idx,
1848 __m.extents().extent(__pos)));
1849 __res += __idx * __m.stride(__pos++);
1850 };
1851 (__update(__indices), ...);
1852 }
1853 return __res;
1854 }
1855 }
1856
1857 template<typename _Extents>
1858 class layout_stride::mapping
1859 {
1860 public:
1861 using extents_type = _Extents;
1862 using index_type = typename extents_type::index_type;
1863 using size_type = typename extents_type::size_type;
1864 using rank_type = typename extents_type::rank_type;
1865 using layout_type = layout_stride;
1866
1867 static_assert(__mdspan::__representable_size<extents_type, index_type>,
1868 "The size of extents_type must be representable as index_type");
1869
1870 constexpr
1871 mapping() noexcept
1872 {
1873 // The precondition is either statically asserted, or automatically
1874 // satisfied because dynamic extents are zero-initialized.
1875 size_t __stride = 1;
1876 for (size_t __i = extents_type::rank(); __i > 0; --__i)
1877 {
1878 _M_strides[__i - 1] = index_type(__stride);
1879 __stride *= size_t(_M_extents.extent(__i - 1));
1880 }
1881 }
1882
1883 constexpr
1884 mapping(const mapping&) noexcept = default;
1885
1886 template<typename _OIndexType>
1887 requires __mdspan::__valid_index_type<const _OIndexType&, index_type>
1888 constexpr
1889 mapping(const extents_type& __exts,
1890 span<_OIndexType, extents_type::rank()> __strides) noexcept
1891 : _M_extents(__exts)
1892 {
1893 for (size_t __i = 0; __i < extents_type::rank(); ++__i)
1894 _M_strides[__i] = index_type(as_const(__strides[__i]));
1895 }
1896
1897 template<typename _OIndexType>
1898 requires __mdspan::__valid_index_type<const _OIndexType&, index_type>
1899 constexpr
1900 mapping(const extents_type& __exts,
1901 const array<_OIndexType, extents_type::rank()>& __strides)
1902 noexcept
1903 : mapping(__exts,
1904 span<const _OIndexType, extents_type::rank()>(__strides))
1905 { }
1906
1907 template<__mdspan::__mapping_alike _StridedMapping>
1908 requires (is_constructible_v<extents_type,
1909 typename _StridedMapping::extents_type>
1910 && _StridedMapping::is_always_unique()
1911 && _StridedMapping::is_always_strided())
1912 constexpr explicit(!(
1913 is_convertible_v<typename _StridedMapping::extents_type, extents_type>
1914 && __mdspan::__standardized_mapping<_StridedMapping>))
1915 mapping(const _StridedMapping& __other) noexcept
1916 : _M_extents(__other.extents())
1917 {
1918 using _OIndexType = _StridedMapping::index_type;
1919 using _OExtents = _StridedMapping::extents_type;
1920
1921 __glibcxx_assert(__mdspan::__offset(__other) == 0);
1922 static_assert(__mdspan::__representable_size<_OExtents, index_type>,
1923 "The size of StridedMapping::extents_type must be representable as"
1924 " index_type");
1925 if constexpr (cmp_greater(__gnu_cxx::__int_traits<_OIndexType>::__max,
1926 __gnu_cxx::__int_traits<index_type>::__max))
1927 __glibcxx_assert(!cmp_less(
1928 __gnu_cxx::__int_traits<index_type>::__max,
1929 __other.required_span_size())
1930 && "other.required_span_size() must be representable"
1931 " as index_type");
1932 if constexpr (extents_type::rank() > 0)
1933 for (size_t __i = 0; __i < extents_type::rank(); ++__i)
1934 _M_strides[__i] = index_type(__other.stride(__i));
1935 }
1936
1937 constexpr mapping&
1938 operator=(const mapping&) noexcept = default;
1939
1940 constexpr const extents_type&
1941 extents() const noexcept { return _M_extents; }
1942
1943 constexpr array<index_type, extents_type::rank()>
1944 strides() const noexcept
1945 {
1946 array<index_type, extents_type::rank()> __ret;
1947 for (size_t __i = 0; __i < extents_type::rank(); ++__i)
1948 __ret[__i] = _M_strides[__i];
1949 return __ret;
1950 }
1951
1952 constexpr index_type
1953 required_span_size() const noexcept
1954 {
1955 if (__mdspan::__empty(_M_extents))
1956 return 0;
1957
1958 index_type __ret = 1;
1959 for (size_t __i = 0; __i < extents_type::rank(); ++__i)
1960 __ret += (_M_extents.extent(__i) - 1) * _M_strides[__i];
1961 return __ret;
1962 }
1963
1964 // _GLIBCXX_RESOLVE_LIB_DEFECTS
1965 // 4314. Missing move in mdspan layout mapping::operator()
1966 template<__mdspan::__valid_index_type<index_type>... _Indices>
1967 requires (sizeof...(_Indices) == extents_type::rank())
1968 constexpr index_type
1969 operator()(_Indices... __indices) const noexcept
1970 {
1971 return __mdspan::__linear_index_strides(*this,
1972 static_cast<index_type>(std::move(__indices))...);
1973 }
1974
1975 static constexpr bool
1976 is_always_unique() noexcept { return true; }
1977
1978 // _GLIBCXX_RESOLVE_LIB_DEFECTS
1979 // 4266. layout_stride::mapping should treat empty mappings as exhaustive
1980 static constexpr bool
1981 is_always_exhaustive() noexcept
1982 {
1983 return (_Extents::rank() == 0) || __mdspan::__contains_zero(
1984 __mdspan::__static_extents<extents_type>());
1985 }
1986
1987 static constexpr bool
1988 is_always_strided() noexcept { return true; }
1989
1990 static constexpr bool
1991 is_unique() noexcept { return true; }
1992
1993 // _GLIBCXX_RESOLVE_LIB_DEFECTS
1994 // 4266. layout_stride::mapping should treat empty mappings as exhaustive
1995 constexpr bool
1996 is_exhaustive() const noexcept
1997 {
1998 if constexpr (!is_always_exhaustive())
1999 {
2000 auto __size = __mdspan::__size(_M_extents);
2001 if(__size > 0)
2002 return __size == required_span_size();
2003 }
2004 return true;
2005 }
2006
2007 static constexpr bool
2008 is_strided() noexcept { return true; }
2009
2010 constexpr index_type
2011 stride(rank_type __r) const noexcept { return _M_strides[__r]; }
2012
2013 template<__mdspan::__mapping_alike _OMapping>
2014 requires ((extents_type::rank() == _OMapping::extents_type::rank())
2015 && _OMapping::is_always_strided())
2016 friend constexpr bool
2017 operator==(const mapping& __self, const _OMapping& __other) noexcept
2018 {
2019 if (__self.extents() != __other.extents())
2020 return false;
2021 if constexpr (extents_type::rank() > 0)
2022 for (size_t __i = 0; __i < extents_type::rank(); ++__i)
2023 if (!cmp_equal(__self.stride(__i), __other.stride(__i)))
2024 return false;
2025 return __mdspan::__offset(__other) == 0;
2026 }
2027
2028 private:
2029#if __glibcxx_submdspan
2030 template<typename... _Slices>
2031 requires (extents_type::rank() == sizeof...(_Slices))
2032 friend constexpr auto
2033 submdspan_mapping(const mapping& __mapping, _Slices... __slices)
2034 {
2035 if constexpr (sizeof...(_Slices) == 0)
2036 return submdspan_mapping_result{__mapping, 0};
2037 else
2038 {
2039 auto __offset = __mdspan::__suboffset(__mapping, __slices...);
2040 auto __sub_exts = __mdspan::__subextents(__mapping.extents(), __slices...);
2041 auto __sub_strides
2042 = __mdspan::__substrides<decltype(__sub_exts)>(__mapping, __slices...);
2043 return submdspan_mapping_result{
2044 layout_stride::mapping(__sub_exts, __sub_strides), __offset};
2045 }
2046 }
2047#endif
2048
2049 using _Strides = typename __array_traits<index_type,
2050 extents_type::rank()>::_Type;
2051 [[no_unique_address]] extents_type _M_extents;
2052 [[no_unique_address]] _Strides _M_strides;
2053 };
2054
2055#ifdef __glibcxx_padded_layouts
2056 namespace __mdspan
2057 {
2058 constexpr size_t
2059 __least_multiple(size_t __x, size_t __y)
2060 {
2061 if (__x <= 1)
2062 return __y;
2063 return (__y / __x + (__y % __x != 0)) * __x ;
2064 }
2065
2066 template<typename _IndexType>
2067 constexpr bool
2068 __is_representable_least_multiple(size_t __x, size_t __y)
2069 {
2070 constexpr auto __y_max = __gnu_cxx::__int_traits<_IndexType>::__max;
2071 if(std::cmp_greater(__y, __y_max))
2072 return false;
2073
2074 if(__x <= 1)
2075 return true;
2076
2077 auto __max_delta = __y_max - static_cast<_IndexType>(__y);
2078 auto __y_mod_x = __y % __x;
2079 auto __delta = (__y_mod_x == 0) ? size_t(0) : (__x - __y_mod_x);
2080 return std::cmp_less_equal(__delta, __max_delta);
2081 }
2082
2083 template<typename _Extents, size_t _PaddingValue, typename _LayoutTraits,
2084 size_t _Rank = _Extents::rank()>
2085 concept __valid_static_stride = (_Extents::rank() <= 1)
2086 || (_PaddingValue == dynamic_extent)
2087 || (_Extents::static_extent(_LayoutTraits::_S_ext_idx) == dynamic_extent)
2088 || (__is_representable_least_multiple<size_t>(
2089 _PaddingValue, _Extents::static_extent(_LayoutTraits::_S_ext_idx)));
2090
2091 template<size_t _PaddedStride, typename _Extents,
2092 typename _LayoutTraits>
2093 consteval bool
2094 __is_representable_padded_size()
2095 {
2096 using _IndexType = typename _Extents::index_type;
2097 auto __sta_exts = __static_extents<_Extents>(
2098 _LayoutTraits::_S_unpad_begin, _LayoutTraits::_S_unpad_end);
2099 size_t __max = __gnu_cxx::__int_traits<_IndexType>::__max;
2100 return __static_quotient(__sta_exts, __max / _PaddedStride) != 0;
2101 }
2102
2103 template<typename _Extents, size_t _PaddedStride, typename _LayoutTraits,
2104 size_t _Rank = _Extents::rank()>
2105 concept __valid_padded_size = (_Rank <= 1)
2106 || (_PaddedStride == dynamic_extent)
2107 || (!__all_static(__static_extents<_Extents>()))
2108 || (__contains_zero(__static_extents<_Extents>()))
2109 || (__is_representable_padded_size<_PaddedStride, _Extents,
2110 _LayoutTraits>());
2111
2112 template<typename _Extents, typename _Stride, typename... _Indices>
2113 constexpr typename _Extents::index_type
2114 __linear_index_leftpad(const _Extents& __exts, _Stride __stride,
2115 _Indices... __indices)
2116 {
2117 // i0 + stride*(i1 + extents.extent(1)*...)
2118 using _IndexType = typename _Extents::index_type;
2119 _IndexType __res = 0;
2120 if constexpr (sizeof...(__indices) > 0)
2121 {
2122 _IndexType __mult = 1;
2123
2124 auto __update_rest = [&, __pos = 1u](_IndexType __idx) mutable
2125 {
2126 __res += __idx * __mult;
2127 __mult *= __exts.extent(__pos);
2128 ++__pos;
2129 };
2130
2131 auto __update = [&](_IndexType __idx, auto... __rest)
2132 {
2133 __res += __idx;
2134 __mult = __stride.extent(0);
2135 (__update_rest(__rest), ...);
2136 };
2137 __update(__indices...);
2138 }
2139 return __res;
2140 }
2141
2142 template<typename _Extents, typename _Stride, typename... _Indices>
2143 constexpr typename _Extents::index_type
2144 __linear_index_rightpad(const _Extents& __exts, _Stride __stride,
2145 _Indices... __indices)
2146 {
2147 // i[n-1] + stride*(i[n-2] + extents.extent(n-2])*...)
2148 using _IndexType = typename _Extents::index_type;
2149 _IndexType __res = 0;
2150 if constexpr (sizeof...(__indices) > 0)
2151 {
2152 _IndexType __mult = 1;
2153 array<_IndexType, sizeof...(__indices)> __ind_arr{__indices...};
2154
2155 auto __update_rest = [&, __pos = __exts.rank()-1](_IndexType) mutable
2156 {
2157 --__pos;
2158 __res += __ind_arr[__pos] * __mult;
2159 __mult *= __exts.extent(__pos);
2160 };
2161
2162 auto __update = [&](_IndexType, auto... __rest)
2163 {
2164 __res += __ind_arr[__exts.rank() - 1];
2165 __mult = __stride.extent(0);
2166 (__update_rest(__rest), ...);
2167 };
2168 __update(__indices...);
2169 }
2170 return __res;
2171 }
2172
2173 template<size_t _Rank>
2174 struct _LeftPaddedLayoutTraits
2175 {
2176 using _LayoutSame = layout_left;
2177 using _LayoutOther = layout_right;
2178
2179 constexpr static const size_t _S_ext_idx = 0;
2180 constexpr static const size_t _S_stride_idx = 1;
2181 constexpr static const size_t _S_unpad_begin = 1;
2182 constexpr static const size_t _S_unpad_end = _Rank;
2183
2184 template<typename _IndexType, size_t _StaticStride, size_t..._Extents>
2185 constexpr static auto
2186 _S_make_padded_extent(
2187 extents<_IndexType, _StaticStride> __stride,
2188 const extents<_IndexType, _Extents...>& __exts)
2189 {
2190 auto __impl = [&]<size_t... _Is>(integer_sequence<size_t, _Is...>)
2191 {
2192 return extents<_IndexType, _StaticStride,
2193 (_Extents...[_Is + 1])...>{
2194 __stride.extent(0), __exts.extent(_Is + 1)...};
2195 };
2196 return __impl(make_index_sequence<sizeof...(_Extents) - 1>());
2197 }
2198 };
2199
2200 template<size_t _Rank>
2201 struct _RightPaddedLayoutTraits
2202 {
2203 using _LayoutSame = layout_right;
2204 using _LayoutOther = layout_left;
2205
2206 constexpr static size_t _S_ext_idx = _Rank - 1;
2207 constexpr static size_t _S_stride_idx = _Rank - 2;
2208 constexpr static size_t _S_unpad_begin = 0;
2209 constexpr static size_t _S_unpad_end = _Rank - 1;
2210
2211 template<typename _IndexType, size_t _StaticStride, size_t..._Extents>
2212 constexpr static auto
2213 _S_make_padded_extent(
2214 extents<_IndexType, _StaticStride> __stride,
2215 const extents<_IndexType, _Extents...>& __exts)
2216 {
2217 auto __impl = [&]<size_t... _Is>(integer_sequence<size_t, _Is...>)
2218 {
2219 return extents<_IndexType, (_Extents...[_Is])..., _StaticStride>{
2220 __exts.extent(_Is)..., __stride.extent(0)};
2221 };
2222 return __impl(make_index_sequence<sizeof...(_Extents) - 1>());
2223 }
2224 };
2225
2226 template<size_t _PaddingValue, typename _Extents, typename _LayoutTraits>
2227 class _PaddedStorage
2228 {
2229 using _LayoutSame = typename _LayoutTraits::_LayoutSame;
2230
2231 public:
2232 using _IndexType = typename _Extents::index_type;
2233 constexpr static size_t _S_rank = _Extents::rank();
2234
2235 // _GLIBCXX_RESOLVE_LIB_DEFECTS
2236 // 4372. Weaken Mandates: for dynamic padding values in padded layouts
2237 static_assert((_PaddingValue == dynamic_extent)
2238 || (cmp_less_equal(_PaddingValue,
2239 __gnu_cxx::__int_traits<_IndexType>::__max)),
2240 "padding_value must be representable as index_type");
2241
2242 static_assert(__representable_size<_Extents, _IndexType>,
2243 "The size of extents_type must be representable as index_type");
2244
2245 static_assert(__valid_static_stride<_Extents, _PaddingValue,
2246 _LayoutTraits>,
2247 "The padded stride must be representable as size_t");
2248
2249 static constexpr size_t _S_static_stride = [] consteval
2250 {
2251 constexpr size_t __rank = _Extents::rank();
2252 if constexpr (__rank <= 1)
2253 return 0;
2254 else
2255 {
2256 constexpr size_t __ext_idx = _LayoutTraits::_S_ext_idx;
2257 constexpr size_t __sta_ext = _Extents::static_extent(__ext_idx);
2258 if constexpr (__sta_ext == 0)
2259 return size_t(0);
2260 else if constexpr (_PaddingValue == dynamic_extent
2261 || __sta_ext == dynamic_extent)
2262 return dynamic_extent;
2263 else
2264 return __least_multiple(_PaddingValue, __sta_ext);
2265 }
2266 }();
2267
2268 static_assert(_S_static_stride == dynamic_extent
2269 || cmp_less_equal(_S_static_stride,
2270 __gnu_cxx::__int_traits<_IndexType>::__max),
2271 "Padded stride must be representable as index_type");
2272
2273 static_assert(__valid_padded_size<_Extents, _S_static_stride,
2274 _LayoutTraits>);
2275
2276 constexpr
2277 _PaddedStorage() noexcept
2278 {
2279 if constexpr (_S_rank > 1)
2280 if constexpr (_S_static_stride == dynamic_extent
2281 && _S_static_padextent() != dynamic_extent)
2282 _M_stride = _Stride{_S_static_padextent()};
2283 }
2284
2285 constexpr explicit
2286 _PaddedStorage(const _Extents& __exts)
2287 : _M_extents(__exts)
2288 {
2289 if constexpr (!__all_static(__static_extents<_Extents>()))
2290 __glibcxx_assert(__is_representable_extents(_M_extents));
2291
2292 if constexpr (_S_rank > 1)
2293 {
2294 _IndexType __stride;
2295 if constexpr (_PaddingValue == dynamic_extent)
2296 __stride = _M_padextent();
2297 else if constexpr (_S_static_padextent() != dynamic_extent)
2298 return;
2299 else
2300 {
2301 __glibcxx_assert(
2302 __is_representable_least_multiple<_IndexType>(
2303 _PaddingValue, _M_padextent()));
2304
2305 __stride = static_cast<_IndexType>(
2306 __least_multiple(_PaddingValue, _M_padextent()));
2307
2308 __glibcxx_assert(__is_representable_extents(
2309 _LayoutTraits::_S_make_padded_extent(
2310 std::dextents<_IndexType, 1>{__stride},
2311 _M_extents)));
2312 }
2313 _M_stride = _Stride{__stride};
2314 }
2315 }
2316
2317 constexpr explicit
2318 _PaddedStorage(const _Extents& __exts, _IndexType __pad)
2319 : _M_extents(__exts)
2320 {
2321 if constexpr (_PaddingValue != dynamic_extent)
2322 __glibcxx_assert(cmp_equal(_PaddingValue, __pad));
2323 if constexpr (_S_rank > 1 && _S_static_stride == dynamic_extent)
2324 {
2325 __glibcxx_assert(
2326 __is_representable_least_multiple<_IndexType>(
2327 __pad, _M_padextent()));
2328
2329 _M_stride = _Stride{static_cast<_IndexType>(
2330 __least_multiple(__pad, _M_padextent()))};
2331
2332 __glibcxx_assert(__is_representable_extents(
2333 _LayoutTraits::_S_make_padded_extent(
2334 _M_stride, _M_extents)));
2335 }
2336 }
2337
2338 template<typename _OExtents>
2339 constexpr explicit
2340 _PaddedStorage(
2341 const typename _LayoutSame::template mapping<_OExtents>& __other)
2342 : _PaddedStorage(_Extents(__other.extents()))
2343 {
2344 constexpr size_t __stride_idx = _LayoutTraits::_S_stride_idx;
2345 constexpr size_t __ext_idx = _LayoutTraits::_S_ext_idx;
2346 if constexpr (_S_rank > 1 && _PaddingValue != dynamic_extent)
2347 {
2348 static_assert(_S_static_stride == dynamic_extent
2349 || _OExtents::static_extent(__ext_idx) == dynamic_extent
2350 || _S_static_stride == _OExtents::static_extent(__ext_idx),
2351 "The padded stride must be compatible with other");
2352
2353 if constexpr (_S_static_stride == dynamic_extent
2354 || _OExtents::static_extent(__stride_idx) == dynamic_extent)
2355 __glibcxx_assert(std::cmp_equal(_M_padstride(),
2356 _M_padextent()));
2357 }
2358 }
2359
2360 template<typename _OExtents>
2361 constexpr explicit
2362 _PaddedStorage(const typename layout_stride::mapping<_OExtents>&
2363 __other)
2364 : _M_extents(__other.extents())
2365 {
2366 __glibcxx_assert(cmp_less_equal(__other.required_span_size(),
2368 ::__max));
2369
2370 constexpr size_t __stride_idx = _LayoutTraits::_S_stride_idx;
2371 if constexpr (_S_rank > 1)
2372 {
2373 if constexpr (_PaddingValue != dynamic_extent)
2374 __glibcxx_assert(cmp_equal(__other.stride(__stride_idx),
2375 _M_calc_padstride())
2376 && "The padded stride must be compatible with other");
2377 if constexpr (_S_static_stride == dynamic_extent)
2378 _M_stride = _Stride{__other.stride(__stride_idx)};
2379 }
2380 }
2381
2382 template<typename _SamePaddedMapping>
2383 constexpr explicit
2384 _PaddedStorage(_LayoutTraits::_LayoutSame,
2385 const _SamePaddedMapping& __other)
2386 : _M_extents(__other.extents())
2387 {
2388 if constexpr (_S_rank > 1)
2389 {
2390 static_assert(_PaddingValue == dynamic_extent
2391 || _SamePaddedMapping::padding_value == dynamic_extent
2392 || _PaddingValue == _SamePaddedMapping::padding_value,
2393 "If neither PaddingValue is dynamic_extent, then they must "
2394 "be equal");
2395
2396 constexpr size_t __stride_idx = _LayoutTraits::_S_stride_idx;
2397 if constexpr (_PaddingValue != dynamic_extent)
2398 __glibcxx_assert(cmp_equal(__other.stride(__stride_idx),
2399 _M_calc_padstride())
2400 && "The padded stride must be compatible with other");
2401 if constexpr (_S_static_stride == dynamic_extent)
2402 _M_stride = _Stride{__other.stride(__stride_idx)};
2403 }
2404 __glibcxx_assert(cmp_less_equal(__other.required_span_size(),
2405 __gnu_cxx::__int_traits<_IndexType>::__max));
2406 }
2407
2408 template<typename _OtherPaddedMapping>
2409 constexpr explicit
2410 _PaddedStorage(_LayoutTraits::_LayoutOther,
2411 const _OtherPaddedMapping& __other) noexcept
2412 : _M_extents(__other.extents())
2413 {
2414 __glibcxx_assert(cmp_less_equal(__other.required_span_size(),
2415 __gnu_cxx::__int_traits<_IndexType>::__max));
2416 }
2417
2418 static constexpr bool
2419 _M_is_always_exhaustive() noexcept
2420 {
2421 if constexpr (_S_rank <= 1)
2422 return true;
2423 else
2424 return _S_static_padextent() != dynamic_extent
2425 && _S_static_stride != dynamic_extent
2426 && _S_static_padextent() == _S_static_stride;
2427 }
2428
2429 constexpr bool
2430 _M_is_exhaustive() const noexcept
2431 {
2432 if constexpr (_M_is_always_exhaustive())
2433 return true;
2434 else
2435 return cmp_equal(_M_padextent(), _M_padstride());
2436 }
2437
2438 constexpr static size_t
2439 _S_static_padextent() noexcept
2440 { return _Extents::static_extent(_LayoutTraits::_S_ext_idx); }
2441
2442 constexpr _IndexType
2443 _M_padextent() const noexcept
2444 { return _M_extents.extent(_LayoutTraits::_S_ext_idx); }
2445
2446 constexpr _IndexType
2447 _M_calc_padstride() const noexcept
2448 {
2449 if constexpr (_S_static_stride != dynamic_extent)
2450 return _S_static_stride;
2451 else if constexpr (_PaddingValue != dynamic_extent)
2452 return __least_multiple(_PaddingValue, _M_padextent());
2453 else
2454 return _M_padextent();
2455 }
2456
2457 constexpr _IndexType
2458 _M_padstride() const noexcept
2459 { return _M_stride.extent(0); }
2460
2461 constexpr _IndexType
2462 _M_required_span_size() const noexcept
2463 {
2464 if constexpr (_S_rank == 0)
2465 return 1;
2466 else if (__mdspan::__empty(_M_extents))
2467 return 0;
2468 else
2469 {
2470 size_t __stride = static_cast<size_t>(_M_padstride());
2471 size_t __prod_rest = __mdspan::__fwd_prod(_M_extents,
2472 _LayoutTraits::_S_unpad_begin, _LayoutTraits::_S_unpad_end);
2473 size_t __delta = _M_padstride() - _M_padextent();
2474 return static_cast<_IndexType>(__stride * __prod_rest - __delta);
2475 }
2476 }
2477
2478 template<typename _SamePaddedMapping>
2479 constexpr bool
2480 _M_equal(const _SamePaddedMapping& __other) const noexcept
2481 {
2482 return _M_extents == __other.extents()
2483 && (_S_rank < 2
2484 || cmp_equal(_M_stride.extent(0),
2485 __other.stride(_LayoutTraits::_S_stride_idx)));
2486 }
2487
2488 using _Stride = std::extents<_IndexType, _S_static_stride>;
2489 [[no_unique_address]] _Stride _M_stride;
2490 [[no_unique_address]] _Extents _M_extents;
2491 };
2492 }
2493
2494 template<size_t _PaddingValue>
2495 template<typename _Extents>
2496 class layout_left_padded<_PaddingValue>::mapping
2497 {
2498 public:
2499 static constexpr size_t padding_value = _PaddingValue;
2500
2501 using extents_type = _Extents;
2502 using index_type = typename extents_type::index_type;
2503 using size_type = typename extents_type::size_type;
2504 using rank_type = typename extents_type::rank_type;
2505 using layout_type = layout_left_padded<padding_value>;
2506
2507 private:
2508 static constexpr size_t _S_rank = extents_type::rank();
2509 using _PaddedStorage = __mdspan::_PaddedStorage<_PaddingValue,
2510 _Extents, __mdspan::_LeftPaddedLayoutTraits<_S_rank>>;
2511 [[no_unique_address]] _PaddedStorage _M_storage;
2512
2513 consteval friend size_t
2514 __mdspan::__get_static_stride<mapping>();
2515
2516 constexpr index_type
2517 _M_extent(size_t __r) const noexcept
2518 { return _M_storage._M_extents.extent(__r); }
2519
2520 constexpr index_type
2521 _M_padstride() const noexcept
2522 { return _M_storage._M_stride.extent(0); }
2523
2524 public:
2525 constexpr
2526 mapping() noexcept
2527 { }
2528
2529 constexpr
2530 mapping(const mapping&) noexcept = default;
2531
2532 constexpr
2533 mapping(const extents_type& __exts)
2534 : _M_storage(__exts)
2535 { }
2536
2537 template<__mdspan::__valid_index_type<index_type> _OIndexType>
2538 constexpr
2539 mapping(const extents_type& __exts, _OIndexType __pad)
2540 : _M_storage(__exts,
2541 __mdspan::__index_type_cast<index_type>(std::move(__pad)))
2542 { }
2543
2544 template<typename _OExtents>
2545 requires is_constructible_v<extents_type, _OExtents>
2546 constexpr explicit(!is_convertible_v<_OExtents, extents_type>)
2547 mapping(const layout_left::mapping<_OExtents>& __other)
2548 : _M_storage(__other)
2549 { }
2550
2551 template<typename _OExtents>
2552 requires is_constructible_v<_OExtents, extents_type>
2553 constexpr explicit(!(_OExtents::rank() == 0
2554 && is_convertible_v<_OExtents, extents_type>))
2555 mapping(const typename layout_stride::mapping<_OExtents>& __other)
2556 : _M_storage(__other)
2557 { __glibcxx_assert(*this == __other); }
2558
2559 template<typename _LeftPaddedMapping>
2560 requires __mdspan::__is_left_padded_mapping<_LeftPaddedMapping>
2561 && is_constructible_v<extents_type,
2562 typename _LeftPaddedMapping::extents_type>
2563 constexpr explicit(
2564 !is_convertible_v<typename _LeftPaddedMapping::extents_type,
2565 extents_type>
2566 || _S_rank > 1 && (padding_value != dynamic_extent
2567 || _LeftPaddedMapping::padding_value == dynamic_extent))
2568 mapping(const _LeftPaddedMapping& __other)
2569 : _M_storage(layout_left{}, __other)
2570 { }
2571
2572 template<typename _RightPaddedMapping>
2573 requires (__mdspan::__is_right_padded_mapping<_RightPaddedMapping>
2574 || __mdspan::__mapping_of<layout_right, _RightPaddedMapping>)
2575 && (_S_rank <= 1)
2576 && is_constructible_v<extents_type,
2577 typename _RightPaddedMapping::extents_type>
2578 constexpr explicit(!is_convertible_v<
2579 typename _RightPaddedMapping::extents_type, extents_type>)
2580 mapping(const _RightPaddedMapping& __other) noexcept
2581 : _M_storage(layout_right{}, __other)
2582 { }
2583
2584 constexpr mapping&
2585 operator=(const mapping&) noexcept = default;
2586
2587 constexpr const extents_type&
2588 extents() const noexcept { return _M_storage._M_extents; }
2589
2590 constexpr array<index_type, _S_rank>
2591 strides() const noexcept
2592 {
2593 array<index_type, _S_rank> __ret;
2594 if constexpr (_S_rank > 0)
2595 __ret[0] = 1;
2596 if constexpr (_S_rank > 1)
2597 __ret[1] = _M_padstride();
2598 if constexpr (_S_rank > 2)
2599 for(size_t __i = 2; __i < _S_rank; ++__i)
2600 __ret[__i] = __ret[__i - 1] * _M_extent(__i - 1);
2601 return __ret;
2602 }
2603
2604 constexpr index_type
2605 required_span_size() const noexcept
2606 { return _M_storage._M_required_span_size(); }
2607
2608 // _GLIBCXX_RESOLVE_LIB_DEFECTS
2609 // 4314. Missing move in mdspan layout mapping::operator()
2610 template<__mdspan::__valid_index_type<index_type>... _Indices>
2611 requires (sizeof...(_Indices) == _S_rank)
2612 constexpr index_type
2613 operator()(_Indices... __indices) const noexcept
2614 {
2615 return __mdspan::__linear_index_leftpad(
2616 extents(), _M_storage._M_stride,
2617 static_cast<index_type>(std::move(__indices))...);
2618 }
2619
2620 static constexpr bool
2621 is_always_exhaustive() noexcept
2622 { return _PaddedStorage::_M_is_always_exhaustive(); }
2623
2624 constexpr bool
2625 is_exhaustive() const noexcept
2626 { return _M_storage._M_is_exhaustive(); }
2627
2628 static constexpr bool
2629 is_always_unique() noexcept { return true; }
2630
2631 static constexpr bool
2632 is_unique() noexcept { return true; }
2633
2634 static constexpr bool
2635 is_always_strided() noexcept { return true; }
2636
2637 static constexpr bool
2638 is_strided() noexcept { return true; }
2639
2640 constexpr index_type
2641 stride(rank_type __r) const noexcept
2642 {
2643 __glibcxx_assert(__r < _S_rank);
2644 if (__r == 0)
2645 return 1;
2646 else
2647 return static_cast<index_type>(
2648 static_cast<size_t>(_M_padstride()) *
2649 static_cast<size_t>(__mdspan::__fwd_prod(extents(), 1, __r)));
2650 }
2651
2652 template<typename _LeftpadMapping>
2653 requires(__mdspan::__is_left_padded_mapping<_LeftpadMapping>
2654 && _LeftpadMapping::extents_type::rank() == _S_rank)
2655 friend constexpr bool
2656 operator==(const mapping& __self, const _LeftpadMapping& __other)
2657 noexcept
2658 { return __self._M_storage._M_equal(__other); }
2659
2660 private:
2661#if __glibcxx_submdspan
2662 template<typename... _Slices>
2663 requires (extents_type::rank() == sizeof...(_Slices))
2664 friend constexpr auto
2665 submdspan_mapping(const mapping& __mapping, _Slices... __slices)
2666 { return __mdspan::__submdspan_mapping_impl(__mapping, __slices...); }
2667#endif // __glibcxx_submdspan
2668 };
2669
2670 template<size_t _PaddingValue>
2671 template<typename _Extents>
2672 class layout_right_padded<_PaddingValue>::mapping {
2673 public:
2674 static constexpr size_t padding_value = _PaddingValue;
2675 using extents_type = _Extents;
2676 using index_type = typename extents_type::index_type;
2677 using size_type = typename extents_type::size_type;
2678 using rank_type = typename extents_type::rank_type;
2679 using layout_type = layout_right_padded<_PaddingValue>;
2680
2681 private:
2682 static constexpr size_t _S_rank = extents_type::rank();
2683 using _PaddedStorage = __mdspan::_PaddedStorage<_PaddingValue,
2684 _Extents, __mdspan::_RightPaddedLayoutTraits<_S_rank>>;
2685 [[no_unique_address]] _PaddedStorage _M_storage;
2686
2687 consteval friend size_t
2688 __mdspan::__get_static_stride<mapping>();
2689
2690 constexpr index_type
2691 _M_extent(size_t __r) const noexcept
2692 { return _M_storage._M_extents.extent(__r); }
2693
2694 constexpr index_type
2695 _M_padstride() const noexcept
2696 { return _M_storage._M_stride.extent(0); }
2697
2698 public:
2699 constexpr
2700 mapping() noexcept
2701 { }
2702
2703 constexpr
2704 mapping(const mapping&) noexcept = default;
2705
2706 constexpr
2707 mapping(const extents_type& __exts)
2708 : _M_storage(__exts)
2709 { }
2710
2711 template<__mdspan::__valid_index_type<index_type> _OIndexType>
2712 constexpr
2713 mapping(const extents_type& __exts, _OIndexType __pad)
2714 : _M_storage(__exts,
2715 __mdspan::__index_type_cast<index_type>(std::move(__pad)))
2716 { }
2717
2718 template<typename _OExtents>
2719 requires is_constructible_v<extents_type, _OExtents>
2720 constexpr explicit(!is_convertible_v<_OExtents, extents_type>)
2721 mapping(const layout_right::mapping<_OExtents>& __other)
2722 : _M_storage(__other)
2723 { }
2724
2725 template<typename _OExtents>
2726 requires is_constructible_v<_OExtents, extents_type>
2727 constexpr explicit(!(_OExtents::rank() == 0
2728 && is_convertible_v<_OExtents, extents_type>))
2729 mapping(const typename layout_stride::mapping<_OExtents>& __other)
2730 : _M_storage(__other)
2731 { __glibcxx_assert(*this == __other); }
2732
2733 template<typename _RightPaddedMapping>
2734 requires __mdspan::__is_right_padded_mapping<_RightPaddedMapping>
2735 && is_constructible_v<extents_type,
2736 typename _RightPaddedMapping::extents_type>
2737 constexpr explicit(
2738 !is_convertible_v<typename _RightPaddedMapping::extents_type,
2739 extents_type>
2740 || _S_rank > 1 && (padding_value != dynamic_extent
2741 || _RightPaddedMapping::padding_value == dynamic_extent))
2742 mapping(const _RightPaddedMapping& __other)
2743 : _M_storage(layout_right{}, __other)
2744 { }
2745
2746 template<typename _LeftPaddedMapping>
2747 requires (__mdspan::__is_left_padded_mapping<_LeftPaddedMapping>
2748 || __mdspan::__mapping_of<layout_left, _LeftPaddedMapping>)
2749 && (_S_rank <= 1)
2750 && is_constructible_v<extents_type,
2751 typename _LeftPaddedMapping::extents_type>
2752 constexpr explicit(!is_convertible_v<
2753 typename _LeftPaddedMapping::extents_type, extents_type>)
2754 mapping(const _LeftPaddedMapping& __other) noexcept
2755 : _M_storage(layout_left{}, __other)
2756 { }
2757
2758 constexpr mapping& operator=(const mapping&) noexcept = default;
2759
2760 constexpr const extents_type&
2761 extents() const noexcept { return _M_storage._M_extents; }
2762
2763 constexpr array<index_type, _S_rank>
2764 strides() const noexcept
2765 {
2766 array<index_type, _S_rank> __ret;
2767 if constexpr (_S_rank > 0)
2768 __ret[_S_rank - 1] = 1;
2769 if constexpr (_S_rank > 1)
2770 __ret[_S_rank - 2] = _M_padstride();
2771 if constexpr (_S_rank > 2)
2772 for(size_t __i = _S_rank - 2; __i > 0; --__i)
2773 __ret[__i - 1] = __ret[__i] * _M_extent(__i);
2774 return __ret;
2775 }
2776
2777 constexpr index_type
2778 required_span_size() const noexcept
2779 { return _M_storage._M_required_span_size(); }
2780
2781 // _GLIBCXX_RESOLVE_LIB_DEFECTS
2782 // 4314. Missing move in mdspan layout mapping::operator()
2783 template<__mdspan::__valid_index_type<index_type>... _Indices>
2784 requires (sizeof...(_Indices) == _S_rank)
2785 constexpr index_type
2786 operator()(_Indices... __indices) const noexcept
2787 {
2788 return __mdspan::__linear_index_rightpad(
2789 extents(), _M_storage._M_stride,
2790 static_cast<index_type>(std::move(__indices))...);
2791 }
2792
2793 static constexpr bool
2794 is_always_exhaustive() noexcept
2795 { return _PaddedStorage::_M_is_always_exhaustive(); }
2796
2797 constexpr bool
2798 is_exhaustive() const noexcept
2799 { return _M_storage._M_is_exhaustive(); }
2800
2801 static constexpr bool
2802 is_always_unique() noexcept { return true; }
2803
2804 static constexpr bool
2805 is_unique() noexcept { return true; }
2806
2807 static constexpr bool
2808 is_always_strided() noexcept { return true; }
2809
2810 static constexpr bool
2811 is_strided() noexcept { return true; }
2812
2813 constexpr index_type
2814 stride(rank_type __r) const noexcept
2815 {
2816 __glibcxx_assert(__r < _S_rank);
2817 if constexpr (_S_rank <= 1)
2818 return 1;
2819 else if (__r == _S_rank - 1)
2820 return 1;
2821 else if (__r == _S_rank - 2)
2822 return _M_padstride();
2823 else
2824 return static_cast<index_type>(
2825 static_cast<size_t>(_M_padstride()) *
2826 static_cast<size_t>(__mdspan::__fwd_prod(
2827 extents(), __r + 1, _S_rank - 1)));
2828 }
2829
2830 template<typename _RightPaddedMapping>
2831 requires(__mdspan::__is_right_padded_mapping<_RightPaddedMapping>
2832 && _RightPaddedMapping::extents_type::rank() == _S_rank)
2833 friend constexpr bool
2834 operator==(const mapping& __self, const _RightPaddedMapping& __other)
2835 noexcept
2836 { return __self._M_storage._M_equal(__other); }
2837
2838#if __glibcxx_submdspan
2839 private:
2840 template<typename... _Slices>
2841 requires (extents_type::rank() == sizeof...(_Slices))
2842 friend constexpr auto
2843 submdspan_mapping(const mapping& __mapping, _Slices... __slices)
2844 { return __mdspan::__submdspan_mapping_impl(__mapping, __slices...); }
2845#endif // __glibcxx_submdspan
2846 };
2847#endif // __glibcxx_padded_layouts
2848
2849 template<typename _ElementType>
2850 struct default_accessor
2851 {
2852 static_assert(!is_array_v<_ElementType>,
2853 "ElementType must not be an array type");
2854 static_assert(!is_abstract_v<_ElementType>,
2855 "ElementType must not be an abstract class type");
2856
2857 using offset_policy = default_accessor;
2858 using element_type = _ElementType;
2859 using reference = element_type&;
2860 using data_handle_type = element_type*;
2861
2862 constexpr
2863 default_accessor() noexcept = default;
2864
2865 template<typename _OElementType>
2866 requires is_convertible_v<_OElementType(*)[], element_type(*)[]>
2867 constexpr
2868 default_accessor(default_accessor<_OElementType>) noexcept
2869 { }
2870
2871 constexpr reference
2872 access(data_handle_type __p, size_t __i) const noexcept
2873 { return __p[__i]; }
2874
2875 constexpr data_handle_type
2876 offset(data_handle_type __p, size_t __i) const noexcept
2877 { return __p + __i; }
2878 };
2879
2880#ifdef __glibcxx_aligned_accessor
2881 template<typename _ElementType, size_t _ByteAlignment>
2882 struct aligned_accessor
2883 {
2884 static_assert(has_single_bit(_ByteAlignment),
2885 "ByteAlignment must be a power of two");
2886 static_assert(_ByteAlignment >= alignof(_ElementType));
2887
2888 using offset_policy = default_accessor<_ElementType>;
2889 using element_type = _ElementType;
2890 using reference = element_type&;
2891 using data_handle_type = element_type*;
2892
2893 static constexpr size_t byte_alignment = _ByteAlignment;
2894
2895 constexpr
2896 aligned_accessor() noexcept = default;
2897
2898 template<typename _OElementType, size_t _OByteAlignment>
2899 requires (_OByteAlignment >= byte_alignment)
2900 && is_convertible_v<_OElementType(*)[], element_type(*)[]>
2901 constexpr
2902 aligned_accessor(aligned_accessor<_OElementType, _OByteAlignment>)
2903 noexcept
2904 { }
2905
2906 template<typename _OElementType>
2907 requires is_convertible_v<_OElementType(*)[], element_type(*)[]>
2908 constexpr explicit
2909 aligned_accessor(default_accessor<_OElementType>) noexcept
2910 { }
2911
2912 template<typename _OElementType>
2913 requires is_convertible_v<element_type(*)[], _OElementType(*)[]>
2914 constexpr
2915 operator default_accessor<_OElementType>() const noexcept
2916 { return {}; }
2917
2918 constexpr reference
2919 access(data_handle_type __p, size_t __i) const noexcept
2920 { return std::assume_aligned<byte_alignment>(__p)[__i]; }
2921
2922 constexpr typename offset_policy::data_handle_type
2923 offset(data_handle_type __p, size_t __i) const noexcept
2924 { return std::assume_aligned<byte_alignment>(__p) + __i; }
2925 };
2926#endif
2927
2928 template<typename _ElementType, typename _Extents,
2929 typename _LayoutPolicy = layout_right,
2930 typename _AccessorPolicy = default_accessor<_ElementType>>
2931 class mdspan
2932 {
2933 static_assert(!is_array_v<_ElementType>,
2934 "ElementType must not be an array type");
2935 static_assert(!is_abstract_v<_ElementType>,
2936 "ElementType must not be an abstract class type");
2937 static_assert(__mdspan::__is_extents<_Extents>,
2938 "Extents must be a specialization of std::extents");
2939 static_assert(is_same_v<_ElementType,
2940 typename _AccessorPolicy::element_type>);
2941
2942 public:
2943 using extents_type = _Extents;
2944 using layout_type = _LayoutPolicy;
2945 using accessor_type = _AccessorPolicy;
2946 using mapping_type = typename layout_type::template mapping<extents_type>;
2947 using element_type = _ElementType;
2948 using value_type = remove_cv_t<element_type>;
2949 using index_type = typename extents_type::index_type;
2950 using size_type = typename extents_type::size_type;
2951 using rank_type = typename extents_type::rank_type;
2952 using data_handle_type = typename accessor_type::data_handle_type;
2953 using reference = typename accessor_type::reference;
2954
2955 static constexpr rank_type
2956 rank() noexcept { return extents_type::rank(); }
2957
2958 static constexpr rank_type
2959 rank_dynamic() noexcept { return extents_type::rank_dynamic(); }
2960
2961 static constexpr size_t
2962 static_extent(rank_type __r) noexcept
2963 { return extents_type::static_extent(__r); }
2964
2965 constexpr index_type
2966 extent(rank_type __r) const noexcept { return extents().extent(__r); }
2967
2968 constexpr
2969 mdspan()
2970 requires (rank_dynamic() > 0)
2971 && is_default_constructible_v<data_handle_type>
2972 && is_default_constructible_v<mapping_type>
2973 && is_default_constructible_v<accessor_type> = default;
2974
2975 constexpr
2976 mdspan(const mdspan& __other) = default;
2977
2978 constexpr
2979 mdspan(mdspan&& __other) = default;
2980
2981 template<__mdspan::__valid_index_type<index_type>... _OIndexTypes>
2982 requires (sizeof...(_OIndexTypes) == rank()
2983 || sizeof...(_OIndexTypes) == rank_dynamic())
2984 && is_constructible_v<mapping_type, extents_type>
2985 && is_default_constructible_v<accessor_type>
2986 constexpr explicit
2987 mdspan(data_handle_type __handle, _OIndexTypes... __exts)
2988 : _M_accessor(),
2989 _M_mapping(_Extents(static_cast<index_type>(std::move(__exts))...)),
2990 _M_handle(std::move(__handle))
2991 { }
2992
2993 template<typename _OIndexType, size_t _Nm>
2994 requires __mdspan::__valid_index_type<const _OIndexType&, index_type>
2995 && (_Nm == rank() || _Nm == rank_dynamic())
2996 && is_constructible_v<mapping_type, extents_type>
2997 && is_default_constructible_v<accessor_type>
2998 constexpr explicit(_Nm != rank_dynamic())
2999 mdspan(data_handle_type __handle, span<_OIndexType, _Nm> __exts)
3000 : _M_accessor(), _M_mapping(extents_type(__exts)),
3001 _M_handle(std::move(__handle))
3002 { }
3003
3004 template<typename _OIndexType, size_t _Nm>
3005 requires __mdspan::__valid_index_type<const _OIndexType&, index_type>
3006 && (_Nm == rank() || _Nm == rank_dynamic())
3007 && is_constructible_v<mapping_type, extents_type>
3008 && is_default_constructible_v<accessor_type>
3009 constexpr explicit(_Nm != rank_dynamic())
3010 mdspan(data_handle_type __handle, const array<_OIndexType, _Nm>& __exts)
3011 : _M_accessor(), _M_mapping(extents_type(__exts)),
3012 _M_handle(std::move(__handle))
3013 { }
3014
3015 constexpr
3016 mdspan(data_handle_type __handle, const extents_type& __exts)
3017 requires is_constructible_v<mapping_type, const extents_type&>
3018 && is_default_constructible_v<accessor_type>
3019 : _M_accessor(), _M_mapping(__exts), _M_handle(std::move(__handle))
3020 { }
3021
3022 constexpr
3023 mdspan(data_handle_type __handle, const mapping_type& __mapping)
3024 requires is_default_constructible_v<accessor_type>
3025 : _M_accessor(), _M_mapping(__mapping), _M_handle(std::move(__handle))
3026 { }
3027
3028 constexpr
3029 mdspan(data_handle_type __handle, const mapping_type& __mapping,
3030 const accessor_type& __accessor)
3031 : _M_accessor(__accessor), _M_mapping(__mapping),
3032 _M_handle(std::move(__handle))
3033 { }
3034
3035 template<typename _OElementType, typename _OExtents, typename _OLayout,
3036 typename _OAccessor>
3037 requires is_constructible_v<mapping_type,
3038 const typename _OLayout::template mapping<_OExtents>&>
3039 && is_constructible_v<accessor_type, const _OAccessor&>
3040 constexpr explicit(!is_convertible_v<
3041 const typename _OLayout::template mapping<_OExtents>&, mapping_type>
3042 || !is_convertible_v<const _OAccessor&, accessor_type>)
3043 mdspan(const mdspan<_OElementType, _OExtents, _OLayout, _OAccessor>&
3044 __other)
3045 : _M_accessor(__other.accessor()), _M_mapping(__other.mapping()),
3046 _M_handle(__other.data_handle())
3047 {
3048 static_assert(is_constructible_v<data_handle_type,
3049 const typename _OAccessor::data_handle_type&>);
3050 static_assert(is_constructible_v<extents_type, _OExtents>);
3051 }
3052
3053 constexpr mdspan&
3054 operator=(const mdspan& __other) = default;
3055
3056 constexpr mdspan&
3057 operator=(mdspan&& __other) = default;
3058
3059 template<__mdspan::__valid_index_type<index_type>... _OIndexTypes>
3060 requires (sizeof...(_OIndexTypes) == rank())
3061 constexpr reference
3062 operator[](_OIndexTypes... __indices) const
3063 {
3064 if constexpr (rank() == 0)
3065 return _M_accessor.access(_M_handle, _M_mapping());
3066 else if constexpr (!(is_integral_v<_OIndexTypes> && ...))
3067 return operator[](__index_int_t<_OIndexTypes>(std::move(__indices))...);
3068 else
3069 {
3070 auto __is_valid_index = [&]<typename _OIntType>(size_t __dim, _OIntType __index)
3071 {
3072 if constexpr (is_signed_v<_OIntType>)
3073 if (__index < 0)
3074 return false;
3075 return std::cmp_less(__index, extents().extent(__dim));
3076 };
3077 auto __is_multi_index = [&]<size_t... _Counts>(index_sequence<_Counts...>)
3078 { return (__is_valid_index(_Counts, __indices) && ...); };
3079
3080 __glibcxx_assert(__is_multi_index(make_index_sequence<rank()>()));
3081 auto __index = _M_mapping(static_cast<index_type>(__indices)...);
3082 return _M_accessor.access(_M_handle, __index);
3083 }
3084 }
3085
3086 template<typename _OIndexType>
3087 requires __mdspan::__valid_index_type<const _OIndexType&, index_type>
3088 constexpr reference
3089 operator[](span<_OIndexType, rank()> __indices) const
3090 {
3091 auto __call = [&]<size_t... _Counts>(index_sequence<_Counts...>)
3092 -> reference
3093 {
3094 return operator[](
3095 __index_int_t<_OIndexType>(as_const(__indices[_Counts]))...);
3096 };
3097 return __call(make_index_sequence<rank()>());
3098 }
3099
3100 template<typename _OIndexType>
3101 requires __mdspan::__valid_index_type<const _OIndexType&, index_type>
3102 constexpr reference
3103 operator[](const array<_OIndexType, rank()>& __indices) const
3104 { return operator[](span<const _OIndexType, rank()>(__indices)); }
3105
3106#if __cplusplus > 202302L
3107 template<__mdspan::__valid_index_type<index_type>... _OIndexTypes>
3108 requires (sizeof...(_OIndexTypes) == rank())
3109 constexpr reference
3110 at(_OIndexTypes... __indices) const
3111 {
3112 if constexpr (rank() == 0)
3113 return _M_accessor.access(_M_handle, _M_mapping());
3114 else if constexpr (!(is_integral_v<_OIndexTypes> && ...))
3115 return at(__index_int_t<_OIndexTypes>(std::move(__indices))...);
3116 else
3117 {
3118 auto __check_bound = [&]<typename _OIntType>(size_t __dim, _OIntType __index)
3119 {
3120 if constexpr (is_signed_v<_OIntType>)
3121 if (__index < 0)
3122 std::__throw_out_of_range_fmt(
3123 __N("mdspan::at: %zuth index is negative"), __dim);
3124
3125 const auto __ext = extents().extent(__dim);
3126 if (std::cmp_greater_equal(__index, __ext))
3127 std::__throw_out_of_range_fmt(
3128 __N("mdspan::at: %zuth index (which is %zu)"
3129 " >= extent(%zu) (which is %zu)"),
3130 __dim, size_t(__index), __dim, size_t(__ext));
3131 };
3132 auto __check_bounds = [&]<size_t... _Counts>(index_sequence<_Counts...>)
3133 { (__check_bound(_Counts, __indices), ...); };
3134
3135 __check_bounds(make_index_sequence<rank()>());
3136 auto __index = _M_mapping(static_cast<index_type>(__indices)...);
3137 return _M_accessor.access(_M_handle, __index);
3138 }
3139 }
3140
3141 template<typename _OIndexType>
3142 requires __mdspan::__valid_index_type<const _OIndexType&, index_type>
3143 constexpr reference
3144 at(span<_OIndexType, rank()> __indices) const
3145 {
3146 auto __call = [&]<size_t... _Counts>(index_sequence<_Counts...>)
3147 -> reference
3148 {
3149 return at(
3150 __index_int_t<_OIndexType>(as_const(__indices[_Counts]))...);
3151 };
3152 return __call(make_index_sequence<rank()>());
3153 }
3154
3155 template<typename _OIndexType>
3156 requires __mdspan::__valid_index_type<const _OIndexType&, index_type>
3157 constexpr reference
3158 at(const array<_OIndexType, rank()>& __indices) const
3159 { return at(span<const _OIndexType, rank()>(__indices)); }
3160#endif // C++26
3161
3162 constexpr size_type
3163 size() const noexcept
3164 {
3165 __glibcxx_assert(cmp_less_equal(_M_mapping.required_span_size(),
3167 ::__max));
3168 return size_type(__mdspan::__size(extents()));
3169 }
3170
3171 [[nodiscard]]
3172 constexpr bool
3173 empty() const noexcept
3174 { return __mdspan::__empty(extents()); }
3175
3176 friend constexpr void
3177 swap(mdspan& __x, mdspan& __y) noexcept
3178 {
3179 using std::swap;
3180 swap(__x._M_mapping, __y._M_mapping);
3181 swap(__x._M_accessor, __y._M_accessor);
3182 swap(__x._M_handle, __y._M_handle);
3183 }
3184
3185 constexpr const extents_type&
3186 extents() const noexcept { return _M_mapping.extents(); }
3187
3188 constexpr const data_handle_type&
3189 data_handle() const noexcept { return _M_handle; }
3190
3191 constexpr const mapping_type&
3192 mapping() const noexcept { return _M_mapping; }
3193
3194 constexpr const accessor_type&
3195 accessor() const noexcept { return _M_accessor; }
3196
3197 // Strengthened noexcept for all `is_*` methods.
3198
3199 static constexpr bool
3200 is_always_unique() noexcept(noexcept(mapping_type::is_always_unique()))
3201 { return mapping_type::is_always_unique(); }
3202
3203 static constexpr bool
3204 is_always_exhaustive()
3205 noexcept(noexcept(mapping_type::is_always_exhaustive()))
3206 { return mapping_type::is_always_exhaustive(); }
3207
3208 static constexpr bool
3209 is_always_strided()
3210 noexcept(noexcept(mapping_type::is_always_strided()))
3211 { return mapping_type::is_always_strided(); }
3212
3213 constexpr bool
3214 is_unique() const noexcept(noexcept(_M_mapping.is_unique()))
3215 { return _M_mapping.is_unique(); }
3216
3217 constexpr bool
3218 is_exhaustive() const noexcept(noexcept(_M_mapping.is_exhaustive()))
3219 { return _M_mapping.is_exhaustive(); }
3220
3221 constexpr bool
3222 is_strided() const noexcept(noexcept(_M_mapping.is_strided()))
3223 { return _M_mapping.is_strided(); }
3224
3225 constexpr index_type
3226 stride(rank_type __r) const { return _M_mapping.stride(__r); }
3227
3228 private:
3229 template<typename _OIndexType>
3230 using __index_int_t = std::__conditional_t<
3231 is_integral_v<_OIndexType>, _OIndexType, index_type>;
3232
3233 [[no_unique_address]] accessor_type _M_accessor = accessor_type();
3234 [[no_unique_address]] mapping_type _M_mapping = mapping_type();
3235 [[no_unique_address]] data_handle_type _M_handle = data_handle_type();
3236 };
3237
3238 template<typename _CArray>
3239 requires is_array_v<_CArray> && (rank_v<_CArray> == 1)
3240 mdspan(_CArray&)
3241 -> mdspan<remove_all_extents_t<_CArray>,
3242 extents<size_t, extent_v<_CArray, 0>>>;
3243
3244 template<typename _Pointer>
3245 requires is_pointer_v<remove_reference_t<_Pointer>>
3246 mdspan(_Pointer&&)
3247 -> mdspan<remove_pointer_t<remove_reference_t<_Pointer>>, extents<size_t>>;
3248
3249 template<typename _ElementType, typename... _Integrals>
3250 requires (is_convertible_v<_Integrals, size_t> && ...)
3251 && (sizeof...(_Integrals) > 0)
3252 explicit mdspan(_ElementType*, _Integrals...)
3253 -> mdspan<_ElementType,
3254 extents<size_t, __detail::__maybe_static_ext<_Integrals>...>>;
3255
3256 template<typename _ElementType, typename _OIndexType, size_t _Nm>
3257 mdspan(_ElementType*, span<_OIndexType, _Nm>)
3258 -> mdspan<_ElementType, dextents<size_t, _Nm>>;
3259
3260 template<typename _ElementType, typename _OIndexType, size_t _Nm>
3261 mdspan(_ElementType*, const array<_OIndexType, _Nm>&)
3262 -> mdspan<_ElementType, dextents<size_t, _Nm>>;
3263
3264 template<typename _ElementType, typename _IndexType, size_t... _ExtentsPack>
3265 mdspan(_ElementType*, const extents<_IndexType, _ExtentsPack...>&)
3266 -> mdspan<_ElementType, extents<_IndexType, _ExtentsPack...>>;
3267
3268 template<typename _ElementType, typename _MappingType>
3269 mdspan(_ElementType*, const _MappingType&)
3270 -> mdspan<_ElementType, typename _MappingType::extents_type,
3271 typename _MappingType::layout_type>;
3272
3273 // _GLIBCXX_RESOLVE_LIB_DEFECTS
3274 // 4511. Inconsistency between the deduction guide of std::mdspan taking...
3275 template<typename _MappingType, typename _AccessorType>
3276 mdspan(typename _AccessorType::data_handle_type, const _MappingType&,
3277 const _AccessorType&)
3278 -> mdspan<typename _AccessorType::element_type,
3279 typename _MappingType::extents_type,
3280 typename _MappingType::layout_type, _AccessorType>;
3281
3282#if __glibcxx_submdspan
3283 namespace __mdspan
3284 {
3285 template<typename _IndexType, typename _Slice>
3286 constexpr auto
3287 __canonical_index(_Slice&& __slice)
3288 {
3289 if constexpr (__detail::__integral_constant_like<_Slice>)
3290 {
3291 static_assert(__is_representable_integer<_IndexType>(_Slice::value));
3292 static_assert(_Slice::value >= 0);
3293 return std::cw<_IndexType(_Slice::value)>;
3294 }
3295 else
3296 return __mdspan::__index_type_cast<_IndexType>(std::move(__slice));
3297 }
3298
3299 template<typename _IndexType,
3300 typename _OffsetType, typename _SpanType, typename _StrideType>
3301 constexpr auto
3302 __canonical_range_slice(_OffsetType __offset, _SpanType __span,
3303 _StrideType __stride)
3304 {
3305 if constexpr (is_same_v<_SpanType, constant_wrapper<_IndexType(0)>>
3306 || is_same_v<_StrideType, constant_wrapper<_IndexType(1)>>)
3307 return extent_slice{
3308 .offset = __offset,
3309 .extent = __span,
3310 .stride = cw<_IndexType(1)>
3311 };
3312 else if constexpr (__is_constant_wrapper_v<_StrideType>)
3313 {
3314 static_assert(_StrideType::value > 0);
3315 if constexpr (__is_constant_wrapper_v<_SpanType>)
3316 return extent_slice{
3317 .offset = __offset,
3318 .extent = cw<_IndexType(1 + (_SpanType::value - 1) / _StrideType::value)>,
3319 .stride = __stride
3320 };
3321 else
3322 return extent_slice{
3323 .offset = __offset,
3324 .extent = _IndexType(__span > 0 ? 1 + (__span - 1) / _StrideType::value : 0),
3325 .stride = __stride
3326 };
3327 }
3328 else if (__span == 0 || __stride == 1)
3329 return extent_slice{
3330 .offset = __offset,
3331 .extent = _IndexType(__span),
3332 .stride = _IndexType(1)
3333 };
3334 else
3335 {
3336 __glibcxx_assert(__stride > 0);
3337 return extent_slice{
3338 .offset = __offset,
3339 .extent = _IndexType(1 + (__span - 1) / __stride),
3340 .stride = __stride
3341 };
3342 }
3343 }
3344
3345 template<typename _IndexType, typename _Slice>
3346 constexpr auto
3347 __slice_cast(_Slice&& __slice)
3348 {
3349 using _SliceType = remove_cvref_t<_Slice>;
3350 if constexpr (is_convertible_v<_SliceType, full_extent_t>)
3351 return static_cast<full_extent_t>(std::move(__slice));
3352 else if constexpr (is_convertible_v<_SliceType, _IndexType>)
3353 return __mdspan::__canonical_index<_IndexType>(std::move(__slice));
3354 else if constexpr (__is_extent_slice<_SliceType>)
3355 return extent_slice{
3356 .offset = __mdspan::__canonical_index<_IndexType>(std::move(__slice.offset)),
3357 .extent = __mdspan::__canonical_index<_IndexType>(std::move(__slice.extent)),
3358 .stride = __mdspan::__canonical_index<_IndexType>(std::move(__slice.stride))
3359 };
3360 else if constexpr (__is_range_slice<_SliceType>)
3361 {
3362 auto __first
3363 = __mdspan::__canonical_index<_IndexType>(std::move(__slice.first));
3364 auto __last
3365 = __mdspan::__canonical_index<_IndexType>(std::move(__slice.last));
3366 return __mdspan::__canonical_range_slice<_IndexType>(
3367 __first,
3368 __mdspan::__canonical_index<_IndexType>(__last - __first),
3369 __mdspan::__canonical_index<_IndexType>(std::move(__slice.stride)));
3370 }
3371 else
3372 {
3373 auto [__sbegin, __send] = std::move(__slice);
3374 auto __cbegin
3375 = __mdspan::__canonical_index<_IndexType>(std::move(__sbegin));
3376 auto __cend
3377 = __mdspan::__canonical_index<_IndexType>(std::move(__send));
3378 auto __cspan
3379 = __mdspan::__canonical_index<_IndexType>(__cend - __cbegin);
3380 return __mdspan::__canonical_range_slice<_IndexType>(
3381 __cbegin, __cspan, cw<_IndexType(1)>);
3382 }
3383 }
3384
3385 template<typename _IndexType, size_t _Extent, typename _OIndexType>
3386 constexpr void
3387 __check_inrange_index(const extents<_IndexType, _Extent>& __ext,
3388 const _OIndexType& __idx)
3389 {
3390 if constexpr (__is_constant_wrapper_v<_OIndexType>
3391 && _Extent != dynamic_extent)
3392 {
3393 static_assert(_OIndexType::value >= 0);
3394 static_assert(std::cmp_less(_OIndexType::value, _Extent));
3395 }
3396 else
3397 __glibcxx_assert(__idx < __ext.extent(0));
3398 }
3399
3400 template<typename _IndexType, size_t _Extent, typename _OIndexType>
3401 constexpr void
3402 __check_valid_index(const extents<_IndexType, _Extent>& __ext,
3403 const _OIndexType& __idx)
3404 {
3405 if constexpr (__is_constant_wrapper_v<_OIndexType>
3406 && _Extent != dynamic_extent)
3407 {
3408 static_assert(_OIndexType::value >= 0);
3409 static_assert(std::cmp_less_equal(_OIndexType::value, _Extent));
3410 }
3411 else
3412 __glibcxx_assert(__idx <= __ext.extent(0));
3413 }
3414
3415 template<typename _IndexType, size_t _Extent, typename _Slice>
3416 constexpr void
3417 __check_valid_slice(const extents<_IndexType, _Extent>& __ext,
3418 const _Slice& __slice)
3419 {
3420 if constexpr (__is_extent_slice<_Slice>)
3421 {
3422 __mdspan::__check_valid_index(__ext, __slice.extent);
3423 // DEVIATION: For empty slices, P3663r3 does not allow us to check
3424 // that this is less than or equal to the k-th extent (at runtime).
3425 // We're only allowed to check if __slice.offset, __slice.extent
3426 // are constant wrappers and __ext is a static extent.
3427 if constexpr (is_same_v<typename _Slice::extent_type,
3428 constant_wrapper<_IndexType(0)>>)
3429 __mdspan::__check_valid_index(__ext, __slice.offset);
3430 else if constexpr (is_same_v<typename _Slice::extent_type,
3431 constant_wrapper<_IndexType(1)>>)
3432 __mdspan::__check_inrange_index(__ext, __slice.offset);
3433 else if constexpr (__is_constant_wrapper_v<typename _Slice::extent_type>)
3434 {
3435 __mdspan::__check_inrange_index(__ext, __slice.offset);
3436 if constexpr (__is_constant_wrapper_v<typename _Slice::stride_type>)
3437 static_assert(_Slice::stride_type::value > 0);
3438 else
3439 __glibcxx_assert(__slice.stride > 0);
3440
3441 if constexpr (_Extent != dynamic_extent
3442 && __is_constant_wrapper_v<typename _Slice::offset_type>)
3443 static_assert(std::cmp_greater_equal(
3444 _Extent - _Slice::offset_type::value,
3445 _Slice::extent_type::value));
3446 if constexpr (_Extent != dynamic_extent
3447 && __is_constant_wrapper_v<typename _Slice::stride_type>)
3448 static_assert(std::cmp_greater(
3449 _Extent,
3450 (_Slice::extent_type::value - 1) * _Slice::stride_type::value));
3451
3452 if constexpr (_Extent != dynamic_extent
3453 && __is_constant_wrapper_v<typename _Slice::offset_type>
3454 && __is_constant_wrapper_v<typename _Slice::stride_type>)
3455 static_assert(std::cmp_greater(
3456 _Extent - _Slice::offset_type::value,
3457 (_Slice::extent_type::value - 1) * _Slice::stride_type::value));
3458 else
3459 __glibcxx_assert(std::cmp_greater(
3460 __ext.extent(0) - __slice.offset,
3461 (_Slice::extent_type::value - 1) * __slice.stride));
3462 }
3463 else if constexpr (is_same_v<typename _Slice::stride_type,
3464 constant_wrapper<_IndexType(1)>>)
3465 {
3466 __mdspan::__check_valid_index(__ext, __slice.offset);
3467 __glibcxx_assert(std::cmp_greater_equal(
3468 __ext.extent(0) - __slice.offset,
3469 __slice.extent));
3470 }
3471 else if (__slice.extent == 0)
3472 __mdspan::__check_valid_index(__ext, __slice.offset);
3473 else
3474 {
3475 __glibcxx_assert(__slice.offset < __ext.extent(0));
3476 __glibcxx_assert(__slice.extent == 1 || __slice.stride > 0);
3477 __glibcxx_assert(__slice.extent == 1 || std::cmp_greater(
3478 __ext.extent(0) - __slice.offset,
3479 (__slice.extent - 1) * __slice.stride));
3480 }
3481 }
3482 else if constexpr (!is_same_v<_Slice, full_extent_t>)
3483 __mdspan::__check_inrange_index(__ext, __slice);
3484 }
3485
3486 template<typename _Extents, typename... _Slices>
3487 constexpr void
3488 __check_valid_slices(const _Extents& __exts, const _Slices&... __slices)
3489 {
3490 constexpr auto __rank = _Extents::rank();
3491 auto __impl = [&]<size_t... _Is>(index_sequence<_Is...>)
3492 {
3493 ((__mdspan::__check_valid_slice(__extract_extent<_Is>(__exts),
3494 __slices...[_Is])),...);
3495 };
3496 __impl(make_index_sequence<__rank>());
3497 }
3498
3499 template<typename _Slice>
3500 using __full_extent_t = std::full_extent_t;
3501
3502 // Enables ADL-only calls from submdspan.
3503 void submdspan_mapping() = delete;
3504
3505 template<typename _Mapping, typename... _Slices>
3506 concept __sliceable_mapping = requires(const _Mapping __m, _Slices... __slices)
3507 {
3508 { submdspan_mapping(__m, __slices...) } -> __submdspan_mapping_result;
3509 };
3510
3511 template<typename _Mapping, typename... _Slices>
3512 constexpr auto
3513 __submapping(const _Mapping& __mapping, _Slices... __slices)
3514 {
3515 __mdspan::__check_valid_slices(__mapping.extents(), __slices...);
3516 return submdspan_mapping(__mapping, __slices...);
3517 }
3518 }
3519
3520 template<typename _IndexType, size_t... _Extents, typename... _RawSlices>
3521 requires (sizeof...(_RawSlices) == sizeof...(_Extents))
3522 constexpr auto
3523 subextents(const extents<_IndexType, _Extents...>& __exts,
3524 _RawSlices... __raw_slices)
3525 {
3526 auto __impl = [&__exts](auto... __slices)
3527 {
3528 __mdspan::__check_valid_slices(__exts, __slices...);
3529 return __mdspan::__subextents(__exts, __slices...);
3530 };
3531 return __impl(__mdspan::__slice_cast<_IndexType>(__raw_slices)...);
3532 }
3533
3534 template<typename _IndexType, size_t... _Extents, typename... _RawSlices>
3535 requires (sizeof...(_Extents) == sizeof...(_RawSlices))
3536 constexpr auto
3537 canonical_slices(const extents<_IndexType, _Extents...>& __exts,
3538 _RawSlices... __raw_slices)
3539 {
3540 auto __impl = [&__exts](auto... __slices)
3541 {
3542 __mdspan::__check_valid_slices(__exts, __slices...);
3543 return std::make_tuple(__slices...);
3544 };
3545 return __impl(__mdspan::__slice_cast<_IndexType>(__raw_slices)...);
3546 }
3547
3548 template<typename _ElementType, typename _Extents, typename _Layout,
3549 typename _Accessor, typename... _RawSlices>
3550 requires (sizeof...(_RawSlices) == _Extents::rank()
3551 && __mdspan::__sliceable_mapping<typename _Layout::template mapping<_Extents>,
3552 __mdspan::__full_extent_t<_RawSlices>...>)
3553 constexpr auto
3554 submdspan(
3555 const mdspan<_ElementType, _Extents, _Layout, _Accessor>& __md,
3556 _RawSlices... __raw_slices)
3557 {
3558 using _IndexType = typename _Extents::index_type;
3559 auto [__mapping, __offset] = __mdspan::__submapping(
3560 __md.mapping(), __mdspan::__slice_cast<_IndexType>(__raw_slices)...);
3561 return std::mdspan(
3562 __md.accessor().offset(__md.data_handle(), __offset),
3563 std::move(__mapping),
3564 typename _Accessor::offset_policy(__md.accessor()));
3565 }
3566#endif // __glibcxx_submdspan
3567
3568_GLIBCXX_END_NAMESPACE_VERSION
3569}
3570#endif
3571#endif
constexpr _Tp * assume_aligned(_Tp *__ptr) noexcept
Inform the compiler that a pointer is aligned.
Definition align.h:90
typename make_unsigned< _Tp >::type make_unsigned_t
Alias template for make_unsigned.
Definition type_traits:2273
constexpr tuple< typename __decay_and_strip< _Elements >::__type... > make_tuple(_Elements &&... __args)
Create a tuple containing copies of the arguments.
Definition tuple:2686
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
ISO C++ entities toplevel namespace is std.
make_integer_sequence< size_t, _Num > make_index_sequence
Alias template make_index_sequence.
Definition utility.h:559
integer_sequence< size_t, _Idx... > index_sequence
Alias template index_sequence.
Definition utility.h:555
__numeric_traits_integer< _Tp > __int_traits
Convenience alias for __numeric_traits<integer-type>.