#include "stdafx.h" #include "IdManager.h" #include "Utilities/Thread.h" shared_mutex id_manager::g_mutex; thread_local DECLARE(idm::g_id); DECLARE(idm::g_map); id_manager::id_map::pointer idm::allocate_id(const id_manager::id_key& info, u32 base, u32 step, u32 count, std::pair invl_range) { // Base type id is stored in value auto& vec = g_map[info.value()]; // Preallocate memory vec.reserve(count); if (vec.size() < count) { // Try to emplace back const u32 _next = base + step * ::size32(vec); g_id = _next; vec.emplace_back(id_manager::id_key(_next, info.type()), nullptr); return &vec.back(); } // Check all IDs starting from "next id" (TODO) for (u32 i = 0, next = base; i < count; i++, next += step) { const auto ptr = &vec[i]; // Look for free ID if (!ptr->second) { // Incremenet ID invalidation counter const u32 id = next | ((ptr->first + (1u << invl_range.first)) & (invl_range.second ? (((1u << invl_range.second) - 1) << invl_range.first) : 0)); g_id = id; ptr->first = id_manager::id_key(id, info.type()); return ptr; } } // Out of IDs return nullptr; } void idm::init() { // Allocate g_map.resize(id_manager::typeinfo::get_count()); idm::clear(); } void idm::clear() { // Call recorded finalization functions for all IDs for (auto& map : g_map) { for (auto& pair : map) { pair.second.reset(); pair.first = {}; } map.clear(); } }