54 template <
typename TYPE>
63 template <
typename TYPE>
64 static void free(TYPE* ptr);
84 template<
typename TYPE,
template <
typename>
typename DERIVED>
96 template <
typename O_TYPE>
128 template<
typename O_TYPE,
typename... Args>
136 template<
typename O_TYPE>
153 template<
typename TYPE>
202 template<
typename TYPE>
213 freeChunk* next =
nullptr;
220 class allocatedChunks {
222 void* chunks =
nullptr;
223 allocatedChunks* next =
nullptr;
233 static constexpr u_integer CHUNK_SIZE =
sizeof(TYPE) >
sizeof(freeChunk*) ?
sizeof(TYPE) :
sizeof(freeChunk*);
238 freeChunk** free_list_head;
239 allocatedChunks* allocated_list_head;
277 void release()
noexcept;
347template<
typename TYPE>
354 return static_cast<TYPE*
>(
operator new(size *
sizeof(TYPE)));
355 }
catch (
const std::bad_alloc&) {
360template<
typename TYPE>
362 ::operator
delete(ptr);
365template <
typename TYPE,
template <
typename>
class DERIVED>
371template<
typename TYPE,
template <
typename>
typename DERIVED>
374template<
typename TYPE,
template <
typename>
typename DERIVED>
375template<
typename O_TYPE,
typename... Args>
377 new (o_ptr) O_TYPE{std::forward<Args>(args)...};
380template<
typename TYPE,
template <
typename>
typename DERIVED>
381template<
typename O_TYPE>
386template<
typename TYPE>
391template<
typename TYPE>
396template<
typename TYPE>
399 while (
static_cast<u_integer>(1) << index < size) {
405template<
typename TYPE>
406void original::objPoolAllocator<TYPE>::poolInit() {
410 this->allocated_list_head =
nullptr;
412 for (
u_integer i = 0; i < this->size_class_count; i++) {
413 this->chunk_count[i] = this->chunk_count_init;
414 this->free_list_head[i] =
nullptr;
415 this->chunks_available[i] = 0;
419template <
typename TYPE>
420void original::objPoolAllocator<TYPE>::chunkAllocate(
const u_integer num_element,
u_integer index)
422 const u_integer block_size = (1 << index) * CHUNK_SIZE;
426 new_allocated_chunk->chunks = new_free_chunk;
427 new_allocated_chunk->next = this->allocated_list_head;
428 this->allocated_list_head = new_allocated_chunk;
430 for (
u_integer i = 0; i < num_element; i++)
432 auto cur_ptr =
reinterpret_cast<freeChunk*
>(new_free_chunk + i * block_size);
433 cur_ptr->next = this->free_list_head[index];
434 this->free_list_head[index] = cur_ptr;
436 this->chunks_available[index] += num_element;
437 this->chunk_count[index] = this->chunk_count[index] + (this->chunk_count[index] >> 1);
440template<
typename TYPE>
441void original::objPoolAllocator<TYPE>::release() noexcept {
442 while (this->allocated_list_head) {
443 auto next_chunk = this->allocated_list_head->next;
446 this->allocated_list_head = next_chunk;
449 if (this->chunk_count) {
451 this->chunk_count =
nullptr;
453 if (this->free_list_head) {
455 this->free_list_head =
nullptr;
457 if (this->chunks_available) {
459 this->chunks_available =
nullptr;
463template <
typename TYPE>
465 : size_class_count(size_class_count), chunk_count_init(count), chunk_count(nullptr),
466 free_list_head(nullptr), allocated_list_head(nullptr), chunks_available(nullptr) {
470template<
typename TYPE>
475 auto size_class_count_max =
max(this->size_class_count, other.size_class_count);
476 auto chunk_count_init_max =
max(this->chunk_count_init, other.chunk_count_init);
481 for (
u_integer i = 0; i < size_class_count_max; i++) {
482 if (i < this->size_class_count && i < other.size_class_count) {
483 chunk_count_max[i] =
max(this->chunk_count[i], other.chunk_count[i]);
484 chunks_available_max[i] = this->chunks_available[i] + other.chunks_available[i];
485 auto cur_list_head_i = other.free_list_head[i];
486 if (cur_list_head_i) {
487 while (cur_list_head_i->next) {
488 cur_list_head_i = cur_list_head_i->next;
490 cur_list_head_i->next = this->free_list_head[i];
491 free_list_head_max[i] = other.free_list_head[i];
493 free_list_head_max[i] = this->free_list_head[i];
495 }
else if (i < this->size_class_count) {
496 chunk_count_max[i] = this->chunk_count[i];
497 chunks_available_max[i] = this->chunks_available[i];
498 free_list_head_max[i] = this->free_list_head[i];
500 chunk_count_max[i] = other.chunk_count[i];
501 chunks_available_max[i] = other.chunks_available[i];
502 free_list_head_max[i] = other.free_list_head[i];
506 auto allocated_list_head_max = other.allocated_list_head;
507 auto cur_allocated_list_head = allocated_list_head_max;
508 if (cur_allocated_list_head) {
509 while (cur_allocated_list_head->next) {
510 cur_allocated_list_head = cur_allocated_list_head->next;
512 cur_allocated_list_head->next = this->allocated_list_head;
514 allocated_list_head_max = this->allocated_list_head;
516 this->allocated_list_head =
nullptr;
517 other.allocated_list_head =
nullptr;
523 this->size_class_count = size_class_count_max;
524 this->chunk_count_init = chunk_count_init_max;
525 this->chunk_count = chunk_count_max;
526 this->free_list_head = free_list_head_max;
527 this->chunks_available = chunks_available_max;
528 this->allocated_list_head = allocated_list_head_max;
533template<
typename TYPE>
538template<
typename TYPE>
545 this->size_class_count = other.size_class_count;
546 this->chunk_count_init = other.chunk_count_init;
547 this->chunk_count = other.chunk_count;
548 this->free_list_head = other.free_list_head;
549 this->allocated_list_head = other.allocated_list_head;
550 this->chunks_available = other.chunks_available;
556template <
typename TYPE>
565 if (index >= this->size_class_count) {
569 if (!this->free_list_head[index] || this->chunks_available[index] * (
static_cast<u_integer>(1) << index) < size) {
570 this->chunkAllocate(
max<u_integer>(1, this->chunk_count[index]), index);
573 auto cur_ptr = this->free_list_head[index];
574 this->free_list_head[index] = this->free_list_head[index]->next;
575 this->chunks_available[index] -= 1;
576 return reinterpret_cast<TYPE*
>(cur_ptr);
579template <
typename TYPE>
588 if (index >= this->size_class_count) {
593 auto p =
reinterpret_cast<freeChunk*
>(ptr);
594 p->next = this->free_list_head[index];
595 this->free_list_head[index] = p;
596 this->chunks_available[index] += 1;
599template<
typename TYPE>
Exception for memory allocation failures.
Definition error.h:184
std::false_type propagate_on_container_swap
No propagation on swap.
Definition allocator.h:89
std::false_type propagate_on_container_copy_assignment
No propagation on copy.
Definition allocator.h:87
DERIVED< O_TYPE > rebind_alloc
Rebinds allocator to different type.
Definition allocator.h:97
virtual ~allocatorBase()=0
Virtual destructor.
std::false_type propagate_on_container_move_assignment
No propagation on move.
Definition allocator.h:88
void construct(O_TYPE *o_ptr, Args &&... args)
Constructs an object in allocated memory.
Definition allocator.h:376
constexpr allocatorBase()
Constructs a new allocatorBase instance.
Definition allocator.h:366
static void destroy(O_TYPE *o_ptr)
Destroys an object without deallocating.
Definition allocator.h:382
virtual void deallocate(TYPE *ptr, u_integer size)=0
Deallocates memory.
virtual TYPE * allocate(u_integer size)=0
Allocates raw memory.
std::false_type propagate_on_container_merge
No propagation on merge.
Definition allocator.h:90
Default memory allocator using allocators utilities.
Definition allocator.h:154
void deallocate(TYPE *ptr, u_integer size) override
Deallocates memory using global operator delete.
Definition allocator.h:392
TYPE * allocate(u_integer size) override
Allocates memory using global operator new.
Definition allocator.h:387
Utility class providing static memory allocation/de-allocation functions.
Definition allocator.h:43
static TYPE * malloc(u_integer size)
Allocates raw memory using global operator new.
Definition allocator.h:348
static void free(TYPE *ptr)
Deallocates memory using global operator delete.
Definition allocator.h:361
Object pool allocator for efficient fixed-size memory management.
Definition allocator.h:204
objPoolAllocator & operator+=(objPoolAllocator &other)
Merges another pool allocator into this one.
Definition allocator.h:471
objPoolAllocator(const objPoolAllocator &)=delete
Copy construction disabled.
TYPE * allocate(u_integer size) override
Allocates memory from the pool.
Definition allocator.h:557
std::true_type propagate_on_container_swap
Allows propagation on swap.
Definition allocator.h:282
void deallocate(TYPE *ptr, u_integer size) override
Returns memory to the pool.
Definition allocator.h:580
std::true_type propagate_on_container_move_assignment
Allows propagation on move.
Definition allocator.h:281
objPoolAllocator(u_integer size_class_count=8, u_integer count=4)
Constructs a new object pool allocator.
Definition allocator.h:464
std::true_type propagate_on_container_merge
Allows propagation on merge.
Definition allocator.h:283
~objPoolAllocator() override
Destructor - releases all allocated memory.
Definition allocator.h:600
objPoolAllocator & operator=(const objPoolAllocator &)=delete
Copy assignment disabled.
Compile-time error assertion utility.
Definition error.h:73
Platform-independent integer type definitions.
Custom exception classes and callback validation utilities.
Mathematical utilities and constants.
Main namespace for the project Original.
Definition algorithms.h:21
TYPE max(TYPE a, TYPE b)
Returns the larger of two given values.
std::uint32_t u_integer
32-bit unsigned integer type for sizes/indexes
Definition config.h:17