qxLib
unique_objects_pool.inl
Go to the documentation of this file.
1 /**
2 
3  @file unique_objects_pool.inl
4  @author Khrapov
5  @date 11.02.2023
6  @copyright © Nick Khrapov, 2023. All right reserved.
7 
8 **/
9 
10 namespace qx
11 {
12 
13 template<class T>
14 inline unique_objects_pool<T>::token::token(const token& otherToken) noexcept
15 {
16  *this = otherToken;
17 }
18 
19 template<class T>
20 unique_objects_pool<T>::token::token(token&& otherToken) noexcept
21 {
22  *this = std::move(otherToken);
23 }
24 
25 template<class T>
26 inline typename unique_objects_pool<T>::token& unique_objects_pool<T>::token::operator=(
27  const token& otherToken) noexcept
28 {
29  m_nId = otherToken.m_nId;
30  m_pPool = otherToken.m_pPool;
31 
32  if (is_valid())
33  m_pPool->increase_counter(m_nId);
34 
35  return *this;
36 }
37 
38 template<class T>
39 typename unique_objects_pool<T>::token& unique_objects_pool<T>::token::operator=(token&& otherToken) noexcept
40 {
41  std::swap(m_nId, otherToken.m_nId);
42  std::swap(m_pPool, otherToken.m_pPool);
43 
44  return *this;
45 }
46 
47 template<class T>
48 inline unique_objects_pool<T>::token::~token() noexcept
49 {
50  if (is_valid())
51  m_pPool->decrease_counter(m_nId);
52 }
53 
54 template<class T>
55 inline bool unique_objects_pool<T>::token::is_valid() const noexcept
56 {
57  return m_pPool && m_nId != kInvalidId;
58 }
59 
60 template<class T>
61 inline unique_objects_pool<T>::token::operator bool() const noexcept
62 {
63  return is_valid();
64 }
65 
66 template<class T>
67 inline const T& unique_objects_pool<T>::token::operator*() const noexcept
68 {
69  return m_pPool->get_value(m_nId);
70 }
71 
72 template<class T>
73 inline const T* unique_objects_pool<T>::token::operator->() const noexcept
74 {
75  return &m_pPool->get_value(m_nId);
76 }
77 
78 template<class T>
79 inline bool unique_objects_pool<T>::token::operator==(const token& other) const noexcept
80 {
81  return m_pPool == other.m_pPool && m_nId == other.m_nId;
82 }
83 
84 template<class T>
85 inline unique_objects_pool<T>::token::token(u64 nId, unique_objects_pool* pPool) noexcept : m_nId(nId)
86  , m_pPool(pPool)
87 {
88 }
89 
90 template<class T>
91 inline unique_objects_pool<T>::unique_objects_pool(bool bAutoCleanup) noexcept : m_bAutoShrink(bAutoCleanup)
92 {
93 }
94 
95 template<class T>
96 template<class U>
98 {
99  QX_PERF_SCOPE();
100 
101  token result;
102 
103  std::unique_lock lock(m_UniqueObjectsPoolMutex);
104  data_by_value& set = m_Pool.template get<value_tag>();
105  if (auto it = set.find(std::as_const(value)); it != set.end())
106  {
107  result = token(it->nId, this);
108  set.modify(
109  it,
110  [](data& _data)
111  {
112  ++_data.nCounter;
113  });
114  }
115  else
116  {
117  data _data { T(std::forward<U>(value)), m_nCurrentId++, 1 };
118  result = token(_data.nId, this);
119  set.insert(_data);
120  }
121 
122  return result;
123 }
124 
125 template<class T>
127 {
128  QX_PERF_SCOPE();
129 
130  if (m_bAutoShrink)
131  return;
132 
133  std::unique_lock lock(m_UniqueObjectsPoolMutex);
134  data_by_id& set = m_Pool.template get<id_tag>();
135  do
136  {
137  auto it = std::find_if(
138  set.begin(),
139  set.end(),
140  [&](const data& _data)
141  {
142  return _data.nCounter == 0;
143  });
144 
145  if (it != set.end())
146  set.erase(it);
147  else
148  break;
149  } while (true);
150 }
151 
152 template<class T>
153 inline size_t unique_objects_pool<T>::size() const
154 {
155  QX_PERF_SCOPE();
156 
157  std::shared_lock lock(m_UniqueObjectsPoolMutex);
158  return m_Pool.size();
159 }
160 
161 template<class T>
162 inline bool unique_objects_pool<T>::empty() const
163 {
164  QX_PERF_SCOPE();
165 
166  return size() == 0;
167 }
168 
169 template<class T>
170 inline void unique_objects_pool<T>::increase_counter(u64 nId) noexcept
171 {
172  QX_PERF_SCOPE();
173 
174  std::unique_lock lock(m_UniqueObjectsPoolMutex);
175  data_by_id& set = m_Pool.template get<id_tag>();
176  if (auto it = set.find(nId); it != set.end())
177  {
178  set.modify(
179  it,
180  [](data& _data)
181  {
182  ++_data.nCounter;
183  });
184  }
185 }
186 
187 template<class T>
188 inline void unique_objects_pool<T>::decrease_counter(u64 nId) noexcept
189 {
190  QX_PERF_SCOPE();
191 
192  std::unique_lock lock(m_UniqueObjectsPoolMutex);
193  data_by_id& set = m_Pool.template get<id_tag>();
194  if (auto it = set.find(nId); it != set.end())
195  {
196  if (m_bAutoShrink && it->nCounter == 1)
197  {
198  set.erase(it);
199  }
200  else
201  {
202  set.modify(
203  it,
204  [](data& _data)
205  {
206  --_data.nCounter;
207  });
208  }
209  }
210 }
211 
212 template<class T>
213 inline const T& unique_objects_pool<T>::get_value(u64 nId) noexcept
214 {
215  QX_PERF_SCOPE();
216 
217  std::shared_lock lock(m_UniqueObjectsPoolMutex);
218  data_by_id& set = m_Pool.template get<id_tag>();
219 
220  // only tokens can access this method -> object always exists
221  return set.find(nId)->value;
222 }
223 
224 } // namespace qx
The token is used for distributed access to a unique object in the pool.
bool is_valid() const noexcept
Is this token have been set (not default)
Class stores unique objects and allows access to them through tokens.
unique_objects_pool(bool bAutoCleanup=true) noexcept
unique_objects_pool object constructor
token get_or_create(U &&value) noexcept
Get or create object.
bool empty() const
Is pool empty.
void shrink()
Remove unused (with counter = 0) objects.
size_t size() const
Get number of objects stored.
uint64_t u64
Definition: typedefs.h:27