4#include "../Foundation/Assert.h"
5#include "../Foundation/Globals.h"
6#include "../Foundation/Memory.h"
7#include "ArenaMapKey.h"
32 using Gen =
typename Key::Generation;
38 ArenaMap(
const ArenaMap& other) : globalsType(other.globalsType) { *
this = other; }
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)));
50 ::memset(newGens, 0, other.itemsSize *
sizeof(Gen));
52 generations = newGens;
53 itemsSize = other.itemsSize;
54 for (
size_t idx = 0; idx < other.itemsSize; ++idx)
56 if (other.generations[idx].used)
58 placementNew(items[idx], other.items[idx]);
60 generations[idx] = other.generations[idx];
62 numUsed = other.numUsed;
70 itemsSize = other.itemsSize;
71 other.items =
nullptr;
73 generations = other.generations;
74 other.generations =
nullptr;
75 numUsed = other.numUsed;
83 template <
typename MapType>
86 MapType* map =
nullptr;
91 const auto numAllocated = map->getNumAllocated();
92 for (++index; index < numAllocated; ++index)
94 if (map->generations[index].used)
104 return it.index == index;
109 return it.index != index;
112 auto& operator*()
const {
return map->items[index]; }
113 auto* operator->()
const {
return &map->items[index]; }
119 ConstIterator cend()
const {
return end(); }
120 ConstIterator begin()
const
124 if (generations[idx].used)
126 return {
this,
static_cast<uint32_t>(idx)};
138 if (generations[idx].used)
140 return {
this,
static_cast<uint32_t>(idx)};
150 for (
size_t idx = 0; idx < itemsSize; ++idx)
152 if (generations[idx].used)
156 generations[idx].used = 0;
162 generations =
nullptr;
167 [[nodiscard]]
size_t size()
const {
return numUsed; }
170 [[nodiscard]]
size_t capacity()
const {
return itemsSize; }
173 [[nodiscard]]
bool isFull()
const {
return itemsSize == numUsed; }
179 [[nodiscard]]
bool resize(
size_t newSize)
183 if (newSize > Key::MaxIndex)
189 generations =
nullptr;
191 T* newItems =
reinterpret_cast<T*
>(allocator.allocate(
this, newSize *
sizeof(T),
alignof(T)));
192 if (newItems ==
nullptr)
195 Gen* newGens =
reinterpret_cast<Gen*
>(allocator.allocate(
this, newSize *
sizeof(Gen),
alignof(Gen)));
196 if (newGens ==
nullptr)
198 ::memset(newGens, 0, newSize *
sizeof(Gen));
199 generations = newGens;
205 template <
typename Value>
206 [[nodiscard]] Key insert(
const Value&
object)
208 Key key = allocateNewKeySlot();
211 placementNew(items[key.index],
object);
216 [[nodiscard]] Key allocate()
218 Key key = allocateNewKeySlot();
221 placementNew(items[key.index]);
226 template <
typename Value>
227 [[nodiscard]] Key insert(Value&&
object)
229 Key key = allocateNewKeySlot();
232 placementNew(items[key.index],
move(
object));
237 [[nodiscard]]
bool containsKey(Key key)
const
239 return key.isValid() and generations[key.index].used != 0 and
240 generations[key.index].generation == key.generation.generation;
243 template <
typename ComparableToValue>
244 [[nodiscard]]
bool containsValue(
const ComparableToValue& value, Key* optionalKey =
nullptr)
const
246 for (
size_t idx = 0; idx < itemsSize; ++idx)
248 if (generations[idx].used and items[idx] == value)
252 optionalKey->index =
static_cast<uint32_t>(idx);
253 optionalKey->generation = generations[idx];
261 [[nodiscard]]
bool remove(Key key)
263 if (generations[key.index] != key.generation)
265 if (generations[key.index].generation + 1 <= Key::MaxGenerations)
267 generations[key.index].generation++;
272 generations[key.index].generation++;
274 generations[key.index].used = 0;
275 items[key.index].~T();
280 [[nodiscard]] T* get(Key key)
282 if (generations[key.index] != key.generation)
284 return &items[key.index];
287 [[nodiscard]]
const T* get(Key key)
const
289 if (generations[key.index] != key.generation)
291 return &items[key.index];
296 size_t itemsSize = 0;
298 Gen* generations =
nullptr;
304 [[nodiscard]] Key allocateNewKeySlot()
306 for (
size_t idx = 0; idx < itemsSize; ++idx)
308 if (generations[idx].used == 0)
310 generations[idx].used = 1;
313 key.generation = generations[idx];
314 key.index =
static_cast<uint32_t>(idx);
#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