Imt.Base C++ API V4.1.1.0
Loading...
Searching...
No Matches
LinkedList.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_LINKED_LIST_H
37#define IMT_BASE_CORE_UTIL_LINKED_LIST_H
38
42#include <type_traits>
43
44namespace imt {
45namespace base {
46namespace core {
47namespace util {
48
52enum class LinkedListType : uint8_t {
53 SINGLE,
54 DOUBLE
55};
56
63// AXIVION Next Construct AutosarC++19_03-A9.6.1 : layout of data is not important for the linked list implementation (no HW interfacing)
64template<typename T, LinkedListType type = LinkedListType::SINGLE>
65struct ListNode { // AXIVION Line AutosarC++19_03-A9.6.1 : layout of data is not important for the linked list implementation (no HW interfacing)
66};
67
75template<typename T>
76struct ListNode<T, LinkedListType::SINGLE> { // AXIVION Line AutosarC++19_03-A9.6.1 : layout of data is not important for the linked list implementation (no HW interfacing)
77 static constexpr LinkedListType LIST_TYPE {LinkedListType::SINGLE}; // AXIVION Line AutosarC++19_03-M0.1.4: is referenced in template push_back below
81 using ItemType = T;
82
87
92};
93
101template<typename T>
102struct ListNode<T, LinkedListType::DOUBLE> { // AXIVION Line AutosarC++19_03-A9.6.1 : layout of data is not important for the linked list implementation (no HW interfacing)
103 static constexpr LinkedListType LIST_TYPE {LinkedListType::DOUBLE}; // AXIVION Line AutosarC++19_03-M0.1.4: is referenced in template push_back below
104
108 using ItemType = T;
109
114
119
124};
125
144template<typename Node, typename T = typename Node::ItemType, typename Allocator = PoolAllocator<Node>, LinkedListType type = LinkedListType::SINGLE>
145class LinkedList final {
146public:
147
151 // AXIVION Line AutosarC++19_03-A9.6.1 : layout of data is not important for the linked list implementation (no HW interfacing)
152 using ItemType = typename Node::ItemType;
153
161 class Iterator {
162 public:
163
169 Iterator(Node* ptr = nullptr) noexcept :
170 m_pNode {ptr} {
171 }
172
176 Iterator& operator=(Node* ptr) & noexcept {
177 if (m_pNode == nullptr) {
178 ASSERT_DEBUG(false);
179 return *this;
180 }
181 m_pNode = ptr;
182 return *this;
183 }
184
192 friend bool operator==(Iterator const& lhs, Iterator const& rhs) noexcept {
193 return (lhs.m_pNode == rhs.m_pNode);
194 }
195
203 friend bool operator!=(Iterator const& lhs, Iterator const& rhs) noexcept {
204 return !(lhs == rhs);
205 }
206
213 Iterator& operator++() noexcept {
214 if (m_pNode != nullptr) {
215 m_pNode = m_pNode->m_pNext;
216 }
217 else {
218 // do nothing
219 }
220 return *this;
221 }
222
229 template<typename N = Node, typename std::enable_if<N::LIST_TYPE == LinkedListType::DOUBLE, bool>::type = true>
230 Iterator& operator--() noexcept {
231 if (m_pNode != nullptr) {
232 m_pNode = m_pNode->m_pPrev;
233 }
234 return *this;
235 }
236
242 Node& operator*() noexcept {
243 ASSERT_EX(m_pNode != nullptr);
244 return *m_pNode;
245 }
246
252 Node const& operator*() const noexcept {
253 ASSERT_EX(m_pNode == nullptr);
254 return *m_pNode;
255 }
256
262 Node* operator->() const noexcept {
263 return m_pNode;
264 }
265
271 // AXIVION Next Line AutosarC++19_03-A13.5.3: STL Interface
272 explicit operator bool() const noexcept {
273 return m_pNode != nullptr;
274 }
275
282 Node* getPtr() const noexcept {
283 return m_pNode;
284 }
285
291 Node const* getConstPtr() const noexcept {
292 return m_pNode;
293 }
294
295 private:
296
300 // AXIVION Next Line AutosarC++19_03-M11.0.1: Allowed usage
301 Node* m_pNode;
302 };
303
307 using ConstIterator = typename LinkedList<Node const>::Iterator; // AXIVION Line AutosarC++19_03-A14.7.1: false positive: all functions are disabled who require m_pPrev when using iterator for single-linked-nodes
308
315 LinkedList(PoolAllocator<Node>& listAllocator, size_t listSize) noexcept :
316 m_allocator {listAllocator},
317 m_listSize {listSize},
318 m_pHead {nullptr},
319 m_pTail {nullptr},
320 m_currentSize {0} {
321 }
322
323 // AXIVION Next Construct CodingStyle-Naming.Method: Allowed naming, STL iterator API.
324 Iterator begin() noexcept {
325 // AXIVION Next Line AutosarC++19_03-A5.2.3: Removal of volatile required
326 return Iterator(const_cast<Node*>(m_pHead));
327 }
328
329 // AXIVION Next Construct CodingStyle-Naming.Method: Allowed naming, STL iterator API.
330 // AXIVION Next Construct AutosarC++19_03-M9.3.3: Must not be static to match STL iterator API.
331 ConstIterator cbegin() const noexcept {
332 // AXIVION Next Line AutosarC++19_03-A5.2.3: Removal of volatile required
333 return ConstIterator(const_cast<Node const*>(m_pHead));
334 }
335
336 // AXIVION Next Construct CodingStyle-Naming.Method: Allowed naming, STL iterator API.
337 // AXIVION Next Construct AutosarC++19_03-M9.3.3: Must not be static to match STL iterator API.
338 Iterator end() noexcept {
339 return nullptr;
340 }
341
342 // AXIVION Next Construct CodingStyle-Naming.Method: Allowed naming, STL iterator API.
343 // AXIVION Next Construct AutosarC++19_03-M9.3.3: Must not be static to match STL iterator API.
344 ConstIterator cend() const noexcept {
345 return nullptr;
346 }
347
354 // AXIVION Next Construct CodingStyle-Naming.Method: Allowed naming, STL iterator API.
355 template<typename N = Node, typename std::enable_if<N::LIST_TYPE == LinkedListType::SINGLE, bool>::type = true>
356 void push_back(typename N::ItemType const& e) noexcept {
357 Node* const pNewNode {m_allocator.allocate()};
358 if (pNewNode == nullptr) {
359 // AXIVION Next Line AutosarC++19_03-A5.1.1: Usage allowed, defined assert message
360 ASSERT_DEBUG1(false, "list allocator full");
361 return;
362 }
363 bool const headIsNullptr {m_pHead == nullptr};
364 bool const tailIsNullptr {m_pTail == nullptr};
365 if (!headIsNullptr && tailIsNullptr) {
366 ASSERT_DEBUG1(false, "m_pTail is in invalid state"); // Axivion Line AutosarC++19_03-A5.1.1 : magic string literal required here
367 return;
368 }
369 pNewNode->m_item = e;
370 pNewNode->m_pNext = nullptr;
371
372 if (m_pHead == nullptr) {
373 m_pHead = pNewNode;
374 m_pTail = pNewNode;
375 }
376 else {
377 m_pTail->m_pNext = pNewNode;
378 m_pTail = m_pTail->m_pNext;
379 }
380 m_currentSize++;
381 }
382
389 // AXIVION Next Construct CodingStyle-Naming.Method: Allowed naming, STL iterator API.
390 template<typename N = Node, typename std::enable_if<N::LIST_TYPE == LinkedListType::DOUBLE, bool>::type = true>
391 void push_back(typename N::ItemType const& e) noexcept {
392 Node* const pNewNode {m_allocator.allocate()};
393 if (pNewNode == nullptr) {
394 // AXIVION Next Line AutosarC++19_03-A5.1.1: Usage allowed, defined assert message
395 ASSERT_DEBUG1(false, "list allocator full");
396 return;
397 }
398 bool const headIsNullptr {m_pHead == nullptr};
399 bool const tailIsNullptr {m_pTail == nullptr};
400 if (!headIsNullptr && tailIsNullptr) {
401 ASSERT_DEBUG1(false, "m_pTail is in invalid state"); // Axivion Line AutosarC++19_03-A5.1.1 : magic string literal required here
402 return;
403 }
404 pNewNode->m_item = e;
405 pNewNode->m_pNext = nullptr;
406
407 if (m_pHead == nullptr) {
408 pNewNode->m_pPrev = nullptr;
409 m_pHead = pNewNode;
410 m_pTail = pNewNode;
411 }
412 else {
413 // AXIVION Next Line AutosarC++19_03-A5.2.3: Removal of volatile required
414 pNewNode->m_pPrev = const_cast<Node*>(m_pTail);
415 m_pTail->m_pNext = pNewNode;
416 m_pTail = m_pTail->m_pNext;
417 }
418 m_currentSize++;
419 }
420
426 // AXIVION Next Construct CodingStyle-Naming.Method: Allowed naming, STL iterator API.
427 void pop_front() noexcept {
428 ASSERT_EX(m_pHead != nullptr);
429 // AXIVION Next Line AutosarC++19_03-A5.2.3: Removal of volatile required
430 Node* pOldNode {const_cast<Node*>(m_pHead)};
431 m_pHead = m_pHead->m_pNext;
432 m_allocator.deallocate(pOldNode);
433 m_currentSize--;
434 }
435
442 // AXIVION Next Construct CodingStyle-Naming.Method: Allowed naming, STL iterator API.
443 ItemType& front() noexcept {
444 ASSERT_EX(m_pHead != nullptr);
445 return *const_cast<T*>(&m_pHead->m_item);
446 }
447
454 // AXIVION Next Construct CodingStyle-Naming.Method: Allowed naming, STL iterator API.
455 ItemType const& front() const noexcept {
456 ASSERT_EX(m_pHead != nullptr);
457 return static_cast<T>(m_pHead->m_item);
458 }
459
465 // AXIVION Next Construct CodingStyle-Naming.Method: Allowed naming, STL iterator API.
466 size_t size() const noexcept {
467 return m_currentSize;
468 }
469
475 // AXIVION Next Construct CodingStyle-Naming.Method: Allowed naming, STL iterator API.
476 bool empty() const noexcept {
477 return m_currentSize == 0;
478 }
479
485 // AXIVION Next Construct CodingStyle-Naming.Method: Allowed naming, STL iterator API.
486 bool full() const noexcept {
487 return m_listSize == m_currentSize;
488 }
489
495 // AXIVION Next Construct CodingStyle-Naming.Method: Allowed naming, STL iterator API.
496 template<typename I = ItemType, typename N = Node, typename std::enable_if<N::LIST_TYPE == LinkedListType::DOUBLE, bool>::type = true>
497 void remove(I const& item) noexcept {
498 Iterator it {begin()};
499 while (it != end()) {
500 if (it->m_item == item) {
501 if ((it.getPtr() == m_pTail) && (it.getPtr() == m_pHead)) { // remove the only item
502 m_pHead = nullptr;
503 m_pTail = nullptr;
504 }
505 else if (it.getPtr() == m_pTail) { // remove the last item
506 if (m_pTail != nullptr) {
507 m_pTail = m_pTail->m_pPrev;
508 }
509 else {
510 ASSERT_DEBUG1(false, "m_pTail is null"); // Axivion Line AutosarC++19_03-A5.1.1 : magic string literal required here
511 }
512 }
513 else if (it.getPtr() == m_pHead) { // remove the first item
514 if (m_pHead != nullptr) {
515 m_pHead = m_pHead->m_pNext;
516 }
517 else {
518 ASSERT_DEBUG1(false, "m_pHead is null"); // Axivion Line AutosarC++19_03-A5.1.1 : magic string literal required here
519 }
520 }
521 else {
522 it->m_pPrev->m_pNext = it->m_pNext;
523 it->m_pNext->m_pPrev = it->m_pPrev;
524 }
525 m_allocator.deallocate(it.getPtr());
526 m_currentSize--;
527 break;
528 }
529 ++it;
530 }
531 }
532
539 // AXIVION Next Construct CodingStyle-Naming.Method: Allowed naming, STL iterator API.
540 template<typename TT>
541 bool contains(const TT& item) const noexcept {
542 ConstIterator it {cbegin()};
543 while (it != cend()) {
544 if (it->m_item == item) {
545 return true;
546 }
547 ++it;
548 }
549 return false;
550 }
551
552private:
553
554 PoolAllocator<Node>& m_allocator;
555
556 // The size of the list.
557 // AXIVION Next Codeline AutosarC++19_03-M0.1.4: field may be not used in member function if full() is not called in the project
558 size_t const m_listSize;
559
560 // Head index, points to the element right after the last written element.
561 // AXIVION Next Codeline AutosarC++19_03-A2.11.1: Allowed usage, prevents optimizer from reordering
562 // AXIVION Next Codeline AutosarC++19_03-A12.1.3: All data members initialized in custom constructor and would collide with AutosarC++19_03-A12.1.2
563 Node volatile* m_pHead;
564
565 // Tail index, points to the element right after the last read element.
566 // AXIVION Next Codeline AutosarC++19_03-A2.11.1: Allowed usage, prevents optimizer from reordering
567 // AXIVION Next Codeline AutosarC++19_03-A12.1.3: All data members initialized in custom constructor and would collide with AutosarC++19_03-A12.1.2
568 Node volatile* m_pTail;
569
570 // AXIVION Next Codeline AutosarC++19_03-A12.1.3: All data members initialized in custom constructor and would collide with AutosarC++19_03-A12.1.2
571 size_t m_currentSize;
572};
573
574} // namespace util
575} // namespace core
576} // namespace base
577} // namespace imt
578
579#endif // IMT_BASE_CORE_UTIL_LINKED_LIST_H
void ASSERT_DEBUG1(bool const condition, char_t const *const pMessage) noexcept
"Assert for debugging only" (ASSERT_DEBUG).
Definition Diagnostics.h:77
void ASSERT_EX(bool const condition) noexcept
Definition Diagnostics.h:66
void ASSERT_DEBUG(bool const condition) noexcept
Definition Diagnostics.h:88
Iterator & operator--() noexcept
Decrements operator.
Definition LinkedList.h:230
Node const & operator*() const noexcept
Gets the current Node as const reference.
Definition LinkedList.h:252
Node * operator->() const noexcept
Dereferencing operator.
Definition LinkedList.h:262
Iterator & operator=(Node *ptr) &noexcept
Asignment operator.
Definition LinkedList.h:176
Node & operator*() noexcept
Gets the current Node as reference.
Definition LinkedList.h:242
Node * getPtr() const noexcept
Gets current pointer.
Definition LinkedList.h:282
friend bool operator==(Iterator const &lhs, Iterator const &rhs) noexcept
Equal operator.
Definition LinkedList.h:192
Node const * getConstPtr() const noexcept
Gets current pointer as const.
Definition LinkedList.h:291
friend bool operator!=(Iterator const &lhs, Iterator const &rhs) noexcept
Not equal operator.
Definition LinkedList.h:203
Iterator & operator++() noexcept
Increments operator.
Definition LinkedList.h:213
Iterator(Node *ptr=nullptr) noexcept
Constructor.
Definition LinkedList.h:169
bool empty() const noexcept
Returns if the list is empty.
Definition LinkedList.h:476
void pop_front() noexcept
Removes the front element of the list.
Definition LinkedList.h:427
typename Node::ItemType ItemType
List item type.
Definition LinkedList.h:152
bool contains(const TT &item) const noexcept
Checks if a certain item0 is in the list.
Definition LinkedList.h:541
LinkedList(PoolAllocator< Node > &listAllocator, size_t listSize) noexcept
Constructor.
Definition LinkedList.h:315
typename LinkedList< Node const >::Iterator ConstIterator
Iterator for the list.
Definition LinkedList.h:307
ItemType const & front() const noexcept
Returns the front element of the list.
Definition LinkedList.h:455
ConstIterator cend() const noexcept
Definition LinkedList.h:344
ItemType & front() noexcept
Returns the front element of the list.
Definition LinkedList.h:443
ConstIterator cbegin() const noexcept
Definition LinkedList.h:331
void push_back(typename N::ItemType const &e) noexcept
Adds a new element to the back of the SINGLE list.
Definition LinkedList.h:356
bool full() const noexcept
Returns if the list is full.
Definition LinkedList.h:486
void remove(I const &item) noexcept
Removes a valid item from the list.
Definition LinkedList.h:497
size_t size() const noexcept
Returns the current number of elements in the linked list.
Definition LinkedList.h:466
fixed size pool allocator
T * allocate() noexcept
Instantiates an object out of the memory pool.
void deallocate(T *const obj) noexcept
Deallocates the pool memory pointed by obj.
LinkedListType
Enum for the linked list types.
Definition LinkedList.h:52
@ DOUBLE
use this enumerator to use the single linked list implementation
This is a application specific file which is used to configure Imt.Base.Core.Math.
#define I
#define bool
Definition stdbool.h:42
unsigned __int8 uint8_t
Definition stdint.h:62
Node used for LinkedList class.
Definition LinkedList.h:65