PcapPlusPlus  Next
ObjectPool.h
1 #pragma once
2 
3 #include <stack>
4 #include <mutex>
5 #include <memory>
6 #include <limits>
7 #include <type_traits>
8 
9 namespace pcpp
10 {
11  namespace internal
12  {
21  template <class T, typename std::enable_if<std::is_default_constructible<T>::value, bool>::type = true>
23  {
24  public:
25  constexpr static std::size_t DEFAULT_POOL_SIZE = 100;
26 #pragma push_macro("max") // Undefine max to avoid conflict with std::numeric_limits<std::size_t>::max()
27 #undef max
28  constexpr static std::size_t INFINITE_POOL_SIZE = std::numeric_limits<std::size_t>::max();
29 #pragma pop_macro("max")
30 
34  explicit DynamicObjectPool(std::size_t maxPoolSize = DEFAULT_POOL_SIZE, std::size_t initialSize = 0)
35  : m_MaxPoolSize(maxPoolSize)
36  {
37  if (initialSize > maxPoolSize)
38  throw std::invalid_argument("Preallocated objects cannot exceed the maximum pool size");
39 
40  if (initialSize > 0)
41  this->preallocate(initialSize);
42  }
43 
44  // These don't strictly need to be deleted, but don't need to be implemented for now either.
45  DynamicObjectPool(const DynamicObjectPool&) = delete;
47  DynamicObjectPool& operator=(const DynamicObjectPool&) = delete;
48  DynamicObjectPool& operator=(DynamicObjectPool&&) = delete;
49 
52  {
53  clear();
54  }
55 
62  std::unique_ptr<T> acquireObject()
63  {
64  return std::unique_ptr<T>(acquireObjectRaw());
65  }
66 
74  {
75  std::unique_lock<std::mutex> lock(m_Mutex);
76 
77  if (m_Pool.empty())
78  {
79  // We don't need the lock anymore, so release it.
80  lock.unlock();
81  return new T();
82  }
83 
84  T* obj = m_Pool.top();
85  m_Pool.pop();
86  return obj;
87  }
88 
95  void releaseObject(std::unique_ptr<T> obj)
96  {
97  releaseObjectRaw(obj.release());
98  }
99 
106  void releaseObjectRaw(T* obj)
107  {
108  std::unique_lock<std::mutex> lock(m_Mutex);
109 
110  if (m_MaxPoolSize == INFINITE_POOL_SIZE || m_Pool.size() < m_MaxPoolSize)
111  {
112  m_Pool.push(obj);
113  }
114  else
115  {
116  // We don't need the lock anymore, so release it.
117  lock.unlock();
118  delete obj;
119  }
120  }
121 
123  std::size_t size() const
124  {
125  std::lock_guard<std::mutex> lock(m_Mutex);
126  return m_Pool.size();
127  }
128 
130  std::size_t maxSize() const
131  {
132  std::lock_guard<std::mutex> lock(m_Mutex);
133  return m_MaxPoolSize;
134  }
135 
137  void setMaxSize(std::size_t maxSize)
138  {
139  std::lock_guard<std::mutex> lock(m_Mutex);
140  m_MaxPoolSize = maxSize;
141 
142  // If the new max size is less than the current size, we need to remove some objects from the pool.
143  while (m_Pool.size() > m_MaxPoolSize)
144  {
145  delete m_Pool.top();
146  m_Pool.pop();
147  }
148  }
149 
152  void preallocate(std::size_t count)
153  {
154  std::unique_lock<std::mutex> lock(m_Mutex);
155 
156  if (m_MaxPoolSize < count)
157  {
158  throw std::invalid_argument("Preallocated objects cannot exceed the maximum pool size");
159  }
160 
161  // If the pool is already larger than the requested count, we don't need to do anything.
162  for (std::size_t i = m_Pool.size(); i < count; i++)
163  {
164  m_Pool.push(new T());
165  }
166  }
167 
169  void clear()
170  {
171  std::unique_lock<std::mutex> lock(m_Mutex);
172  while (!m_Pool.empty())
173  {
174  delete m_Pool.top();
175  m_Pool.pop();
176  }
177  }
178 
179  private:
180  std::size_t m_MaxPoolSize;
181  mutable std::mutex m_Mutex;
182  std::stack<T*> m_Pool;
183  };
184  } // namespace internal
185 } // namespace pcpp
A generic object pool implementation.
Definition: ObjectPool.h:23
std::size_t maxSize() const
Gets the maximum number of objects in the pool.
Definition: ObjectPool.h:130
void preallocate(std::size_t count)
Pre-allocates up to a minimum number of objects in the pool.
Definition: ObjectPool.h:152
std::size_t size() const
Gets the current number of objects in the pool.
Definition: ObjectPool.h:123
void releaseObjectRaw(T *obj)
Releases a raw pointer to an object back to the pool.
Definition: ObjectPool.h:106
std::unique_ptr< T > acquireObject()
Acquires a unique pointer to an object from the pool.
Definition: ObjectPool.h:62
void clear()
Deallocates and releases all objects currently held by the pool.
Definition: ObjectPool.h:169
DynamicObjectPool(std::size_t maxPoolSize=DEFAULT_POOL_SIZE, std::size_t initialSize=0)
Definition: ObjectPool.h:34
T * acquireObjectRaw()
Acquires a raw pointer to an object from the pool.
Definition: ObjectPool.h:73
void setMaxSize(std::size_t maxSize)
Sets the maximum number of objects in the pool.
Definition: ObjectPool.h:137
~DynamicObjectPool()
A destructor for this class that deletes all objects in the pool.
Definition: ObjectPool.h:51
void releaseObject(std::unique_ptr< T > obj)
Releases a unique pointer to an object back to the pool.
Definition: ObjectPool.h:95
The main namespace for the PcapPlusPlus lib.