Infinity Engine v0.6.20
C++ API Documentation
Loading...
Searching...
No Matches
Span.hpp
1// INFINITY_API_PUBLIC
2
3/*
4This is an implementation of C++20's std::span
5http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2019/n4820.pdf
6*/
7
8// Copyright Tristan Brindle 2018.
9// Distributed under the Boost Software License, Version 1.0.
10// (See accompanying file ../../LICENSE_1_0.txt or copy at
11// https://www.boost.org/LICENSE_1_0.txt)
12
13#ifndef TCB_SPAN_HPP_INCLUDED
14#define TCB_SPAN_HPP_INCLUDED
15
16#include <array>
17#include <cstddef>
18#include <cstdint>
19#include <type_traits>
20
21#ifndef TCB_SPAN_NO_EXCEPTIONS
22// Attempt to discover whether we're being compiled with exception support
23#if !(defined(__cpp_exceptions) || defined(__EXCEPTIONS) || defined(_CPPUNWIND))
24#define TCB_SPAN_NO_EXCEPTIONS
25#endif
26#endif
27
28#ifndef TCB_SPAN_NO_EXCEPTIONS
29#include <cstdio>
30#include <stdexcept>
31#endif
32
33// Various feature test macros
34
35#ifndef TCB_SPAN_NAMESPACE_NAME
36#define TCB_SPAN_NAMESPACE_NAME Infinity::Types::Core
37#endif
38
39#if __cplusplus >= 201703L || (defined(_MSVC_LANG) && _MSVC_LANG >= 201703L)
40#define TCB_SPAN_HAVE_CPP17
41#endif
42
43#if __cplusplus >= 201402L || (defined(_MSVC_LANG) && _MSVC_LANG >= 201402L)
44#define TCB_SPAN_HAVE_CPP14
45#endif
46
48
49// Establish default contract checking behavior
50#if !defined(TCB_SPAN_THROW_ON_CONTRACT_VIOLATION) && \
51 !defined(TCB_SPAN_TERMINATE_ON_CONTRACT_VIOLATION) && \
52 !defined(TCB_SPAN_NO_CONTRACT_CHECKING)
53#if defined(NDEBUG) || !defined(TCB_SPAN_HAVE_CPP14)
54#define TCB_SPAN_NO_CONTRACT_CHECKING
55#else
56#define TCB_SPAN_TERMINATE_ON_CONTRACT_VIOLATION
57#endif
58#endif
59
60#define TCB_SPAN_THROW_ON_CONTRACT_VIOLATION
61#if defined(TCB_SPAN_THROW_ON_CONTRACT_VIOLATION)
62struct contract_violation_error : std::logic_error {
63 explicit contract_violation_error(const char* msg) : std::logic_error(msg)
64 {}
65};
66
67inline void contract_violation(const char* msg)
68{
69 throw contract_violation_error(msg);
70}
71
72#elif defined(TCB_SPAN_TERMINATE_ON_CONTRACT_VIOLATION)
73[[noreturn]] inline void contract_violation(const char* /*unused*/)
74{
75 std::terminate();
76}
77#endif
78
79#if !defined(TCB_SPAN_NO_CONTRACT_CHECKING)
80#define TCB_SPAN_STRINGIFY(cond) #cond
81#define TCB_SPAN_EXPECT(cond) \
82 cond ? (void) 0 : contract_violation("Expected " TCB_SPAN_STRINGIFY(cond))
83#else
84#define TCB_SPAN_EXPECT(cond)
85#endif
86
87#if defined(TCB_SPAN_HAVE_CPP17) || defined(__cpp_inline_variables)
88#define TCB_SPAN_INLINE_VAR inline
89#else
90#define TCB_SPAN_INLINE_VAR
91#endif
92
93#if defined(TCB_SPAN_HAVE_CPP14) || \
94 (defined(__cpp_constexpr) && __cpp_constexpr >= 201304)
95#define TCB_SPAN_HAVE_CPP14_CONSTEXPR
96#endif
97
98#if defined(TCB_SPAN_HAVE_CPP14_CONSTEXPR)
99#define TCB_SPAN_CONSTEXPR14 constexpr
100#else
101#define TCB_SPAN_CONSTEXPR14
102#endif
103
104#if defined(TCB_SPAN_HAVE_CPP14_CONSTEXPR) && \
105 (!defined(_MSC_VER) || _MSC_VER > 1900)
106#define TCB_SPAN_CONSTEXPR_ASSIGN constexpr
107#else
108#define TCB_SPAN_CONSTEXPR_ASSIGN
109#endif
110
111#if defined(TCB_SPAN_NO_CONTRACT_CHECKING)
112#define TCB_SPAN_CONSTEXPR11 constexpr
113#else
114#define TCB_SPAN_CONSTEXPR11 TCB_SPAN_CONSTEXPR14
115#endif
116
117#if defined(TCB_SPAN_HAVE_CPP17) || defined(__cpp_deduction_guides)
118#define TCB_SPAN_HAVE_DEDUCTION_GUIDES
119#endif
120
121#if defined(TCB_SPAN_HAVE_CPP17) || defined(__cpp_lib_byte)
122#define TCB_SPAN_HAVE_STD_BYTE
123#endif
124
125#if defined(TCB_SPAN_HAVE_CPP17) || defined(__cpp_lib_array_constexpr)
126#define TCB_SPAN_HAVE_CONSTEXPR_STD_ARRAY_ETC
127#endif
128
129#if defined(TCB_SPAN_HAVE_CONSTEXPR_STD_ARRAY_ETC)
130#define TCB_SPAN_ARRAY_CONSTEXPR constexpr
131#else
132#define TCB_SPAN_ARRAY_CONSTEXPR
133#endif
134
135#ifdef TCB_SPAN_HAVE_STD_BYTE
136using byte = std::byte;
137#else
138using byte = unsigned char;
139#endif
140
141#if defined(TCB_SPAN_HAVE_CPP17)
142#define TCB_SPAN_NODISCARD [[nodiscard]]
143#else
144#define TCB_SPAN_NODISCARD
145#endif
146
147TCB_SPAN_INLINE_VAR constexpr size_t dynamic_extent = SIZE_MAX;
148
149template <typename ElementType, size_t Extent = dynamic_extent>
150class span;
151
152namespace detail {
153
154template <typename E, size_t S>
156 constexpr span_storage() noexcept = default;
157
158 constexpr span_storage(E* p_ptr, size_t /*unused*/) noexcept
159 : ptr(p_ptr)
160 {}
161
162 E* ptr = nullptr;
163 static constexpr size_t size = S;
164};
165
166template <typename E>
168 constexpr span_storage() noexcept = default;
169
170 constexpr span_storage(E* p_ptr, size_t p_size) noexcept
171 : ptr(p_ptr), size(p_size)
172 {}
173
174 E* ptr = nullptr;
175 size_t size = 0;
176};
177
178// Reimplementation of C++17 std::size() and std::data()
179#if defined(TCB_SPAN_HAVE_CPP17) || \
180 defined(__cpp_lib_nonmember_container_access)
181using std::data;
182using std::size;
183#else
184template <class C>
185constexpr auto size(const C& c) -> decltype(c.size())
186{
187 return c.size();
188}
189
190template <class T, size_t N>
191constexpr size_t size(const T (&)[N]) noexcept
192{
193 return N;
194}
195
196template <class C>
197constexpr auto data(C& c) -> decltype(c.data())
198{
199 return c.data();
200}
201
202template <class C>
203constexpr auto data(const C& c) -> decltype(c.data())
204{
205 return c.data();
206}
207
208template <class T, size_t N>
209constexpr T* data(T (&array)[N]) noexcept
210{
211 return array;
212}
213
214template <class E>
215constexpr const E* data(std::initializer_list<E> il) noexcept
216{
217 return il.begin();
218}
219#endif // TCB_SPAN_HAVE_CPP17
220
221#if defined(TCB_SPAN_HAVE_CPP17) || defined(__cpp_lib_void_t)
222using std::void_t;
223#else
224template <typename...>
225using void_t = void;
226#endif
227
228template <typename T>
230 typename std::remove_cv<typename std::remove_reference<T>::type>::type;
231
232template <typename>
233struct is_span : std::false_type {};
234
235template <typename T, size_t S>
236struct is_span<span<T, S>> : std::true_type {};
237
238template <typename>
239struct is_std_array : std::false_type {};
240
241template <typename T, size_t N>
242struct is_std_array<std::array<T, N>> : std::true_type {};
243
244template <typename, typename = void>
245struct has_size_and_data : std::false_type {};
246
247template <typename T>
248struct has_size_and_data<T, void_t<decltype(detail::size(std::declval<T>())),
249 decltype(detail::data(std::declval<T>()))>>
250 : std::true_type {};
251
252template <typename C, typename U = uncvref_t<C>>
254 static constexpr bool value =
256 !std::is_array<U>::value && has_size_and_data<C>::value;
257};
258
259template <typename T>
260using remove_pointer_t = typename std::remove_pointer<T>::type;
261
262template <typename, typename, typename = void>
263struct is_container_element_type_compatible : std::false_type {};
264
265template <typename T, typename E>
267 T, E,
268 typename std::enable_if<
269 !std::is_same<
270 typename std::remove_cv<decltype(detail::data(std::declval<T>()))>::type,
271 void>::value &&
272 std::is_convertible<
273 remove_pointer_t<decltype(detail::data(std::declval<T>()))> (*)[],
274 E (*)[]>::value
275 >::type>
276 : std::true_type {};
277
278template <typename, typename = size_t>
279struct is_complete : std::false_type {};
280
281template <typename T>
282struct is_complete<T, decltype(sizeof(T))> : std::true_type {};
283
284} // namespace detail
285
286template <typename ElementType, size_t Extent>
287class span {
288 static_assert(std::is_object<ElementType>::value,
289 "A span's ElementType must be an object type (not a "
290 "reference type or void)");
292 "A span's ElementType must be a complete type (not a forward "
293 "declaration)");
294 static_assert(!std::is_abstract<ElementType>::value,
295 "A span's ElementType cannot be an abstract class type");
296
298
299public:
300 // constants and types
301 using element_type = ElementType;
302 using value_type = typename std::remove_cv<ElementType>::type;
303 using size_type = size_t;
304 using difference_type = std::ptrdiff_t;
310 using reverse_iterator = std::reverse_iterator<iterator>;
311
312 static constexpr size_type extent = Extent;
313
314 // [span.cons], span constructors, copy, assignment, and destructor
315 template <
316 size_t E = Extent,
317 typename std::enable_if<(E == dynamic_extent || E <= 0), int>::type = 0>
318 constexpr span() noexcept
319 {}
320
321 TCB_SPAN_CONSTEXPR11 span(pointer ptr, size_type count)
322 : storage_(ptr, count)
323 {
324 TCB_SPAN_EXPECT(extent == dynamic_extent || count == extent);
325 }
326
327 TCB_SPAN_CONSTEXPR11 span(pointer first_elem, pointer last_elem)
328 : storage_(first_elem, last_elem - first_elem)
329 {
330 TCB_SPAN_EXPECT(extent == dynamic_extent ||
331 last_elem - first_elem ==
332 static_cast<std::ptrdiff_t>(extent));
333 }
334
335 template <size_t N, size_t E = Extent,
336 typename std::enable_if<
337 (E == dynamic_extent || N == E) &&
339 element_type (&)[N], ElementType>::value,
340 int>::type = 0>
341 constexpr span(element_type (&arr)[N]) noexcept : storage_(arr, N)
342 {}
343
344 template <typename T, size_t N, size_t E = Extent,
345 typename std::enable_if<
346 (E == dynamic_extent || N == E) &&
348 std::array<T, N>&, ElementType>::value,
349 int>::type = 0>
350 TCB_SPAN_ARRAY_CONSTEXPR span(std::array<T, N>& arr) noexcept
351 : storage_(arr.data(), N)
352 {}
353
354 template <typename T, size_t N, size_t E = Extent,
355 typename std::enable_if<
356 (E == dynamic_extent || N == E) &&
358 const std::array<T, N>&, ElementType>::value,
359 int>::type = 0>
360 TCB_SPAN_ARRAY_CONSTEXPR span(const std::array<T, N>& arr) noexcept
361 : storage_(arr.data(), N)
362 {}
363
364 template <
365 typename Container, size_t E = Extent,
366 typename std::enable_if<
369 Container&, ElementType>::value,
370 int>::type = 0>
371 constexpr span(Container& cont)
372 : storage_(detail::data(cont), detail::size(cont))
373 {}
374
375 template <
376 typename Container, size_t E = Extent,
377 typename std::enable_if<
380 const Container&, ElementType>::value,
381 int>::type = 0>
382 constexpr span(const Container& cont)
383 : storage_(detail::data(cont), detail::size(cont))
384 {}
385
386 constexpr span(const span& other) noexcept = default;
387
388 template <typename OtherElementType, size_t OtherExtent,
389 typename std::enable_if<
390 (Extent == dynamic_extent || OtherExtent == dynamic_extent ||
391 Extent == OtherExtent) &&
392 std::is_convertible<OtherElementType (*)[],
393 ElementType (*)[]>::value,
394 int>::type = 0>
395 constexpr span(const span<OtherElementType, OtherExtent>& other) noexcept
396 : storage_(other.data(), other.size())
397 {}
398
399 ~span() noexcept = default;
400
401 TCB_SPAN_CONSTEXPR_ASSIGN span&
402 operator=(const span& other) noexcept = default;
403
404 // [span.sub], span subviews
405 template <size_t Count>
406 TCB_SPAN_CONSTEXPR11 span<element_type, Count> first() const
407 {
408 TCB_SPAN_EXPECT(Count <= size());
409 return {data(), Count};
410 }
411
412 template <size_t Count>
413 TCB_SPAN_CONSTEXPR11 span<element_type, Count> last() const
414 {
415 TCB_SPAN_EXPECT(Count <= size());
416 return {data() + (size() - Count), Count};
417 }
418
419 template <size_t Offset, size_t Count = dynamic_extent>
421 span<ElementType, Count != dynamic_extent
422 ? Count
423 : (Extent != dynamic_extent ? Extent - Offset
424 : dynamic_extent)>;
425
426 template <size_t Offset, size_t Count = dynamic_extent>
427 TCB_SPAN_CONSTEXPR11 subspan_return_t<Offset, Count> subspan() const
428 {
429 TCB_SPAN_EXPECT(Offset <= size() &&
430 (Count == dynamic_extent || Offset + Count <= size()));
431 return {data() + Offset,
432 Count != dynamic_extent ? Count : size() - Offset};
433 }
434
435 TCB_SPAN_CONSTEXPR11 span<element_type, dynamic_extent>
436 first(size_type count) const
437 {
438 TCB_SPAN_EXPECT(count <= size());
439 return {data(), count};
440 }
441
442 TCB_SPAN_CONSTEXPR11 span<element_type, dynamic_extent>
443 last(size_type count) const
444 {
445 TCB_SPAN_EXPECT(count <= size());
446 return {data() + (size() - count), count};
447 }
448
449 TCB_SPAN_CONSTEXPR11 span<element_type, dynamic_extent>
450 subspan(size_type offset, size_type count = dynamic_extent) const
451 {
452 TCB_SPAN_EXPECT(offset <= size() &&
453 (count == dynamic_extent || offset + count <= size()));
454 return {data() + offset,
455 count == dynamic_extent ? size() - offset : count};
456 }
457
458 // [span.obs], span observers
459 constexpr size_type size() const noexcept { return storage_.size; }
460
461 constexpr size_type size_bytes() const noexcept
462 {
463 return size() * sizeof(element_type);
464 }
465
466 TCB_SPAN_NODISCARD constexpr bool empty() const noexcept
467 {
468 return size() == 0;
469 }
470
471 // [span.elem], span element access
472 TCB_SPAN_CONSTEXPR11 reference operator[](size_type idx) const
473 {
474 TCB_SPAN_EXPECT(idx < size());
475 return *(data() + idx);
476 }
477
478 TCB_SPAN_CONSTEXPR11 reference front() const
479 {
480 TCB_SPAN_EXPECT(!empty());
481 return *data();
482 }
483
484 TCB_SPAN_CONSTEXPR11 reference back() const
485 {
486 TCB_SPAN_EXPECT(!empty());
487 return *(data() + (size() - 1));
488 }
489
490 constexpr pointer data() const noexcept { return storage_.ptr; }
491
492 // [span.iterators], span iterator support
493 constexpr iterator begin() const noexcept { return data(); }
494
495 constexpr iterator end() const noexcept { return data() + size(); }
496
497 TCB_SPAN_ARRAY_CONSTEXPR reverse_iterator rbegin() const noexcept
498 {
499 return reverse_iterator(end());
500 }
501
502 TCB_SPAN_ARRAY_CONSTEXPR reverse_iterator rend() const noexcept
503 {
504 return reverse_iterator(begin());
505 }
506
507private:
508 storage_type storage_{};
509};
510
511#ifdef TCB_SPAN_HAVE_DEDUCTION_GUIDES
512
513/* Deduction Guides */
514template <class T, size_t N>
515span(T (&)[N])->span<T, N>;
516
517template <class T, size_t N>
518span(std::array<T, N>&)->span<T, N>;
519
520template <class T, size_t N>
521span(const std::array<T, N>&)->span<const T, N>;
522
523template <class Container>
524span(Container&)->span<typename std::remove_reference<
525 decltype(*detail::data(std::declval<Container&>()))>::type>;
526
527template <class Container>
528span(const Container&)->span<const typename Container::value_type>;
529
530#endif // TCB_HAVE_DEDUCTION_GUIDES
531
532template <typename ElementType, size_t Extent>
533constexpr span<ElementType, Extent>
535{
536 return s;
537}
538
539template <typename T, size_t N>
540constexpr span<T, N> make_span(T (&arr)[N]) noexcept
541{
542 return {arr};
543}
544
545template <typename T, size_t N>
546TCB_SPAN_ARRAY_CONSTEXPR span<T, N> make_span(std::array<T, N>& arr) noexcept
547{
548 return {arr};
549}
550
551template <typename T, size_t N>
552TCB_SPAN_ARRAY_CONSTEXPR span<const T, N>
553make_span(const std::array<T, N>& arr) noexcept
554{
555 return {arr};
556}
557
558template <typename Container>
559constexpr span<typename std::remove_reference<
560 decltype(*detail::data(std::declval<Container&>()))>::type>
561make_span(Container& cont)
562{
563 return {cont};
564}
565
566template <typename Container>
567constexpr span<const typename Container::value_type>
568make_span(const Container& cont)
569{
570 return {cont};
571}
572
573template <typename ElementType, size_t Extent>
574span<const byte, ((Extent == dynamic_extent) ? dynamic_extent
575 : sizeof(ElementType) * Extent)>
576as_bytes(span<ElementType, Extent> s) noexcept
577{
578 return {reinterpret_cast<const byte*>(s.data()), s.size_bytes()};
579}
580
581template <
582 class ElementType, size_t Extent,
583 typename std::enable_if<!std::is_const<ElementType>::value, int>::type = 0>
584span<byte, ((Extent == dynamic_extent) ? dynamic_extent
585 : sizeof(ElementType) * Extent)>
586as_writable_bytes(span<ElementType, Extent> s) noexcept
587{
588 return {reinterpret_cast<byte*>(s.data()), s.size_bytes()};
589}
590
591template <size_t N, typename E, size_t S>
592constexpr auto get(span<E, S> s) -> decltype(s[N])
593{
594 return s[N];
595}
596
597} // namespace TCB_SPAN_NAMESPACE_NAME
598
599namespace std {
600
601template <typename ElementType, size_t Extent>
602class tuple_size<TCB_SPAN_NAMESPACE_NAME::span<ElementType, Extent>>
603 : public integral_constant<size_t, Extent> {};
604
605template <typename ElementType>
606class tuple_size<TCB_SPAN_NAMESPACE_NAME::span<
607 ElementType, TCB_SPAN_NAMESPACE_NAME::dynamic_extent>>; // not defined
608
609template <size_t I, typename ElementType, size_t Extent>
610class tuple_element<I, TCB_SPAN_NAMESPACE_NAME::span<ElementType, Extent>> {
611public:
612 static_assert(Extent != TCB_SPAN_NAMESPACE_NAME::dynamic_extent &&
613 I < Extent,
614 "");
615 using type = ElementType;
616};
617
618} // end namespace std
619
620#endif // TCB_SPAN_HPP_INCLUDED
Definition Span.hpp:287
constexpr span(Container &cont)
Definition Span.hpp:371
TCB_SPAN_CONSTEXPR11 span< element_type, Count > last() const
Definition Span.hpp:413
constexpr span() noexcept
Definition Span.hpp:318
constexpr iterator end() const noexcept
Definition Span.hpp:495
~span() noexcept=default
constexpr span(const Container &cont)
Definition Span.hpp:382
std::reverse_iterator< iterator > reverse_iterator
Definition Span.hpp:310
ElementType element_type
Definition Span.hpp:301
constexpr iterator begin() const noexcept
Definition Span.hpp:493
TCB_SPAN_CONSTEXPR11 span(pointer first_elem, pointer last_elem)
Definition Span.hpp:327
static constexpr size_type extent
Definition Span.hpp:312
const element_type * const_pointer
Definition Span.hpp:306
constexpr span(const span &other) noexcept=default
typename std::remove_cv< ElementType >::type value_type
Definition Span.hpp:302
TCB_SPAN_ARRAY_CONSTEXPR reverse_iterator rbegin() const noexcept
Definition Span.hpp:497
const element_type & const_reference
Definition Span.hpp:308
std::ptrdiff_t difference_type
Definition Span.hpp:304
TCB_SPAN_CONSTEXPR11 span< element_type, dynamic_extent > first(size_type count) const
Definition Span.hpp:436
pointer iterator
Definition Span.hpp:309
TCB_SPAN_CONSTEXPR11 reference back() const
Definition Span.hpp:484
constexpr size_type size() const noexcept
Definition Span.hpp:459
TCB_SPAN_CONSTEXPR11 reference front() const
Definition Span.hpp:478
constexpr size_type size_bytes() const noexcept
Definition Span.hpp:461
TCB_SPAN_CONSTEXPR11 reference operator[](size_type idx) const
Definition Span.hpp:472
TCB_SPAN_CONSTEXPR11 subspan_return_t< Offset, Count > subspan() const
Definition Span.hpp:427
constexpr span(element_type(&arr)[N]) noexcept
Definition Span.hpp:341
TCB_SPAN_ARRAY_CONSTEXPR span(std::array< T, N > &arr) noexcept
Definition Span.hpp:350
TCB_SPAN_ARRAY_CONSTEXPR span(const std::array< T, N > &arr) noexcept
Definition Span.hpp:360
TCB_SPAN_CONSTEXPR11 span< element_type, dynamic_extent > subspan(size_type offset, size_type count=dynamic_extent) const
Definition Span.hpp:450
TCB_SPAN_CONSTEXPR11 span(pointer ptr, size_type count)
Definition Span.hpp:321
constexpr pointer data() const noexcept
Definition Span.hpp:490
TCB_SPAN_CONSTEXPR11 span< element_type, dynamic_extent > last(size_type count) const
Definition Span.hpp:443
TCB_SPAN_ARRAY_CONSTEXPR reverse_iterator rend() const noexcept
Definition Span.hpp:502
constexpr span(const span< OtherElementType, OtherExtent > &other) noexcept
Definition Span.hpp:395
element_type * pointer
Definition Span.hpp:305
element_type & reference
Definition Span.hpp:307
size_t size_type
Definition Span.hpp:303
TCB_SPAN_NODISCARD constexpr bool empty() const noexcept
Definition Span.hpp:466
typename std::remove_cv< typename std::remove_reference< T >::type >::type uncvref_t
Definition Span.hpp:230
constexpr auto size(const C &c) -> decltype(c.size())
Definition Span.hpp:185
void void_t
Definition Span.hpp:225
typename std::remove_pointer< T >::type remove_pointer_t
Definition Span.hpp:260
constexpr auto data(C &c) -> decltype(c.data())
Definition Span.hpp:197
Definition Span.hpp:47
span< byte,((Extent==dynamic_extent) ? dynamic_extent :sizeof(ElementType) *Extent)> as_writable_bytes(span< ElementType, Extent > s) noexcept
Definition Span.hpp:586
constexpr span< ElementType, Extent > make_span(span< ElementType, Extent > s) noexcept
Definition Span.hpp:534
span< const byte,((Extent==dynamic_extent) ? dynamic_extent :sizeof(ElementType) *Extent)> as_bytes(span< ElementType, Extent > s) noexcept
Definition Span.hpp:576
void contract_violation(const char *msg)
Definition Span.hpp:67
constexpr auto get(span< E, S > s) -> decltype(s[N])
Definition Span.hpp:592
unsigned char byte
Definition Span.hpp:138
TCB_SPAN_INLINE_VAR constexpr size_t dynamic_extent
Definition Span.hpp:147
Definition Span.hpp:599
contract_violation_error(const char *msg)
Definition Span.hpp:63
static constexpr bool value
Definition Span.hpp:254
constexpr span_storage() noexcept=default
static constexpr size_t size
Definition Span.hpp:163