From eb38ff43ab8716cf1919717e75f2e5352e95e55c Mon Sep 17 00:00:00 2001 From: Francisco Date: Thu, 25 Mar 2021 18:03:52 +0000 Subject: [PATCH] fixed size buffer pool bugfix - deleted threads now return the cached memory blocks to the pool shared memory block container --- lib/include/srsran/adt/pool/fixed_size_pool.h | 38 +++++++++++++------ lib/test/adt/mem_pool_test.cc | 1 + 2 files changed, 27 insertions(+), 12 deletions(-) diff --git a/lib/include/srsran/adt/pool/fixed_size_pool.h b/lib/include/srsran/adt/pool/fixed_size_pool.h index e32de3ee4..9bfee3b9d 100644 --- a/lib/include/srsran/adt/pool/fixed_size_pool.h +++ b/lib/include/srsran/adt/pool/fixed_size_pool.h @@ -34,6 +34,7 @@ template class concurrent_fixed_memory_pool { static_assert(ObjSize > 256, "This pool is particularly designed for large objects"); + using pool_type = concurrent_fixed_memory_pool; struct obj_storage_t { typename std::aligned_storage::type buffer; @@ -41,7 +42,7 @@ class concurrent_fixed_memory_pool explicit obj_storage_t(std::thread::id id_) : worker_id(id_) {} }; - const static size_t batch_steal_size = 10; + const static size_t batch_steal_size = 16; // ctor only accessible from singleton get_instance() explicit concurrent_fixed_memory_pool(size_t nof_objects_) @@ -82,18 +83,18 @@ public: void* allocate_node(size_t sz) { srsran_assert(sz <= ObjSize, "Allocated node size=%zd exceeds max object size=%zd", sz, ObjSize); - memblock_cache* worker_cache = get_worker_cache(); + worker_ctxt* worker_ctxt = get_worker_cache(); - void* node = worker_cache->try_pop(); + void* node = worker_ctxt->cache.try_pop(); if (node == nullptr) { // fill the thread local cache enough for this and next allocations std::array popped_blocks; size_t n = shared_mem_cache.try_pop(popped_blocks); for (size_t i = 0; i < n; ++i) { - new (popped_blocks[i]) obj_storage_t(std::this_thread::get_id()); - worker_cache->push(static_cast(popped_blocks[i])); + new (popped_blocks[i]) obj_storage_t(worker_ctxt->id); + worker_ctxt->cache.push(static_cast(popped_blocks[i])); } - node = worker_cache->try_pop(); + node = worker_ctxt->cache.try_pop(); } #ifdef SRSRAN_BUFFER_POOL_LOG_ENABLED @@ -107,8 +108,8 @@ public: void deallocate_node(void* p) { srsran_assert(p != nullptr, "Deallocated nodes must have valid address"); - memblock_cache* worker_cache = get_worker_cache(); - obj_storage_t* block_ptr = static_cast(p); + worker_ctxt* worker_ctxt = get_worker_cache(); + obj_storage_t* block_ptr = static_cast(p); if (DebugSanitizeAddress) { std::lock_guard lock(mutex); @@ -119,7 +120,7 @@ public: (long unsigned)block_ptr); } - if (block_ptr->worker_id != std::this_thread::get_id() or worker_cache->size() >= max_objs_per_cache) { + if (block_ptr->worker_id != worker_ctxt->id or worker_ctxt->cache.size() >= max_objs_per_cache) { // if block was allocated in a different thread or local cache reached max capacity, send block to shared // container shared_mem_cache.push(static_cast(block_ptr)); @@ -127,7 +128,7 @@ public: } // push to local memory block cache - worker_cache->push(static_cast(p)); + worker_ctxt->cache.push(static_cast(p)); } void enable_logger(bool enabled) @@ -147,9 +148,22 @@ public: } private: - memblock_cache* get_worker_cache() + struct worker_ctxt { + std::thread::id id; + memblock_cache cache; + + worker_ctxt() : id(std::this_thread::get_id()) {} + ~worker_ctxt() + { + while (not cache.is_empty()) { + pool_type::get_instance()->shared_mem_cache.push(cache.try_pop()); + } + } + }; + + worker_ctxt* get_worker_cache() { - thread_local memblock_cache worker_cache; + thread_local worker_ctxt worker_cache; return &worker_cache; } diff --git a/lib/test/adt/mem_pool_test.cc b/lib/test/adt/mem_pool_test.cc index cedb2cc6f..a98a5c59e 100644 --- a/lib/test/adt/mem_pool_test.cc +++ b/lib/test/adt/mem_pool_test.cc @@ -134,6 +134,7 @@ void test_fixedsize_pool() TESTASSERT(obj != nullptr); } stop.store(true); + fixed_pool->print_all_buffers(); t.join(); } fixed_pool->print_all_buffers();