Sane C++ Libraries
C++ Platform Abstraction Libraries
Span.h
1// Copyright (c) Stefano Cristiano
2// SPDX-License-Identifier: MIT
3#pragma once
4#include "../Foundation/InitializerList.h"
5#include "../Foundation/LibC.h" // memcmp
6#include "../Foundation/TypeTraits.h" // SameConstnessAs
7
8namespace SC
9{
10template <typename Type>
11struct Span;
12
13struct SpanStringView;
14struct SpanString;
15namespace detail
16{
17// clang-format off
18template<typename U> struct SpanSizeOfType { static constexpr auto size = sizeof(U); };
19template<> struct SpanSizeOfType<void> { static constexpr auto size = 1; };
20template<> struct SpanSizeOfType<const void> { static constexpr auto size = 1; };
21// clang-format on
22} // namespace detail
23} // namespace SC
24
27
30template <typename Type>
32{
33 private:
34 // clang-format off
35 using SizeType = size_t;
36 using VoidType = typename TypeTraits::SameConstnessAs<Type, void>::type;
37 template <typename U> using TypeIfNotVoid = typename TypeTraits::EnableIf<not TypeTraits::IsSame<U, VoidType>::value, Type>::type;
38 template <typename U> using SameType = TypeTraits::IsSame<typename TypeTraits::RemoveConst<U>::type, typename TypeTraits::RemoveConst<Type>::type>;
39 template <typename U> using EnableNotVoid = typename TypeTraits::EnableIf<SameType<U>::value and not TypeTraits::IsSame<U, VoidType>::value, bool>::type;
40 // clang-format on
41 Type* items;
42 SizeType sizeElements;
43
44 public:
45 template <size_t N, typename U = Type, EnableNotVoid<U> = true>
46 constexpr Span(U (&itemsArray)[N]) : items(itemsArray), sizeElements(N)
47 {}
48
50 constexpr Span() : items(nullptr), sizeElements(0) {}
51
55 constexpr Span(Type* items, SizeType sizeInElements) : items(items), sizeElements(sizeInElements) {}
56
59 template <typename U = Type>
60 constexpr Span(TypeIfNotVoid<U>& type) : items(&type), sizeElements(1)
61 {}
62
65 template <typename U = Type>
66 constexpr Span(std::initializer_list<TypeIfNotVoid<U>> list) : items(nullptr), sizeElements(0)
67 {
68 // We need this two step initialization to avoid warnings on all compilers
69 items = list.begin();
70 sizeElements = list.size();
71 }
72
73 // clang-format off
74 template <typename U = Type> operator Span<const TypeIfNotVoid<U>>() const { return {items, sizeElements}; }
75 template <typename U = Type> operator Span< TypeIfNotVoid<U>>() { return {items, sizeElements}; }
76 operator Span<const void>() const { return Span<const void>(items, sizeElements * detail::SpanSizeOfType<Type>::size); }
77 operator Span<void>() { return Span<void>(items, sizeElements * detail::SpanSizeOfType<Type>::size); }
78 // clang-format on
79
84 template <typename T>
85 [[nodiscard]] static Span<Type> reinterpret_object(T& value)
86 {
87 return {reinterpret_cast<Type*>(&value), sizeof(T) / detail::SpanSizeOfType<Type>::size};
88 }
89
94 [[nodiscard]] static Span<Type> reinterpret_bytes(VoidType* rawMemory, SizeType sizeInBytes)
95 {
96 return Span(reinterpret_cast<Type*>(rawMemory), sizeInBytes / detail::SpanSizeOfType<Type>::size);
97 }
98
100 template <typename T>
102 {
103 return Span<const T>(reinterpret_cast<const T*>(items), sizeInBytes() / sizeof(T));
104 }
105
107 template <typename T>
109 {
110 return Span<T>(reinterpret_cast<T*>(items), sizeInBytes() / sizeof(T));
111 }
112
115 [[nodiscard]] constexpr const Type* begin() const { return items; }
116
119 [[nodiscard]] constexpr const Type* end() const { return items + sizeElements; }
120
123 [[nodiscard]] constexpr const Type* data() const { return items; }
124
127 [[nodiscard]] constexpr Type* begin() { return items; }
128
131 [[nodiscard]] constexpr Type* end() { return items + sizeElements; }
132
135 [[nodiscard]] constexpr Type* data() { return items; }
136
139 [[nodiscard]] constexpr SizeType sizeInElements() const { return sizeElements; }
140
143 [[nodiscard]] constexpr SizeType sizeInBytes() const { return sizeElements * detail::SpanSizeOfType<Type>::size; }
144
151 [[nodiscard]] constexpr bool sliceStart(SizeType offsetInElements, Span& destination) const
152 {
153 if (offsetInElements <= sizeInElements())
154 {
155 destination = Span(items + offsetInElements, (sizeInElements() - offsetInElements));
156 return true;
157 }
158 return false;
159 }
160
168 [[nodiscard]] constexpr bool sliceStartLength(SizeType offsetInElements, SizeType lengthInElements,
169 Span& destination) const
170 {
171 if (offsetInElements + lengthInElements <= sizeInElements())
172 {
173 destination = Span(items + offsetInElements, lengthInElements);
174 return true;
175 }
176 return false;
177 }
178
182 [[nodiscard]] bool sliceFromStartUntil(Span other, Span& output) const
183 {
184 const auto diff = other.items - items;
185 if (diff < 0 or static_cast<SizeType>(diff) > sizeInBytes())
186 {
187 return false;
188 }
189 else
190 {
191 output = Span(items, static_cast<SizeType>(diff) / detail::SpanSizeOfType<Type>::size);
192 return true;
193 }
194 }
195
198 [[nodiscard]] constexpr bool empty() const { return sizeElements == 0; }
199
206 template <typename U>
207 [[nodiscard]] constexpr bool contains(const U& value, SizeType* index = nullptr) const
208 {
209 return find([&](auto& current) { return current == value; }, index);
210 }
211
217 template <typename Lambda>
218 [[nodiscard]] constexpr bool find(Lambda&& lambda, SizeType* index = nullptr) const
219 {
220 for (SizeType idx = 0; idx < sizeElements; ++idx)
221 {
222 if (lambda(items[idx]))
223 {
224 if (index)
225 {
226 *index = idx;
227 }
228 return true;
229 }
230 }
231 return false;
232 }
233
234 // clang-format off
235 template <typename U = Type> TypeIfNotVoid<U>& operator[](SizeType idx) { return items[idx]; }
236 template <typename U = Type> const TypeIfNotVoid<U>& operator[](SizeType idx) const { return items[idx]; }
237 // clang-format on
238
240 template <typename IntType>
241 Type* get(IntType idx)
242 {
243 if (idx >= 0 and idx < static_cast<IntType>(sizeElements))
244 return items + idx;
245 return nullptr;
246 }
247
249 template <typename IntType>
250 const Type* get(IntType idx) const
251 {
252 if (idx >= 0 and idx < static_cast<IntType>(sizeElements))
253 return items + idx;
254 return nullptr;
255 }
256
258 template <typename U>
259 [[nodiscard]] bool memcmpWith(const Span<U> other) const
260 {
261 if (sizeInBytes() != other.sizeInBytes())
262 return false;
263 if (sizeInBytes() == 0)
264 return true;
265 return ::memcmp(items, other.data(), sizeInBytes()) == 0;
266 }
267
269 template <typename U>
270 [[nodiscard]] bool memcpyTo(Span<U>& other) const
271 {
272 if (other.sizeInBytes() < sizeInBytes())
273 return false;
274 ::memcpy(other.data(), items, sizeInBytes());
275 other = {other.data(), sizeInBytes() / sizeof(U)};
276 return true;
277 }
278};
279
280#if SC_PLATFORM_WINDOWS
281#define SC_NATIVE_STR(str) L##str
282#else
283#define SC_NATIVE_STR(str) str
284#endif
285
288{
289 constexpr SpanStringView() = default;
290 constexpr SpanStringView(const char* string, size_t stringLength) : text(string, stringLength) {}
291 template <size_t N>
292 constexpr SpanStringView(const char (&charLiteral)[N]) : text(charLiteral)
293 {}
294
296 template <int N>
297 [[nodiscard]] bool writeNullTerminated(char (&buffer)[N]) const
298 {
299 if (N < text.sizeInElements() + 1)
300 return false;
301 ::memcpy(&buffer[0], text.data(), text.sizeInElements());
302 buffer[N - 1] = 0;
303 return true;
304 }
305
306 Span<const char> text;
307};
308
311{
312 template <size_t N>
313 constexpr SpanString(char (&buffer)[N]) : text(buffer)
314 {}
315
316 operator SpanStringView() const
317 {
319 sv.text = text;
320 return sv;
321 }
322
323 SC::Span<char> text;
324};
unsigned long size_t
Platform independent unsigned size type.
Definition: PrimitiveTypes.h:56
View over a contiguous sequence of items (pointer + size in elements).
Definition: Span.h:32
static Span< Type > reinterpret_object(T &value)
Constructs a Span reinterpreting memory pointed by object of type T as a type Type
Definition: Span.h:85
constexpr bool find(Lambda &&lambda, SizeType *index=nullptr) const
Finds the first item in span matching criteria given by the lambda.
Definition: Span.h:218
constexpr const Type * data() const
Returns pointer to first element of the span.
Definition: Span.h:123
static Span< Type > reinterpret_bytes(VoidType *rawMemory, SizeType sizeInBytes)
Construct a span reinterpreting raw memory (void* or const void*) to Type or const Type
Definition: Span.h:94
const Type * get(IntType idx) const
Gets the item at given index or nullptr if index is negative or bigger than size.
Definition: Span.h:250
constexpr SizeType sizeInBytes() const
Size of Span in bytes.
Definition: Span.h:143
bool memcmpWith(const Span< U > other) const
Compares this span with another one byte by byte.
Definition: Span.h:259
Span< T > reinterpret_as_array_of()
Reinterprets the current span as an array of the specified type.
Definition: Span.h:108
constexpr bool sliceStart(SizeType offsetInElements, Span &destination) const
Creates another Span, starting at an offset in elements from current Span, until end.
Definition: Span.h:151
constexpr bool contains(const U &value, SizeType *index=nullptr) const
Check if the current span contains a given value.
Definition: Span.h:207
constexpr Span()
Builds an empty Span.
Definition: Span.h:50
constexpr Type * begin()
Returns pointer to first element of the span.
Definition: Span.h:127
constexpr Type * end()
Returns pointer to one after the last element of the span.
Definition: Span.h:131
constexpr const Type * begin() const
Returns pointer to first element of the span.
Definition: Span.h:115
Type * get(IntType idx)
Gets the item at given index or nullptr if index is negative or bigger than size.
Definition: Span.h:241
constexpr bool sliceStartLength(SizeType offsetInElements, SizeType lengthInElements, Span &destination) const
Creates another Span, starting at an offset in elements from current Span of specified length.
Definition: Span.h:168
bool memcpyTo(Span< U > &other) const
Bitwise copies contents of this Span over another (non-overlapping)
Definition: Span.h:270
constexpr Span(Type *items, SizeType sizeInElements)
Builds a Span from an array.
Definition: Span.h:55
Span< const T > reinterpret_as_array_of() const
Reinterprets the current span as an array of the specified type.
Definition: Span.h:101
constexpr Type * data()
Returns pointer to first element of the span.
Definition: Span.h:135
bool sliceFromStartUntil(Span other, Span &output) const
Creates another Span shorter or equal than the current one such that its end equals other....
Definition: Span.h:182
constexpr bool empty() const
Check if Span is empty.
Definition: Span.h:198
constexpr SizeType sizeInElements() const
Size of Span in elements.
Definition: Span.h:139
constexpr Span(std::initializer_list< TypeIfNotVoid< U > > list)
Span specialized constructor (mainly used for converting const char* to StringView)
Definition: Span.h:66
constexpr const Type * end() const
Returns pointer to one after the last element of the span.
Definition: Span.h:119
constexpr Span(TypeIfNotVoid< U > &type)
Builds a Span from a single object.
Definition: Span.h:60
An writable view over an ASCII string (to avoid including Strings library)
Definition: Span.h:311
An read-only view over an ASCII string (to avoid including Strings library)
Definition: Span.h:288
bool writeNullTerminated(char(&buffer)[N]) const
Writes current string view over a sized char array buffer, adding a null terminator.
Definition: Span.h:297
EnableIf conditionally defines a type if a boolean template parameter is true.
Definition: TypeTraits.h:25
IsSame evaluates to true if the provided types T and U are the same, false otherwise.
Definition: TypeTraits.h:29