ttg 1.0.0-alpha
Template Task Graph (TTG): flowgraph-based programming model for high-performance distributed-memory algorithms
Loading...
Searching...
No Matches
span.h
Go to the documentation of this file.
1#ifndef TTG_UTIL_SPAN_H
2#define TTG_UTIL_SPAN_H
3
4/* if TCB_SPAN_NAMESPACE_NAME already defined someone imported TCB span somewhere,
5 * then reset it here to the ttg namespace, don't forget to revert back */
6#ifdef TCB_SPAN_NAMESPACE_NAME
7
8#define TTG_TCB_SPAN_NAMESPACE_NAME_AT_TOP_OF_SPAN_H TCB_SPAN_NAMESPACE_NAME
9#undef TCB_SPAN_NAMESPACE_NAME
10
11#endif
12
13/* Use the TTG namespace */
14#define TCB_SPAN_NAMESPACE_NAME ttg
15
16/*
17This is an implementation of C++20's std::span
18http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2019/n4820.pdf
19Code taken from https://github.com/tcbrindle/span/
20*/
21
22// Copyright Tristan Brindle 2018.
23// Distributed under the Boost Software License, Version 1.0.
24// (See accompanying file ../../LICENSE_1_0.txt or copy at
25// https://www.boost.org/LICENSE_1_0.txt)
26
27#include <array>
28#include <cstddef>
29#include <cstdint>
30#include <type_traits>
31#include <exception>
32
33#ifndef TCB_SPAN_NO_EXCEPTIONS
34// Attempt to discover whether we're being compiled with exception support
35#if !(defined(__cpp_exceptions) || defined(__EXCEPTIONS) || defined(_CPPUNWIND))
36#define TCB_SPAN_NO_EXCEPTIONS
37#endif
38#endif
39
40#ifndef TCB_SPAN_NO_EXCEPTIONS
41#include <cstdio>
42#include <stdexcept>
43#endif
44
45// Various feature test macros
46
47#ifndef TCB_SPAN_NAMESPACE_NAME
48#define TCB_SPAN_NAMESPACE_NAME tcb
49#endif
50
51#if __cplusplus >= 201703L || (defined(_MSVC_LANG) && _MSVC_LANG >= 201703L)
52#define TCB_SPAN_HAVE_CPP17
53#endif
54
55#if __cplusplus >= 201402L || (defined(_MSVC_LANG) && _MSVC_LANG >= 201402L)
56#define TCB_SPAN_HAVE_CPP14
57#endif
58
60
61// Establish default contract checking behavior
62#if !defined(TCB_SPAN_THROW_ON_CONTRACT_VIOLATION) && !defined(TCB_SPAN_TERMINATE_ON_CONTRACT_VIOLATION) && \
63 !defined(TCB_SPAN_NO_CONTRACT_CHECKING)
64#if defined(NDEBUG) || !defined(TCB_SPAN_HAVE_CPP14)
65#define TCB_SPAN_NO_CONTRACT_CHECKING
66#else
67#define TCB_SPAN_TERMINATE_ON_CONTRACT_VIOLATION
68#endif
69#endif
70
71#if defined(TCB_SPAN_THROW_ON_CONTRACT_VIOLATION)
72 struct contract_violation_error : std::logic_error {
73 explicit contract_violation_error(const char* msg) : std::logic_error(msg) {}
74 };
75
76 inline void contract_violation(const char* msg) { throw contract_violation_error(msg); }
77
78#elif defined(TCB_SPAN_TERMINATE_ON_CONTRACT_VIOLATION)
79 [[noreturn]] inline void contract_violation(const char* /*unused*/) { std::terminate(); }
80#endif
81
82#if !defined(TCB_SPAN_NO_CONTRACT_CHECKING)
83#define TCB_SPAN_STRINGIFY(cond) #cond
84#define TCB_SPAN_EXPECT(cond) cond ? (void)0 : contract_violation("Expected " TCB_SPAN_STRINGIFY(cond))
85#else
86#define TCB_SPAN_EXPECT(cond)
87#endif
88
89#if defined(TCB_SPAN_HAVE_CPP17) || defined(__cpp_inline_variables)
90#define TCB_SPAN_INLINE_VAR inline
91#else
92#define TCB_SPAN_INLINE_VAR
93#endif
94
95#if defined(TCB_SPAN_HAVE_CPP14) || (defined(__cpp_constexpr) && __cpp_constexpr >= 201304)
96#define TCB_SPAN_HAVE_CPP14_CONSTEXPR
97#endif
98
99#if defined(TCB_SPAN_HAVE_CPP14_CONSTEXPR)
100#define TCB_SPAN_CONSTEXPR14 constexpr
101#else
102#define TCB_SPAN_CONSTEXPR14
103#endif
104
105#if defined(TCB_SPAN_HAVE_CPP14_CONSTEXPR) && (!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
136 using byte = std::byte;
137#else
138 using 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
147 TCB_SPAN_INLINE_VAR constexpr std::size_t dynamic_extent = SIZE_MAX;
148
149 template <typename ElementType, std::size_t Extent = dynamic_extent>
150 class span;
151
152 namespace detail {
153
154 template <typename E, std::size_t S>
156 constexpr span_storage() noexcept = default;
157
158 constexpr span_storage(E* p_ptr, std::size_t /*unused*/) noexcept : ptr(p_ptr) {}
159
160 E* ptr = nullptr;
161 static constexpr std::size_t size = S;
162 };
163
164 template <typename E>
166 constexpr span_storage() noexcept = default;
167
168 constexpr span_storage(E* p_ptr, std::size_t p_size) noexcept : ptr(p_ptr), size(p_size) {}
169
170 E* ptr = nullptr;
171 std::size_t size = 0;
172 };
173
174// Reimplementation of C++17 std::size() and std::data()
175#if defined(TCB_SPAN_HAVE_CPP17) || defined(__cpp_lib_nonmember_container_access)
176 using std::data;
177 using std::size;
178#else
179 template <class C>
180 constexpr auto size(const C& c) -> decltype(c.size()) {
181 return c.size();
182 }
183
184 template <class T, std::size_t N>
185 constexpr std::size_t size(const T (&)[N]) noexcept {
186 return N;
187 }
188
189 template <class C>
190 constexpr auto data(C& c) -> decltype(c.data()) {
191 return c.data();
192 }
193
194 template <class C>
195 constexpr auto data(const C& c) -> decltype(c.data()) {
196 return c.data();
197 }
198
199 template <class T, std::size_t N>
200 constexpr T* data(T (&array)[N]) noexcept {
201 return array;
202 }
203
204 template <class E>
205 constexpr const E* data(std::initializer_list<E> il) noexcept {
206 return il.begin();
207 }
208#endif // TCB_SPAN_HAVE_CPP17
209
210#if defined(TCB_SPAN_HAVE_CPP17) || defined(__cpp_lib_void_t)
211 using std::void_t;
212#else
213 template <typename...>
214 using void_t = void;
215#endif
216
217 template <typename T>
218 using uncvref_t = typename std::remove_cv<typename std::remove_reference<T>::type>::type;
219
220 template <typename>
221 struct is_span : std::false_type {};
222
223 template <typename T, std::size_t S>
224 struct is_span<span<T, S> > : std::true_type {};
225
226 template <typename>
227 struct is_std_array : std::false_type {};
228
229 template <typename T, std::size_t N>
230 struct is_std_array<std::array<T, N> > : std::true_type {};
231
232 template <typename, typename = void>
233 struct has_size_and_data : std::false_type {};
234
235 template <typename T>
237 T, void_t<decltype(detail::size(std::declval<T>())), decltype(detail::data(std::declval<T>()))> >
238 : std::true_type {};
239
240 template <typename C, typename U = uncvref_t<C> >
242 static constexpr bool value =
243 !is_span<U>::value && !is_std_array<U>::value && !std::is_array<U>::value && has_size_and_data<C>::value;
244 };
245
246 template <typename T>
247 using remove_pointer_t = typename std::remove_pointer<T>::type;
248
249 template <typename, typename, typename = void>
250 struct is_container_element_type_compatible : std::false_type {};
251
252 template <typename T, typename E>
254 T, E,
255 typename std::enable_if<
256 !std::is_same<typename std::remove_cv<decltype(detail::data(std::declval<T>()))>::type, void>::value>::type>
257 : std::is_convertible<remove_pointer_t<decltype(detail::data(std::declval<T>()))> (*)[], E (*)[]> {};
258
259 template <typename, typename = size_t>
260 struct is_complete : std::false_type {};
261
262 template <typename T>
263 struct is_complete<T, decltype(sizeof(T))> : std::true_type {};
264
265 } // namespace detail
266
267 template <typename ElementType, std::size_t Extent>
268 class span {
269 static_assert(std::is_object<ElementType>::value,
270 "A span's ElementType must be an object type (not a "
271 "reference type or void)");
273 "A span's ElementType must be a complete type (not a forward "
274 "declaration)");
275 static_assert(!std::is_abstract<ElementType>::value, "A span's ElementType cannot be an abstract class type");
276
278
279 public:
280 // constants and types
281 using element_type = ElementType;
282 using value_type = typename std::remove_cv<ElementType>::type;
283 using size_type = std::size_t;
284 using difference_type = std::ptrdiff_t;
290 using reverse_iterator = std::reverse_iterator<iterator>;
291
292 static constexpr size_type extent = Extent;
293
294 // [span.cons], span constructors, copy, assignment, and destructor
295 template <std::size_t E = Extent, typename std::enable_if<(E == dynamic_extent || E <= 0), int>::type = 0>
296 constexpr span() noexcept {}
297
298 TCB_SPAN_CONSTEXPR11 span(pointer ptr, size_type count) : storage_(ptr, count) {
299 TCB_SPAN_EXPECT(extent == dynamic_extent || count == extent);
300 }
301
302 TCB_SPAN_CONSTEXPR11 span(pointer first_elem, pointer last_elem) : storage_(first_elem, last_elem - first_elem) {
303 TCB_SPAN_EXPECT(extent == dynamic_extent || last_elem - first_elem == static_cast<std::ptrdiff_t>(extent));
304 }
305
306 template <std::size_t N, std::size_t E = Extent,
307 typename std::enable_if<(E == dynamic_extent || N == E) && detail::is_container_element_type_compatible<
308 element_type (&)[N], ElementType>::value,
309 int>::type = 0>
310 constexpr span(element_type (&arr)[N]) noexcept : storage_(arr, N) {}
311
312 template <std::size_t N, std::size_t E = Extent,
313 typename std::enable_if<
314 (E == dynamic_extent || N == E) &&
315 detail::is_container_element_type_compatible<std::array<value_type, N>&, ElementType>::value,
316 int>::type = 0>
317 TCB_SPAN_ARRAY_CONSTEXPR span(std::array<value_type, N>& arr) noexcept : storage_(arr.data(), N) {}
318
319 template <std::size_t N, std::size_t E = Extent,
320 typename std::enable_if<(E == dynamic_extent || N == E) &&
321 detail::is_container_element_type_compatible<const std::array<value_type, N>&,
322 ElementType>::value,
323 int>::type = 0>
324 TCB_SPAN_ARRAY_CONSTEXPR span(const std::array<value_type, N>& arr) noexcept : storage_(arr.data(), N) {}
325
326 template <typename Container, std::size_t E = Extent,
327 typename std::enable_if<E == dynamic_extent && detail::is_container<Container>::value &&
329 int>::type = 0>
330 constexpr span(Container& cont) : storage_(detail::data(cont), detail::size(cont)) {}
331
332 template <
333 typename Container, std::size_t E = Extent,
334 typename std::enable_if<E == dynamic_extent && detail::is_container<Container>::value &&
336 int>::type = 0>
337 constexpr span(const Container& cont) : storage_(detail::data(cont), detail::size(cont)) {}
338
339 constexpr span(const span& other) noexcept = default;
340
341 template <typename OtherElementType, std::size_t OtherExtent,
342 typename std::enable_if<(Extent == OtherExtent || Extent == dynamic_extent) &&
343 std::is_convertible<OtherElementType (*)[], ElementType (*)[]>::value,
344 int>::type = 0>
345 constexpr span(const span<OtherElementType, OtherExtent>& other) noexcept : storage_(other.data(), other.size()) {}
346
347 ~span() noexcept = default;
348
349 TCB_SPAN_CONSTEXPR_ASSIGN span& operator=(const span& other) noexcept = default;
350
351 // [span.sub], span subviews
352 template <std::size_t Count>
353 TCB_SPAN_CONSTEXPR11 span<element_type, Count> first() const {
354 TCB_SPAN_EXPECT(Count <= size());
355 return {data(), Count};
356 }
357
358 template <std::size_t Count>
360 TCB_SPAN_EXPECT(Count <= size());
361 return {data() + (size() - Count), Count};
362 }
363
364 template <std::size_t Offset, std::size_t Count = dynamic_extent>
366 span<ElementType,
367 Count != dynamic_extent ? Count : (Extent != dynamic_extent ? Extent - Offset : dynamic_extent)>;
368
369 template <std::size_t Offset, std::size_t Count = dynamic_extent>
371 TCB_SPAN_EXPECT(Offset <= size() && (Count == dynamic_extent || Offset + Count <= size()));
372 return {data() + Offset, Count != dynamic_extent ? Count : size() - Offset};
373 }
374
376 TCB_SPAN_EXPECT(count <= size());
377 return {data(), count};
378 }
379
381 TCB_SPAN_EXPECT(count <= size());
382 return {data() + (size() - count), count};
383 }
384
386 size_type count = dynamic_extent) const {
387 TCB_SPAN_EXPECT(offset <= size() && (count == dynamic_extent || offset + count <= size()));
388 return {data() + offset, count == dynamic_extent ? size() - offset : count};
389 }
390
391 // [span.obs], span observers
392 constexpr size_type size() const noexcept { return storage_.size; }
393
394 constexpr size_type size_bytes() const noexcept { return size() * sizeof(element_type); }
395
396 TCB_SPAN_NODISCARD constexpr bool empty() const noexcept { return size() == 0; }
397
398 // [span.elem], span element access
400 TCB_SPAN_EXPECT(idx < size());
401 return *(data() + idx);
402 }
403
406 return *data();
407 }
408
411 return *(data() + (size() - 1));
412 }
413
414 constexpr pointer data() const noexcept { return storage_.ptr; }
415
416 // [span.iterators], span iterator support
417 constexpr iterator begin() const noexcept { return data(); }
418
419 constexpr iterator end() const noexcept { return data() + size(); }
420
422
424
425 private:
426 storage_type storage_{};
427 };
428
429#ifdef TCB_SPAN_HAVE_DEDUCTION_GUIDES
430
431 /* Deduction Guides */
432 template <class T, size_t N>
433 span(T (&)[N]) -> span<T, N>;
434
435 template <class T, size_t N>
436 span(std::array<T, N>&) -> span<T, N>;
437
438 template <class T, size_t N>
439 span(const std::array<T, N>&) -> span<const T, N>;
440
441 template <class Container>
442 span(Container&) -> span<typename Container::value_type>;
443
444 template <class Container>
445 span(const Container&) -> span<const typename Container::value_type>;
446
447#endif // TCB_HAVE_DEDUCTION_GUIDES
448
449 template <typename ElementType, std::size_t Extent>
451 return s;
452 }
453
454 template <typename T, std::size_t N>
455 constexpr span<T, N> make_span(T (&arr)[N]) noexcept {
456 return {arr};
457 }
458
459 template <typename T, std::size_t N>
460 TCB_SPAN_ARRAY_CONSTEXPR span<T, N> make_span(std::array<T, N>& arr) noexcept {
461 return {arr};
462 }
463
464 template <typename T, std::size_t N>
465 TCB_SPAN_ARRAY_CONSTEXPR span<const T, N> make_span(const std::array<T, N>& arr) noexcept {
466 return {arr};
467 }
468
469 template <typename Container>
471 return {cont};
472 }
473
474 template <typename Container>
475 constexpr span<const typename Container::value_type> make_span(const Container& cont) {
476 return {cont};
477 }
478
479 template <typename ElementType, std::size_t Extent>
480 span<const byte, ((Extent == dynamic_extent) ? dynamic_extent : sizeof(ElementType) * Extent)> as_bytes(
481 span<ElementType, Extent> s) noexcept {
482 return {reinterpret_cast<const byte*>(s.data()), s.size_bytes()};
483 }
484
485 template <class ElementType, size_t Extent,
486 typename std::enable_if<!std::is_const<ElementType>::value, int>::type = 0>
487 span<byte, ((Extent == dynamic_extent) ? dynamic_extent : sizeof(ElementType) * Extent)> as_writable_bytes(
488 span<ElementType, Extent> s) noexcept {
489 return {reinterpret_cast<byte*>(s.data()), s.size_bytes()};
490 }
491
492 template <std::size_t N, typename E, std::size_t S>
493 constexpr auto get(span<E, S> s) -> decltype(s[N]) {
494 return s[N];
495 }
496
497} // namespace TCB_SPAN_NAMESPACE_NAME
498
499namespace std {
500
501 template <typename ElementType, size_t Extent>
502 class tuple_size<TCB_SPAN_NAMESPACE_NAME::span<ElementType, Extent> > : public integral_constant<size_t, Extent> {};
503
504 template <typename ElementType>
505 class tuple_size<TCB_SPAN_NAMESPACE_NAME::span<ElementType, TCB_SPAN_NAMESPACE_NAME::dynamic_extent> >; // not
506 // defined
507
508 template <size_t I, typename ElementType, size_t Extent>
509 class tuple_element<I, TCB_SPAN_NAMESPACE_NAME::span<ElementType, Extent> > {
510 public:
511 static_assert(Extent != TCB_SPAN_NAMESPACE_NAME::dynamic_extent && I < Extent, "");
512 using type = ElementType;
513 };
514
515} // end namespace std
516
517/* revert TCB_SPAN_NAMESPACE_NAME to the original value, if needed */
518#ifdef TTG_TCB_SPAN_NAMESPACE_NAME_AT_TOP_OF_SPAN_H
519
520#undef TCB_SPAN_NAMESPACE_NAME
521#define TCB_SPAN_NAMESPACE_NAME TTG_TCB_SPAN_NAMESPACE_NAME_AT_TOP_OF_SPAN_H
522#undef TTG_TCB_SPAN_NAMESPACE_NAME_AT_TOP_OF_SPAN_H
523
524#endif
525
526#endif // TTG_UTIL_SPAN_H
constexpr span(Container &cont)
Definition span.h:330
TCB_SPAN_CONSTEXPR11 span< element_type, Count > last() const
Definition span.h:359
TCB_SPAN_ARRAY_CONSTEXPR span(const std::array< value_type, N > &arr) noexcept
Definition span.h:324
constexpr iterator end() const noexcept
Definition span.h:419
~span() noexcept=default
constexpr span(const Container &cont)
Definition span.h:337
std::reverse_iterator< iterator > reverse_iterator
Definition span.h:290
constexpr iterator begin() const noexcept
Definition span.h:417
TCB_SPAN_ARRAY_CONSTEXPR span(std::array< value_type, N > &arr) noexcept
Definition span.h:317
static constexpr size_type extent
Definition span.h:292
const element_type * const_pointer
Definition span.h:286
constexpr span(const span &other) noexcept=default
typename std::remove_cv< ElementType >::type value_type
Definition span.h:282
TCB_SPAN_ARRAY_CONSTEXPR reverse_iterator rbegin() const noexcept
Definition span.h:421
const element_type & const_reference
Definition span.h:288
std::ptrdiff_t difference_type
Definition span.h:284
TCB_SPAN_CONSTEXPR11 span< element_type, dynamic_extent > first(size_type count) const
Definition span.h:375
TCB_SPAN_CONSTEXPR11 reference back() const
Definition span.h:409
constexpr size_type size() const noexcept
Definition span.h:392
TCB_SPAN_CONSTEXPR11 reference front() const
Definition span.h:404
constexpr size_type size_bytes() const noexcept
Definition span.h:394
TCB_SPAN_CONSTEXPR11 reference operator[](size_type idx) const
Definition span.h:399
TCB_SPAN_CONSTEXPR11 subspan_return_t< Offset, Count > subspan() const
Definition span.h:370
constexpr span(element_type(&arr)[N]) noexcept
Definition span.h:310
TCB_SPAN_CONSTEXPR11 span< element_type, dynamic_extent > subspan(size_type offset, size_type count=dynamic_extent) const
Definition span.h:385
constexpr pointer data() const noexcept
Definition span.h:414
TCB_SPAN_CONSTEXPR11 span< element_type, dynamic_extent > last(size_type count) const
Definition span.h:380
TCB_SPAN_ARRAY_CONSTEXPR reverse_iterator rend() const noexcept
Definition span.h:423
constexpr span(const span< OtherElementType, OtherExtent > &other) noexcept
Definition span.h:345
element_type * pointer
Definition span.h:285
element_type & reference
Definition span.h:287
TCB_SPAN_NODISCARD constexpr bool empty() const noexcept
Definition span.h:396
typename std::remove_cv< typename std::remove_reference< T >::type >::type uncvref_t
Definition span.h:218
typename std::remove_pointer< T >::type remove_pointer_t
Definition span.h:247
constexpr auto data(C &c) -> decltype(c.data())
Definition span.h:190
span< byte,((Extent==dynamic_extent) ? dynamic_extent :sizeof(ElementType) *Extent)> as_writable_bytes(span< ElementType, Extent > s) noexcept
Definition span.h:487
constexpr span< ElementType, Extent > make_span(span< ElementType, Extent > s) noexcept
Definition span.h:450
span< const byte,((Extent==dynamic_extent) ? dynamic_extent :sizeof(ElementType) *Extent)> as_bytes(span< ElementType, Extent > s) noexcept
Definition span.h:480
constexpr auto get(span< E, S > s) -> decltype(s[N])
Definition span.h:493
unsigned char byte
Definition span.h:138
TCB_SPAN_INLINE_VAR constexpr std::size_t dynamic_extent
Definition span.h:147
STL namespace.
#define TCB_SPAN_EXPECT(cond)
Definition span.h:86
#define TCB_SPAN_INLINE_VAR
Definition span.h:92
#define TCB_SPAN_NODISCARD
Definition span.h:144
#define TCB_SPAN_ARRAY_CONSTEXPR
Definition span.h:132
#define TCB_SPAN_CONSTEXPR_ASSIGN
Definition span.h:108
#define TCB_SPAN_CONSTEXPR11
Definition span.h:112
constexpr span_storage() noexcept=default
static constexpr std::size_t size
Definition span.h:161