Imt.Base C++ API V4.1.1.0
Loading...
Searching...
No Matches
PoolAllocator.h
Go to the documentation of this file.
1// (c) IMT - Information Management Technology AG, CH-9470 Buchs, www.imt.ch.
2//
3// ActiveParts (AP) and the corresponding Data Flow Framework (DFF) is invented and designed by Jakob Daescher.
4// ANY USE OF THIS CODE CONSTITUTES ACCEPTANCE OF THE TERMS OF THE COPYRIGHT NOTICE.
5// ===================================================================================================
6// COPYRIGHT NOTICE
7// ===================================================================================================
8// Copyright (C) 2005-2075, IMT Information Management Technology AG, 9470 Buchs, Switzerland
9// All rights reserved.
10// This code is proprietary software of IMT Information Management Technology AG (hereinafter: "IMT").
11// Proprietary software is computer software licensed under exclusive legal right of IMT.
12//
13// The licensee is given the irrevocable, perpetual, worldwide, non-exclusive right and license to use,
14// execute and reproduce the software in binary form within the licensed products.
15//
16// Redistribution and use in source forms, without modification, are permitted provided that the following conditions are met:
17// (1) Copying of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
18// (2) Copying of source code is only allowed for regulatory documentation and archiving purposes
19// (3) Redistributions in binary form must reproduce the above copyright notice,
20// this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
21//
22// IMT provide no reassurances that the source code provided does not infringe
23// any patent, copyright, or any other intellectual property rights of third parties.
24// IMT disclaim any liability to any recipient for claims brought against
25// recipient by any third party for infringement of that parties intellectual property rights.
26//
27// THIS SOFTWARE IS PROVIDED BY IMT AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
28// INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
29// IN NO EVENT SHALL IMT OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY,
30// OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCURE-MENT OF SUBSTITUTE GOODS OR SERVICES;
31// LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
32// WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
33// IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
34// ===================================================================================================
35
36#ifndef IMT_BASE_CORE_UTIL_POOL_ALLOCATOR_H
37#define IMT_BASE_CORE_UTIL_POOL_ALLOCATOR_H
38
39#include <array>
42#include <new>
43#include <type_traits>
44
45namespace imt {
46namespace base {
47namespace core {
48namespace util {
49
66template<typename T>
68public:
69
76 // AXIVION Next Codeline AutosarC++19_03-A18.1.1: use C-style array for compatibility
77 PoolAllocator(T pool[], size_t poolSize) noexcept;
78
89 T* allocate() noexcept;
90
97 void deallocate(T* const obj) noexcept;
98
104 bool isFull() const noexcept {
105 return m_pFreeList == nullptr;
106 }
107
113 size_t remainingSize() const noexcept {
114 return capacity() - m_itemCount;
115 }
116
122 void const* endAddr() const noexcept {
123 return m_pEndAddr;
124 }
125
131 void const* startAddr() const noexcept {
132 return m_pStartAddr;
133 }
134
140 size_t capacity() const noexcept {
141 return m_capacity;
142 }
143
148 static size_t getValueSize() noexcept {
149 return sizeof(T);
150 }
151#ifndef _UNITTEST
152
153private:
154
155#endif
156 // AXIVION Next Construct AutosarC++19_03-A9.6.1: the static_assert checks belows are used to check the required conditions regarding pointer size
157 // AXIVION Next Construct AutosarC++19_03-A9.5.1: Allowed usage
158 // AXIVION Next Construct AutosarC++19_03-M3.2.2: how can the ODR (one definition rule) be violated with sizeof(T), there is no possibility that its different, else it is not the same PoolAllocator template instance
159 union Node {
160 // AXIVION Construct AutosarC++19_03-A2.7.3: Allowed missing comments, only local relevance
161 // AXIVION Construct AutosarC++19_03-M0.1.3: required construct, because Node must have the size of T
162 std::array<uint8_t, sizeof(T)> m_data;
163 Node* m_pNext;
164 };
165 // AXIVION Next Construct AutosarC++19_03-A2.7.3: Allowed missing comments, only private relevance
166 // AXIVION Next Construct AutosarC++19_03-M11.0.1: Allowed here, not private only for test reasons
167 Node* m_pFreeList;
168
172 // AXIVION Next Line AutosarC++19_03-M11.0.1: Allowed here, not private only for test reasons
173 void const* m_pEndAddr;
174 // AXIVION Next Construct AutosarC++19_03-A2.7.3: Allowed missing comments, only private relevance
175 // AXIVION Next Construct AutosarC++19_03-M11.0.1: Allowed here, not private only for test reasons
176 void const* m_pStartAddr;
177 // AXIVION Next Construct AutosarC++19_03-A2.7.3: Allowed missing comments, only private relevance
178 // AXIVION Next Construct AutosarC++19_03-M11.0.1: Allowed here, not private only for test reasons
179 size_t const m_capacity;
180 // AXIVION Next Construct AutosarC++19_03-A2.7.3: Allowed missing comments, only private relevance
181 // AXIVION Next Construct AutosarC++19_03-M11.0.1: Allowed here, not private only for test reasons
182 // AXIVION Next Construct AutosarC++19_03-A12.1.3: All data members initialized in custom constructor and would collide with AutosarC++19_03-A12.1.2
183 std::size_t m_itemCount;
184
185 // Assertion used to ensure that the size of T is equal to the size of Node. If this is not the case,
186 // please refer the alignment of T. The alignment of T must be at least equal or greater than the alignment
187 // of Node*. One exception is if the size is dividable by sizeof(Node*).
188 static_assert(sizeof(T) == sizeof(Node), "Node size missmatch");
189 static_assert(sizeof(T) >= sizeof(Node*), "Node size missmatch");
190};
191
192template<typename T>
193// AXIVION Next Routine AutosarC++19_03-M5.0.15: Allowed usage, because the m_pFreeList is used for an array only
194// AXIVION Next Codeline AutosarC++19_03-A18.1.1: use C-style array for compatibility
195PoolAllocator<T>::PoolAllocator(T pool[], size_t poolSize) noexcept :
196 // AXIVION Next Line AutosarC++19_03-A5.2.4: Allowed construct, static asserts ensures that a T and Node has the same size and that the size of T is greater or equivalent than the size of Node*.
197 m_pFreeList {reinterpret_cast<Node*>(pool)},
198 // AXIVION Next Line AutosarC++19_03-M5.0.16 / AutosarC++19_03-A5.2.5: required construct, the end address points to the first address after the pool section
199 m_pEndAddr {&pool[poolSize]}, // AXIVION Line AutosarC++19_03-A5.3.2 : NULL pointer check is made in the constructor instruction block at first line
200 m_pStartAddr {pool},
201 m_capacity {poolSize},
202 m_itemCount {0} {
203 memset(pool, 0, poolSize);
204 ASSERT_EX(nullptr != pool);
205 if (m_pFreeList != nullptr) {
206 // link array elements together
207 for (size_t i {0}; i < (poolSize - 1); ++i) {
208 // AXIVION Next Codeline AutosarC++19_03-M5.0.16 / AutosarC++19_03-A5.2.5: poolSize matches with length of the pool array
209 m_pFreeList[i].m_pNext = &m_pFreeList[i + 1]; // AXIVION Line AutosarC++19_03-M0.3.1: pointer is not NULL at dereference then checked and also in array bounds
210 }
211 // AXIVION Next Codeline AutosarC++19_03-M5.0.16 / AutosarC++19_03-A5.2.5 / AutosarC++19_03-M0.3.1: poolSize matches with length of the pool array
212 m_pFreeList[poolSize - 1].m_pNext = nullptr;
213 }
214}
215
216template<typename T>
217inline T* PoolAllocator<T>::allocate() noexcept {
218 if (m_pFreeList == nullptr) {
219 return nullptr;
220 }
221
222 Node* const pNext {m_pFreeList->m_pNext};
223
224 if (!std::is_trivially_copyable<T>::value) {
225 // AXIVION Next Codeline AutosarC++19_03-A18.5.8: Allowed usage, placement new operator doesn't use dynamic allocation, whilst still allowing proper initialization
226 // Axivion Next Codeline AutosarC++19_03-A18.5.2 : placement new operator is allowed within this rule.
227 // Axivion Next Codeline AutosarC++19_03-A18.5.8 : placement new operator is allowed within this rule.
228 // Axivion Next Codeline AutosarC++19_03-A18.5.10 : m_pFreeList is an array of Node type. With the static assert for Node == T, it is ensured that the alignment must be correct.
229 T* const pReturn {new (m_pFreeList) T};
230 m_pFreeList = pNext;
231 m_itemCount++;
232 return pReturn;
233 }
234 else {
235 // Axivion Next Codeline AutosarC++19_03-A18.5.10 : m_pFreeList is an array of Node type. With the static assert for Node == T, it is ensured that the alignment must be correct.
236 // Axivion Next Codeline AutosarC++19_03-A5.2.4: Allowed usage, because the T must be valid due to the static_assert
237 T* const pReturn {reinterpret_cast<T*>(&m_pFreeList->m_data)};
238 m_pFreeList = pNext;
239 m_itemCount++;
240 return pReturn;
241 }
242}
243
244template<typename T>
245inline void PoolAllocator<T>::deallocate(T* const obj) noexcept { // Axivion Line AutosarC++19_03-A8.4.8: false-positive: it is a input parameter
246 ASSERT_EX1(nullptr != obj, "passed nullptr to ObjectPoolAllocator::deallocate");
247 // AXIVION Construct AutosarC++19_03-M5.0.18: pointer comparision of unrelated pointers required to determine if they are from this allocator
248 ASSERT_EX(m_pStartAddr <= obj);
249 ASSERT_EX(m_pEndAddr > obj);
250
251 if (!std::is_trivially_copyable<T>::value) { // AXIVION Line AutosarC++19_03-M0.1.2, AutosarC++19_03-M0.1.9: condition can be always true in a project
252 obj->~T();
253 }
254 // AXIVION Next Line AutosarC++19_03-A5.2.4: Allowed usage, because the node must be valid due to the static_assert
255 Node* const pNode {reinterpret_cast<Node*>(obj)};
256 pNode->m_pNext = m_pFreeList; // AXIVION Line AutosarC++19_03-A5.3.2 : pNode NULL check is done via the obj NULL check above
257 m_pFreeList = pNode;
258 m_itemCount--; // AXIVION Line AutosarC++19_03-A4.7.1 / AutosarC++19_03-M0.3.1: underflow not possible then not more obj can exist then the counter value
259}
260
261} // namespace util
262} // namespace core
263} // namespace base
264} // namespace imt
265
266#endif // IMT_BASE_CORE_UTIL_POOL_ALLOCATOR_H
void ASSERT_EX(bool const condition) noexcept
Definition Diagnostics.h:66
void ASSERT_EX1(bool const condition, char_t const *const pMessage) noexcept
"Assert and throw exception" (ASSERT_EX).
Definition Diagnostics.h:55
fixed size pool allocator
void const * endAddr() const noexcept
returns the last valid item address in the pool allocator
bool isFull() const noexcept
get is allocator full flag
void const * startAddr() const noexcept
returns the start address
T * allocate() noexcept
Instantiates an object out of the memory pool.
void deallocate(T *const obj) noexcept
Deallocates the pool memory pointed by obj.
PoolAllocator(T pool[], size_t poolSize) noexcept
Ctor.
size_t remainingSize() const noexcept
return the remaining size of Ts
static size_t getValueSize() noexcept
get size of T
size_t capacity() const noexcept
return the maixmal size
This is a application specific file which is used to configure Imt.Base.Core.Math.
unsigned __int8 uint8_t
Definition stdint.h:62