1#ifndef LIBTEDDY_DETAILS_NODE_POOL_HPP
2#define LIBTEDDY_DETAILS_NODE_POOL_HPP
4#include <libteddy/details/config.hpp>
5#include <libteddy/details/debug.hpp>
6#include <libteddy/details/node.hpp>
7#include <libteddy/details/tools.hpp>
15template<
class Data,
class Degree>
22 node_pool(int64 mainPoolSize, int64 extraPoolSize);
27 auto operator= (
node_pool const&) =
delete;
30 [[nodiscard]]
auto get_available_node_count () const -> int64;
32 [[nodiscard]] auto get_main_pool_size () const -> int64;
34 template<class... Args>
35 [[nodiscard]] auto create (Args&&... args) ->
node_t*;
55 [[nodiscard]]
static auto allocate_pool (int64 size, pool_item* next)
62 static auto deallocate_pool (pool_item* poolPtr, node_t* lastNode)
67 node_t* nextPoolNode_;
71 int64 availableNodeCount_;
74template<
class Data,
class Degree>
75node_pool<Data, Degree>::node_pool(
76 int64
const mainPoolSize,
77 int64
const overflowPoolSize
79 pools_(allocate_pool(mainPoolSize, nullptr)),
80 nextPoolNode_(pools_->pool_),
82 mainPoolSize_(mainPoolSize),
83 extraPoolSize_(overflowPoolSize),
84 availableNodeCount_(mainPoolSize)
86#ifdef LIBTEDDY_VERBOSE
88 "node_pool::node_pool\tAllocating initial pool with size ",
95template<
class Data,
class Degree>
96node_pool<Data, Degree>::node_pool(node_pool&& other) noexcept :
97 pools_(utils::exchange(other.mainPool_,
nullptr)),
98 nextPoolNode_(utils::exchange(other.nextPoolNode_,
nullptr)),
99 freeNodes_(utils::exchange(other.freeNodes_,
nullptr)),
100 mainPoolSize_(utils::exchange(other.mainPoolSize_, -1)),
101 extraPoolSize_(utils::exchange(other.extraPoolSize_, -1)),
102 availableNodeCount_(utils::exchange(other.availableNodeCount_, -1))
106template<
class Data,
class Degree>
112 pools_ = deallocate_pool(pools_, nextPoolNode_);
117 while (pools_ && pools_->next_)
119 node_t*
const lastNode = pools_->pool_ + extraPoolSize_;
120 pools_ = deallocate_pool(pools_, lastNode);
128 node_t*
const lastNode = pools_->pool_ + mainPoolSize_;
129 pools_ = deallocate_pool(pools_, lastNode);
133template<
class Data,
class Degree>
136 return availableNodeCount_;
139template<
class Data,
class Degree>
140auto node_pool<Data, Degree>::get_main_pool_size() const -> int64
142 return mainPoolSize_;
145template<
class Data,
class Degree>
146template<
class... Args>
147auto node_pool<Data, Degree>::create(Args&&... args) -> node_t*
149 assert(availableNodeCount_ > 0);
150 --availableNodeCount_;
152 node_t* node =
nullptr;
156 freeNodes_ = freeNodes_->get_next();
161 node = nextPoolNode_;
165 return static_cast<node_t*
>(::new (node) node_t(args...));
168template<
class Data,
class Degree>
169auto node_pool<Data, Degree>::destroy(node_t*
const node) ->
void
171 ++availableNodeCount_;
172 node->set_next(freeNodes_);
176template<
class Data,
class Degree>
177auto node_pool<Data, Degree>::grow() ->
void
179#ifdef LIBTEDDY_VERBOSE
181 "node_pool::grow\tallocating overflow pool with size ",
187 pools_ = allocate_pool(extraPoolSize_, pools_);
188 nextPoolNode_ = pools_->pool_;
189 availableNodeCount_ += extraPoolSize_;
192template<
class Data,
class Degree>
193auto node_pool<Data, Degree>::allocate_pool(
195 pool_item*
const next
198 return new pool_item {
199 static_cast<node_t*
>(std::malloc(as_usize(size) *
sizeof(node_t))),
203template<
class Data,
class Degree>
204auto node_pool<Data, Degree>::deallocate_pool(
205 pool_item*
const pool,
206 node_t*
const lastNode
209 node_t* node = pool->pool_;
210 while (node < lastNode)
215 pool_item* next = pool->next_;
216 std::free(pool->pool_);
Definition node_pool.hpp:17
~node_pool()
Definition node_pool.hpp:107