Sane C++ Libraries
C++ Platform Abstraction Libraries
Vector.h
1// Copyright (c) Stefano Cristiano
2// SPDX-License-Identifier: MIT
3#pragma once
4#include "../Algorithms/AlgorithmRemove.h" // removeIf
5#include "../Foundation/Internal/Segment.inl"
6#include "../Foundation/Internal/SegmentTrivial.inl"
7#include "../Foundation/TypeTraits.h" // IsTriviallyCopyable
8
9namespace SC
10{
11namespace detail
12{
13
14template <typename T, bool isTrivial = TypeTraits::IsTriviallyCopyable<T>::value>
15struct SegmentVTable : public SegmentTrivial<T>
16{
17};
18
19template <typename T>
20struct SegmentVTable<T, false>
21{
22 static void destruct(Span<T> data) noexcept
23 {
24 forEach(data, [](auto, T& item) { item.~T(); });
25 }
26
27 template <typename U>
28 static void copyConstructAs(Span<T> data, Span<const U> value) noexcept
29 {
30 const U& single = value[0];
31 forEach(data, [&single](auto, T& item) { placementNew(item, single); });
32 }
33
34 template <typename U>
35 static void copyConstruct(Span<T> data, const U* src) noexcept
36 {
37 forEach(data, [src](auto idx, T& item) { placementNew(item, src[idx]); });
38 }
39
40 template <typename U>
41 static void copyAssign(Span<T> data, const U* src) noexcept
42 {
43 forEach(data, [src](auto idx, T& item) { item = src[idx]; });
44 }
45
46 template <typename U>
47 static void moveConstruct(Span<T> data, U* src) noexcept
48 {
49 forEach(data, [src](auto idx, T& item) { placementNew(item, move(src[idx])); });
50 }
51
52 template <typename U>
53 static void moveAssign(Span<T> data, U* src) noexcept
54 {
55 forEach(data, [src](auto idx, T& item) { item = move(src[idx]); });
56 }
57
58 template <typename U>
59 static void copyInsert(Span<T> headerData, Span<const U> values) noexcept
60 {
61 // This function inserts values at the beginning of headerData
62 T* data = headerData.data();
63 const U* src = values.data();
64
65 const size_t numElements = headerData.sizeInElements();
66 const size_t numToInsert = values.sizeInElements();
67
68 if (numElements == 0)
69 {
70 // All newly inserted elements will be copy-constructed in the uninitialized area
71 for (size_t idx = 0; idx < numToInsert; ++idx)
72 {
73 placementNew(data[idx], src[idx]);
74 }
75 }
76 else
77 {
78 // Move construct some elements in the not initialized area
79 for (size_t idx = numElements; idx < numElements + numToInsert; ++idx)
80 {
81 // Guard against using slots that are before segment start.
82 // In the last loop at end of this scope such slots must be
83 // initialized with placement new instead of assignment operator
84 if (idx >= numToInsert)
85 {
86 placementNew(data[idx], move(data[idx - numToInsert]));
87 }
88 }
89
90 // Move assign some elements to slots in "post-move" state left from previous loop
91 for (size_t idx = numElements - 1; idx >= numToInsert; --idx)
92 {
93 if (idx >= numToInsert)
94 {
95 data[idx] = move(data[idx - numToInsert]);
96 }
97 }
98
99 // Copy assign source data to slots in "post-move" state left from previous loop
100 for (size_t idx = 0; idx < numToInsert; ++idx)
101 {
102 // See note in the first loop in this scope to understand use of assignment vs. placement new
103 if (idx < numElements)
104 {
105 data[idx] = src[idx];
106 }
107 else
108 {
109 placementNew(data[idx], src[idx]);
110 }
111 }
112 }
113 }
114
115 static void remove(Span<T> headerData, size_t numToRemove) noexcept
116 {
117 T* data = headerData.data();
118
119 const size_t numElements = headerData.sizeInElements();
120
121 for (size_t idx = 0; idx < numElements - numToRemove; ++idx)
122 {
123 data[idx] = move(data[idx + numToRemove]);
124 }
125 for (size_t idx = numElements - numToRemove; idx < numElements; ++idx)
126 {
127 data[idx].~T();
128 }
129 }
130
131 private:
132 template <typename Lambda>
133 static void forEach(Span<T> data, Lambda&& lambda) noexcept
134 {
135 const size_t numElements = data.sizeInElements();
136 T* elements = data.data();
137 for (size_t idx = 0; idx < numElements; ++idx)
138 {
139 lambda(idx, elements[idx]);
140 }
141 }
142};
143
144// Executes operations using trivial or non trivial variations of the move/copy/construct functions.
145// Trivial check cannot be a template param of this class because it would cause Vector to need
146// the entire definition of T to declare Vector<T>, making it impossible to create "recursive" structures
147// like SC::Build::WriteInternal::RenderGroup (that has a "Vector<RenderGroup> children" as member )
148template <typename T>
149struct ObjectVTable
150{
151 using Type = T;
152 static void destruct(Span<T> data) noexcept { SegmentVTable<T>::destruct(data); }
153 // clang-format off
154 template <typename U> static void copyConstructAs(Span<T> data, Span<const U> value) noexcept { SegmentVTable<T>::template copyConstructAs<U>(data, value);}
155 template <typename U> static void copyConstruct(Span<T> data, const U* src) noexcept { SegmentVTable<T>::template copyConstruct<U>(data, src);}
156 template <typename U> static void copyAssign(Span<T> data, const U* src) noexcept { SegmentVTable<T>::template copyAssign<U>(data, src);}
157 template <typename U> static void moveConstruct(Span<T> data, U* src) noexcept { SegmentVTable<T>::template moveConstruct<U>(data, src);}
158 template <typename U> static void moveAssign(Span<T> data, U* src) noexcept { SegmentVTable<T>::template moveAssign<U>(data, src);}
159 template <typename U> static void copyInsert(Span<T> data, Span<const U> values) noexcept { SegmentVTable<T>::template copyInsert<U>(data, values);}
160 // clang-format on
161 static void remove(Span<T> data, size_t numElements) noexcept { SegmentVTable<T>::remove(data, numElements); }
162};
163
164template <typename T>
165struct VectorVTable : public ObjectVTable<T>, public SegmentSelfRelativePointer<T>
166{
167 static constexpr bool IsArray = false;
168};
169} // namespace detail
170
173
176
187template <typename T>
188struct Vector : public Segment<detail::VectorVTable<T>>
189{
191
192 // Inherits all constructors from Segment
193 using Parent::Parent;
194
201 template <typename U>
202 [[nodiscard]] bool contains(const U& value, size_t* index = nullptr) const noexcept
203 {
204 return Parent::toSpanConst().contains(value, index);
205 }
206
212 template <typename Lambda>
213 [[nodiscard]] bool find(Lambda&& lambda, size_t* index = nullptr) const noexcept
214 {
215 return Parent::toSpanConst().find(move(lambda), index);
216 }
217
222 template <typename Lambda>
223 [[nodiscard]] bool removeAll(Lambda&& criteria) noexcept
224 {
225 T* itBeg = Parent::begin();
226 T* itEnd = Parent::end();
227 T* it = Algorithms::removeIf(itBeg, itEnd, forward<Lambda>(criteria));
228
229 const size_t numElements = static_cast<size_t>(itEnd - it);
230 const size_t offset = static_cast<size_t>(it - itBeg);
231 detail::VectorVTable<T>::destruct({Parent::data() + offset, numElements});
232 Parent::header.sizeBytes -= static_cast<decltype(Parent::header.sizeBytes)>(numElements * sizeof(T));
233 return it != itEnd;
234 }
235
240 template <typename U>
241 [[nodiscard]] bool remove(const U& value) noexcept
242 {
243 return removeAll([&](auto& item) { return item == value; });
244 }
245};
246
258template <typename T, int N>
259struct SmallVector : public Vector<T>
260{
261 // clang-format off
262 SmallVector(SegmentAllocator allocator = SegmentAllocator::Global) noexcept : Vector<T>( N * sizeof(T), allocator) {}
263 ~SmallVector() noexcept {}
264 SmallVector(const SmallVector& other) noexcept : SmallVector() { Vector<T>::operator=(other); }
265 SmallVector(SmallVector&& other) noexcept : SmallVector() { Vector<T>::operator=(move(other)); }
266 SmallVector& operator=(const SmallVector& other) noexcept { Vector<T>::operator=(other); return *this; }
267 SmallVector& operator=(SmallVector&& other) noexcept { Vector<T>::operator=(move(other)); return *this; }
268
269 SmallVector(const Vector<T>& other) noexcept : SmallVector() { Vector<T>::operator=(other); }
270 SmallVector(Vector<T>&& other) noexcept : SmallVector() { Vector<T>::operator=(move(other)); }
271 SmallVector(std::initializer_list<T> list) noexcept : SmallVector() { SC_ASSERT_RELEASE(Vector<T>::assign({list.begin(), list.size()})); }
272 // clang-format on
273 protected:
274 SmallVector(int num, SegmentAllocator allocator) : Vector<T>(N, allocator) { (void)num; }
275
276 private:
277 uint64_t inlineCapacity = N * sizeof(T);
278 union
279 {
280 T inlineData[N];
281 };
282};
283
284template <typename T>
285using VectorTL = detail::SegmentCustom<Vector<T>, Vector<T>, 0, SegmentAllocator::ThreadLocal>;
286template <typename T, int N>
287using SmallVectorTL = detail::SegmentCustom<SmallVector<T, N>, Vector<T>, N, SegmentAllocator::ThreadLocal>;
288
290
291} // namespace SC
ForwardIterator removeIf(ForwardIterator first, ForwardIterator last, UnaryPredicate &&predicate)
Removes all items in the given range, satisfying the given predicate.
Definition: AlgorithmRemove.h:22
#define SC_ASSERT_RELEASE(e)
Assert expression e to be true.
Definition: Assert.h:66
constexpr T && move(T &value)
Converts an lvalue to an rvalue reference.
Definition: Compiler.h:269
unsigned long long uint64_t
Platform independent (8) bytes unsigned int.
Definition: PrimitiveTypes.h:42
A slice of contiguous memory, prefixed by and header containing size and capacity.
Definition: Segment.h:113
Span< const T > toSpanConst() const noexcept SC_LANGUAGE_LIFETIME_BOUND
Definition: Segment.h:206
A Vector that can hold up to N elements inline and > N on heap.
Definition: Vector.h:260
A contiguous sequence of heap allocated elements.
Definition: Vector.h:189
bool remove(const U &value) noexcept
Removes all values equal to value
Definition: Vector.h:241
bool find(Lambda &&lambda, size_t *index=nullptr) const noexcept
Finds the first item in array matching criteria given by the lambda.
Definition: Vector.h:213
bool contains(const U &value, size_t *index=nullptr) const noexcept
Check if the current array contains a given value.
Definition: Vector.h:202
bool removeAll(Lambda &&criteria) noexcept
Removes all items matching criteria given by Lambda.
Definition: Vector.h:223