Sane C++ Libraries
C++ Platform Abstraction Libraries
All Classes Namespaces Functions Variables Typedefs Enumerations Enumerator Modules Pages
ArenaMap.h
1// Copyright (c) Stefano Cristiano
2// SPDX-License-Identifier: MIT
3#pragma once
4#include "../Foundation/Assert.h"
5#include "../Foundation/Globals.h"
6#include "../Foundation/Memory.h"
7#include "ArenaMapKey.h"
8
9namespace SC
10{
11template <typename T>
12struct ArenaMap;
13} // namespace SC
14
17
28template <typename T>
30{
31 using Key = ArenaMapKey<T>;
32 using Gen = typename Key::Generation;
33
34 ArenaMap(Globals::Type globalsType = Globals::Global) : globalsType(globalsType) {}
35
36 ~ArenaMap() { clear(); }
37
38 ArenaMap(const ArenaMap& other) : globalsType(other.globalsType) { *this = other; }
39
40 ArenaMap(ArenaMap&& other) : globalsType(other.globalsType) { *this = move(other); }
41
42 ArenaMap& operator=(const ArenaMap& other)
43 {
44 clear();
45 auto& allocator = Globals::get(globalsType).allocator;
46 T* newItems = reinterpret_cast<T*>(allocator.allocate(this, other.itemsSize * sizeof(T), alignof(T)));
47 Gen* newGens = reinterpret_cast<Gen*>(allocator.allocate(this, other.itemsSize * sizeof(Gen), alignof(Gen)));
48 SC_ASSERT_RELEASE(newItems);
49 SC_ASSERT_RELEASE(newGens);
50 ::memset(newGens, 0, other.itemsSize * sizeof(Gen));
51 items = newItems;
52 generations = newGens;
53 itemsSize = other.itemsSize;
54 for (size_t idx = 0; idx < other.itemsSize; ++idx)
55 {
56 if (other.generations[idx].used)
57 {
58 placementNew(items[idx], other.items[idx]);
59 }
60 generations[idx] = other.generations[idx];
61 }
62 numUsed = other.numUsed;
63 return *this;
64 }
65
66 ArenaMap& operator=(ArenaMap&& other)
67 {
68 clear();
69 items = other.items;
70 itemsSize = other.itemsSize;
71 other.items = nullptr;
72 other.itemsSize = 0;
73 generations = other.generations;
74 other.generations = nullptr;
75 numUsed = other.numUsed;
76 other.numUsed = 0;
77 return *this;
78 }
79
81 uint32_t getNumAllocated() const { return static_cast<uint32_t>(itemsSize); }
82
83 template <typename MapType>
85 {
86 MapType* map = nullptr;
87 uint32_t index = 0;
88
89 void operator++()
90 {
91 const auto numAllocated = map->getNumAllocated();
92 for (++index; index < numAllocated; ++index)
93 {
94 if (map->generations[index].used)
95 {
96 break;
97 }
98 }
99 }
100
101 bool operator==(ArenaMapIterator it) const
102 {
103 SC_ASSERT_DEBUG(it.map == map and map != nullptr);
104 return it.index == index;
105 }
106 bool operator!=(ArenaMapIterator it) const
107 {
108 SC_ASSERT_DEBUG(it.map == map and map != nullptr);
109 return it.index != index;
110 }
111
112 auto& operator*() const { return map->items[index]; }
113 auto* operator->() const { return &map->items[index]; }
114 };
117
118 ConstIterator cbegin() const { return begin(); }
119 ConstIterator cend() const { return end(); }
120 ConstIterator begin() const
121 {
122 for (size_t idx = 0; idx < getNumAllocated(); ++idx)
123 {
124 if (generations[idx].used)
125 {
126 return {this, static_cast<uint32_t>(idx)};
127 }
128 }
129 return end();
130 }
131
132 ConstIterator end() const { return {this, getNumAllocated()}; }
133
134 Iterator begin()
135 {
136 for (size_t idx = 0; idx < getNumAllocated(); ++idx)
137 {
138 if (generations[idx].used)
139 {
140 return {this, static_cast<uint32_t>(idx)};
141 }
142 }
143 return end();
144 }
145
146 Iterator end() { return {this, getNumAllocated()}; }
147
148 void clear()
149 {
150 for (size_t idx = 0; idx < itemsSize; ++idx)
151 {
152 if (generations[idx].used)
153 {
154 items[idx].~T();
155 }
156 generations[idx].used = 0;
157 }
158 Globals::get(globalsType).allocator.release(items);
159 items = nullptr;
160 itemsSize = 0;
161 Globals::get(globalsType).allocator.release(generations);
162 generations = nullptr;
163 numUsed = 0;
164 }
165
167 [[nodiscard]] size_t size() const { return numUsed; }
168
170 [[nodiscard]] size_t capacity() const { return itemsSize; }
171
173 [[nodiscard]] bool isFull() const { return itemsSize == numUsed; }
174
179 [[nodiscard]] bool resize(size_t newSize)
180 {
181 if (numUsed != 0)
182 return false;
183 if (newSize > Key::MaxIndex)
184 return false;
185
186 Globals::get(globalsType).allocator.release(items);
187 items = nullptr;
188 Globals::get(globalsType).allocator.release(generations);
189 generations = nullptr;
190 auto& allocator = Globals::get(globalsType).allocator;
191 T* newItems = reinterpret_cast<T*>(allocator.allocate(this, newSize * sizeof(T), alignof(T)));
192 if (newItems == nullptr)
193 return false;
194 items = newItems;
195 Gen* newGens = reinterpret_cast<Gen*>(allocator.allocate(this, newSize * sizeof(Gen), alignof(Gen)));
196 if (newGens == nullptr)
197 return false;
198 ::memset(newGens, 0, newSize * sizeof(Gen));
199 generations = newGens;
200 itemsSize = newSize;
201 numUsed = 0;
202 return true;
203 }
204
205 template <typename Value>
206 [[nodiscard]] Key insert(const Value& object)
207 {
208 Key key = allocateNewKeySlot();
209 if (key.isValid())
210 {
211 placementNew(items[key.index], object);
212 }
213 return key;
214 }
215
216 [[nodiscard]] Key allocate()
217 {
218 Key key = allocateNewKeySlot();
219 if (key.isValid())
220 {
221 placementNew(items[key.index]);
222 }
223 return key;
224 }
225
226 template <typename Value>
227 [[nodiscard]] Key insert(Value&& object)
228 {
229 Key key = allocateNewKeySlot();
230 if (key.isValid())
231 {
232 placementNew(items[key.index], move(object));
233 }
234 return key;
235 }
236
237 [[nodiscard]] bool containsKey(Key key) const
238 {
239 return key.isValid() and generations[key.index].used != 0 and
240 generations[key.index].generation == key.generation.generation;
241 }
242
243 template <typename ComparableToValue>
244 [[nodiscard]] bool containsValue(const ComparableToValue& value, Key* optionalKey = nullptr) const
245 {
246 for (size_t idx = 0; idx < itemsSize; ++idx)
247 {
248 if (generations[idx].used and items[idx] == value)
249 {
250 if (optionalKey)
251 {
252 optionalKey->index = static_cast<uint32_t>(idx);
253 optionalKey->generation = generations[idx];
254 }
255 return true;
256 }
257 }
258 return false;
259 }
260
261 [[nodiscard]] bool remove(Key key)
262 {
263 if (generations[key.index] != key.generation)
264 return false;
265 if (generations[key.index].generation + 1 <= Key::MaxGenerations)
266 {
267 generations[key.index].generation++;
268 }
269 else
270 {
271 // TODO: Should we give option of failing on generation overflow?
272 generations[key.index].generation++;
273 }
274 generations[key.index].used = 0;
275 items[key.index].~T();
276 numUsed--;
277 return true;
278 }
279
280 [[nodiscard]] T* get(Key key)
281 {
282 if (generations[key.index] != key.generation)
283 return nullptr;
284 return &items[key.index];
285 }
286
287 [[nodiscard]] const T* get(Key key) const
288 {
289 if (generations[key.index] != key.generation)
290 return nullptr;
291 return &items[key.index];
292 }
293
294 private:
295 T* items = nullptr;
296 size_t itemsSize = 0;
297
298 Gen* generations = nullptr;
299
300 uint32_t numUsed = 0;
301
302 Globals::Type globalsType;
303
304 [[nodiscard]] Key allocateNewKeySlot()
305 {
306 for (size_t idx = 0; idx < itemsSize; ++idx)
307 {
308 if (generations[idx].used == 0)
309 {
310 generations[idx].used = 1;
311
312 Key key;
313 key.generation = generations[idx];
314 key.index = static_cast<uint32_t>(idx);
315 numUsed++;
316 return key;
317 }
318 }
319 return {};
320 }
321};
#define SC_ASSERT_DEBUG(e)
Assert expression e to be true.
Definition: Assert.h:82
#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 int uint32_t
Platform independent (4) bytes unsigned int.
Definition: PrimitiveTypes.h:38
Definition: ArenaMap.h:85
A sparse vector keeping objects at a stable memory location.
Definition: ArenaMap.h:30
bool resize(size_t newSize)
Changes the size of the arena.
Definition: ArenaMap.h:179
size_t size() const
Get the number of used slots in the arena.
Definition: ArenaMap.h:167
uint32_t getNumAllocated() const
Get the maximum number of objects that can be stored in this map.
Definition: ArenaMap.h:81
size_t capacity() const
Get the total number slots in the arena.
Definition: ArenaMap.h:170
bool isFull() const
Returns true if size() == capacity(), that means the arena is full.
Definition: ArenaMap.h:173
A sparse vector keeping objects at a stable memory location.
Definition: ArenaMapKey.h:22
static Globals & get(Type type)
Obtains current set of Globals.
Type
Definition: Globals.h:35
@ Global
Shared globals (NOT thread-safe)
Definition: Globals.h:36
void release(void *memory)
Free memory allocated by MemoryAllocator::allocate and / or reallocated by MemoryAllocator::reallocat...
Definition: Memory.h:81