2020-01-02 20:31:05 +00:00
|
|
|
#include "voxel_memory_pool.h"
|
|
|
|
#include "core/print_string.h"
|
|
|
|
#include "core/variant.h"
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
VoxelMemoryPool *g_memory_pool = nullptr;
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
void VoxelMemoryPool::create_singleton() {
|
|
|
|
CRASH_COND(g_memory_pool != nullptr);
|
|
|
|
g_memory_pool = memnew(VoxelMemoryPool);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelMemoryPool::destroy_singleton() {
|
|
|
|
CRASH_COND(g_memory_pool == nullptr);
|
|
|
|
VoxelMemoryPool *pool = g_memory_pool;
|
|
|
|
g_memory_pool = nullptr;
|
|
|
|
memdelete(pool);
|
|
|
|
}
|
|
|
|
|
|
|
|
VoxelMemoryPool *VoxelMemoryPool::get_singleton() {
|
|
|
|
CRASH_COND(g_memory_pool == nullptr);
|
|
|
|
return g_memory_pool;
|
|
|
|
}
|
|
|
|
|
|
|
|
VoxelMemoryPool::VoxelMemoryPool() {
|
|
|
|
_mutex = Mutex::create();
|
|
|
|
}
|
|
|
|
|
|
|
|
VoxelMemoryPool::~VoxelMemoryPool() {
|
2020-01-26 01:59:53 +00:00
|
|
|
#ifdef TOOLS_ENABLED
|
2020-01-02 20:31:05 +00:00
|
|
|
debug_print();
|
2020-01-26 01:59:53 +00:00
|
|
|
#endif
|
2020-01-02 20:31:05 +00:00
|
|
|
clear();
|
|
|
|
memdelete(_mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t *VoxelMemoryPool::allocate(uint32_t size) {
|
|
|
|
MutexLock lock(_mutex);
|
|
|
|
Pool *pool = get_or_create_pool(size);
|
|
|
|
uint8_t *block;
|
|
|
|
if (pool->blocks.size() > 0) {
|
|
|
|
block = pool->blocks.back();
|
|
|
|
pool->blocks.pop_back();
|
|
|
|
} else {
|
|
|
|
block = (uint8_t *)memalloc(size * sizeof(uint8_t));
|
|
|
|
}
|
2020-01-26 01:59:53 +00:00
|
|
|
++_used_blocks;
|
2020-01-02 20:31:05 +00:00
|
|
|
return block;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelMemoryPool::recycle(uint8_t *block, uint32_t size) {
|
|
|
|
MutexLock lock(_mutex);
|
2020-01-07 21:24:14 +00:00
|
|
|
Pool *pool = _pools[size]; // If not found, entry will be created! It would be an error
|
2020-01-02 20:31:05 +00:00
|
|
|
// Check recycling before having allocated
|
|
|
|
CRASH_COND(pool == nullptr);
|
|
|
|
pool->blocks.push_back(block);
|
2020-01-26 01:59:53 +00:00
|
|
|
--_used_blocks;
|
2020-01-02 20:31:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelMemoryPool::clear() {
|
|
|
|
MutexLock lock(_mutex);
|
2020-07-25 23:19:08 +01:00
|
|
|
const uint32_t *key = nullptr;
|
2020-01-02 20:31:05 +00:00
|
|
|
while ((key = _pools.next(key))) {
|
|
|
|
Pool *pool = _pools.get(*key);
|
|
|
|
CRASH_COND(pool == nullptr);
|
|
|
|
for (auto it = pool->blocks.begin(); it != pool->blocks.end(); ++it) {
|
|
|
|
uint8_t *ptr = *it;
|
|
|
|
CRASH_COND(ptr == nullptr);
|
|
|
|
memfree(ptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_pools.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelMemoryPool::debug_print() {
|
|
|
|
MutexLock lock(_mutex);
|
|
|
|
print_line("-------- VoxelMemoryPool ----------");
|
2020-07-25 23:19:08 +01:00
|
|
|
const uint32_t *key = nullptr;
|
2020-01-02 20:31:05 +00:00
|
|
|
int i = 0;
|
|
|
|
while ((key = _pools.next(key))) {
|
|
|
|
Pool *pool = _pools.get(*key);
|
2020-03-28 23:20:49 +08:00
|
|
|
print_line(String("Pool {0} for size {1}: {2} blocks").format(varray(i, *key, static_cast<int>(pool->blocks.size()))));
|
2020-01-02 20:31:05 +00:00
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-26 01:59:53 +00:00
|
|
|
unsigned int VoxelMemoryPool::debug_get_used_blocks() const {
|
|
|
|
MutexLock lock(_mutex);
|
|
|
|
return _used_blocks;
|
|
|
|
}
|
|
|
|
|
2020-01-02 20:31:05 +00:00
|
|
|
VoxelMemoryPool::Pool *VoxelMemoryPool::get_or_create_pool(uint32_t size) {
|
|
|
|
Pool *pool;
|
|
|
|
Pool **ppool = _pools.getptr(size);
|
|
|
|
if (ppool == nullptr) {
|
|
|
|
pool = memnew(Pool);
|
2020-01-07 21:24:14 +00:00
|
|
|
CRASH_COND(pool == nullptr);
|
2020-01-02 20:31:05 +00:00
|
|
|
_pools.set(size, pool);
|
|
|
|
} else {
|
|
|
|
pool = *ppool;
|
2020-01-07 21:24:14 +00:00
|
|
|
CRASH_COND(pool == nullptr);
|
2020-01-02 20:31:05 +00:00
|
|
|
}
|
|
|
|
return pool;
|
|
|
|
}
|