GM6000 Digital Heater Controller Branch: main
SDX-1330
SList.h
Go to the documentation of this file.
1#ifndef Cpl_Container_SList_h_
2#define Cpl_Container_SList_h_
3/*-----------------------------------------------------------------------------
4* This file is part of the Colony.Core Project. The Colony.Core Project is an
5* open source project with a BSD type of licensing agreement. See the license
6* agreement (license.txt) in the top/ directory or on the Internet at
7* http://integerfox.com/colony.core/license.txt
8*
9* Copyright (c) 2014-2022 John T. Taylor
10*
11* Redistributions of the source code must retain the above copyright notice.
12*----------------------------------------------------------------------------*/
13/** @file */
14
15#include "Cpl/Container/Item.h"
16
17
18///
19namespace Cpl {
20///
21namespace Container {
22
23
24/** This template class implements a singly linked list
25 which maintains the ordering imposed on it by the application.
26 NOTE: ITEM must be a subclass of Item.
27 */
28template <class ITEM>
29class SList
30{
31private:
32 /** Points to the first item in the list.
33 */
34 ITEM * m_headPtr;
35
36 /** Points to the last item in the list.
37 */
38 ITEM* m_tailPtr;
39
40public:
41 /** Public constructor initializes head and tail pointers.
42 */
44
45 /** This is a special constructor for when the list is
46 statically declared (i.e. it is initialized as part of
47 C++ startup BEFORE main() is executed. C/C++ guarantees
48 that all statically declared data will be initialized
49 to zero by default (see r.8.4 in C++ Programming Language,
50 Second Edition). Since the head & tail pointers are
51 initialized to zero - then the C/C++ default is OK. Why
52 do I care about all this? Because if you attempt to build
53 static list(s), then the order of when the list is
54 constructed vs. when the static elements are added to the
55 list is a problem! To avoid this problem, a alternate
56 constructor that does NOT initialize the head & tail pointers
57 is provided. It assumes that the pointers are already set
58 zero because the list is "static". Whew!
59 */
61
62
63public: ///@name Operations on the List itself
64 ///@{
65 /** Moves the content of the this queue to the specified queue.
66 */
68
69 /** Empties the list. All references to the item(s) in the
70 list are lost.
71 */
73 ///@}
74
75public: ///@name View as a FIFO
76 ///@{
77 /** Removes the first item in the list. Returns 0 if the list
78 is empty.
79 */
80 ITEM* get( void ) noexcept;
81
82 /** Adds the item as the last item in the list
83 */
84 void put( ITEM& item ) noexcept;
85
86 /** Return a pointer to the first item in the list.
87 The returned item remains in the list. Returns 0
88 if the list is empty.
89 */
91
92 /** Return a pointer to the last item in the list.
93 The returned item remains in the list. Returns 0
94 if the list is empty.
95 */
97 ///@}
98
99
100public: ///@name View as a STACK
101 ///@{
102 /** Removes the top element from stack and return a pointer to
103 it as a result. Returns 0, if the stack is empty
104 */
105 ITEM* pop( void ) noexcept;
106
107 /** Adds the ITEM item to top of the stack.
108 */
109 void push( ITEM& item ) noexcept;
110
111 /** Return a pointer to the top ITEM item in the stack.
112 The returned item remains in the queue. Returns 0
113 if the stack is empty.
114 */
116 ///@}
117
118public: ///@name View as Ordered List
119 ///@{
120 /** Removes the first item in the list. Returns 0 if the list
121 is empty.
122 */
124
125 /** Removes the last item in the list. Returns 0 if the list
126 is empty.
127 */
129
130 /** Adds the item as the first item in the list.
131 */
132 void putFirst( ITEM& item ) noexcept;
133
134 /** Adds the item as the last item in the list.
135 */
136 void putLast( ITEM& item ) noexcept;
137
138 /** Remove specified ITEM element from the list.
139 Returns true if the specified element was found and
140 removed from the list, else false.
141 */
142 bool remove( ITEM& item ) noexcept;
143
144 /** Insert the "item" ITEM into the list behind the
145 "after" ITEM element. If 'after' is 0, then 'item'
146 is added to the head of the list.
147 */
149
150 /** Insert the "item" ITEM into the list ahead of the
151 "before" ITEM element. If 'before' is 0, then 'item'
152 is added to the tail of the list. Note: This insert
153 operation is more expensive than insertAfter() because
154 a traversal of the list is required to find the
155 'before' item
156 */
158
159 /** Returns true if the specified item is already in the
160 list, else false.
161 */
162 bool find( const ITEM& item ) const noexcept;
163
164 /** Return a pointer to the first item in the list.
165 The returned item remains in the list. Returns 0
166 if the list is empty.
167 */
169
170 /** Return a pointer to the last item in the list.
171 The returned item remains in the list. Returns 0
172 if the list is empty.
173 */
175
176 /** Return a pointer to the item after the item "item".
177 Both items remain in the list. Returns 0 when the
178 end-of-list is reached.
179 */
181 ///@}
182
183
184private: ///@name Prevent the container from being copied
185 ///@{
186 /// Prevent access to the copy constructor -->Containers can not be copied!
187 SList( const SList& m );
188
189 /// Prevent access to the assignment operator -->Containers can not be copied!
191 ///@}
192
193};
194
195/////////////////////////////////////////////////////////////////////////////
196// INLINE IMPLEMENTAION
197/////////////////////////////////////////////////////////////////////////////
198
199
202 :m_headPtr( 0 ), m_tailPtr( 0 )
203{
204}
205
206template <class ITEM>
207SList<ITEM>::SList( const char* /* notUsed */ ) noexcept
208{
209}
210
211
212template <class ITEM>
213inline void SList<ITEM>::move( SList<ITEM>& dst ) noexcept
214{
215 // clear the destination list
216 dst.clearTheList();
217
218 // Copy each item (so the debug info is correct)
219 ITEM* nextPtr;
220 while ( (nextPtr=get()) )
221 {
222 dst.put( *nextPtr );
223 }
224}
225
226template <class ITEM>
228{
229 // Drain list so the debug traps work correctly
230 while ( get() )
231 ;
232}
233
234template <class ITEM>
235inline ITEM* SList<ITEM>::get( void ) noexcept
236{
237 ITEM* nextPtr;
238 if ( (nextPtr=m_headPtr) )
239 {
240 if ( !(m_headPtr=(ITEM*) (nextPtr->m_nextPtr_)) )
241 {
242 m_tailPtr = 0;
243 }
244 }
245 Item::remove_( nextPtr );
246 return nextPtr;
247}
248
249template <class ITEM>
250inline ITEM* SList<ITEM>::getFirst( void ) noexcept
251{
252 return get();
253}
254
255template <class ITEM>
256inline ITEM* SList<ITEM>::getLast( void ) noexcept
257{
258 ITEM* lastPtr = m_tailPtr;
259 if ( lastPtr )
260 {
261 remove( *m_tailPtr );
262 }
263 return lastPtr;
264}
265
266template <class ITEM>
267inline void SList<ITEM>::putFirst( ITEM& item ) noexcept
268{
269 if ( item.insert_( this ) )
270 {
271 if ( m_headPtr )
272 {
273 item.m_nextPtr_ = m_headPtr;
274 m_headPtr = &item;
275 }
276 else
277 {
278 m_headPtr = m_tailPtr = &item;
279 item.m_nextPtr_ = 0;
280 }
281 }
282}
283
284template <class ITEM>
285inline void SList<ITEM>::put( ITEM& item ) noexcept
286{
287 if ( item.insert_( this ) )
288 {
289 if ( m_headPtr )
290 {
291 m_tailPtr->m_nextPtr_ = &item;
292 }
293 else
294 {
295 m_headPtr = &item;
296 }
297 item.m_nextPtr_ = 0;
298 m_tailPtr = &item;
299 }
300}
301
302template <class ITEM>
303inline void SList<ITEM>::putLast( ITEM& item ) noexcept
304{
305 put( item );
306}
307
308template <class ITEM>
309inline bool SList<ITEM>::remove( ITEM& item ) noexcept
310{
311 if ( item.isInContainer_( this ) )
312 {
313 ITEM* nxtPtr;
314 ITEM* prvPtr;
316 {
317 if ( nxtPtr == &item )
318 {
319 if ( prvPtr )
320 {
321 if ( !(prvPtr->m_nextPtr_=nxtPtr->m_nextPtr_) )
322 {
323 m_tailPtr = prvPtr;
324 }
325 }
326 else
327 {
328 if ( !(m_headPtr=(ITEM*) nxtPtr->m_nextPtr_) )
329 {
330 m_tailPtr = 0;
331 }
332 }
333 Item::remove_( &item );
334 return true;
335 }
336 }
337 // Note: I should never get here (i have already checked that I am in list at the top of the method)
338 }
339 return false;
340}
341
342
343template <class ITEM>
344inline void SList<ITEM>::insertAfter( ITEM& after, ITEM& item ) noexcept
345{
346 if ( item.insert_( this ) )
347 {
348 item.m_nextPtr_ = after.m_nextPtr_;
349 if ( !item.m_nextPtr_ )
350 {
351 m_tailPtr = &item;
352 }
353 after.m_nextPtr_ = &item;
354 }
355}
356
357template <class ITEM>
358inline void SList<ITEM>::insertBefore( ITEM& before, ITEM& item ) noexcept
359{
360 if ( item.insert_( this ) )
361 {
362 ITEM* nxtPtr;
363 ITEM* prvPtr;
365 {
366 if ( nxtPtr == &before )
367 {
368 item.m_nextPtr_ = nxtPtr;
369 if ( prvPtr )
370 {
371 prvPtr->m_nextPtr_ = &item;
372 }
373 else {
374 m_headPtr = &item;
375 }
376 break;
377 }
378 }
379 }
380}
381
382template <class ITEM>
383inline bool SList<ITEM>::find( const ITEM& item ) const noexcept
384{
385 return item.isInContainer_( this );
386}
387
388template <class ITEM>
389inline ITEM* SList<ITEM>::first( void ) const noexcept
390{
391 return m_headPtr;
392}
393
394template <class ITEM>
395inline ITEM* SList<ITEM>::last( void ) const noexcept
396{
397 return m_tailPtr;
398}
399
400template <class ITEM>
401inline ITEM* SList<ITEM>::next( const ITEM& item ) const noexcept
402{
403 return (ITEM*) (item.m_nextPtr_);
404}
405
406template <class ITEM>
407inline void SList<ITEM>::push( ITEM& item ) noexcept
408{
409 putFirst( item );
410}
411
412template <class ITEM>
413inline ITEM* SList<ITEM>::pop( void ) noexcept
414{
415 return get();
416}
417
418template <class ITEM>
419inline ITEM* SList<ITEM>::top( void ) const noexcept
420{
421 return first();
422}
423
424template <class ITEM>
425inline ITEM* SList<ITEM>::head( void ) const noexcept
426{
427 return first();
428}
429
430template <class ITEM>
431inline ITEM* SList<ITEM>::tail( void ) const noexcept
432{
433 return last();
434}
435
436
437}; // end namespaces
438};
439#endif // end header latch
static void remove_(Item *itemPtr) noexcept
Helper method to do the proper 'clean-up' for the multiple-containers-error-trap when removing an ite...
This template class implements a THREAD SAFE Ring Buffer.
Definition RingBufferMT.h:33
This template class implements a singly linked list which maintains the ordering imposed on it by the...
Definition SList.h:30
void put(ITEM &item) noexcept
Adds the item as the last item in the list.
Definition SList.h:285
void putLast(ITEM &item) noexcept
Adds the item as the last item in the list.
Definition SList.h:303
bool find(const ITEM &item) const noexcept
Returns true if the specified item is already in the list, else false.
Definition SList.h:383
ITEM * getLast(void) noexcept
Removes the last item in the list.
Definition SList.h:256
void putFirst(ITEM &item) noexcept
Adds the item as the first item in the list.
Definition SList.h:267
void clearTheList() noexcept
Empties the list.
Definition SList.h:227
void move(SList< ITEM > &dst) noexcept
Moves the content of the this queue to the specified queue.
Definition SList.h:213
ITEM * getFirst(void) noexcept
Removes the first item in the list.
Definition SList.h:250
SList() noexcept
Public constructor initializes head and tail pointers.
Definition SList.h:201
ITEM * pop(void) noexcept
Removes the top element from stack and return a pointer to it as a result.
Definition SList.h:413
ITEM * last(void) const noexcept
Return a pointer to the last item in the list.
Definition SList.h:395
ITEM * head(void) const noexcept
Return a pointer to the first item in the list.
Definition SList.h:425
void push(ITEM &item) noexcept
Adds the ITEM item to top of the stack.
Definition SList.h:407
ITEM * tail(void) const noexcept
Return a pointer to the last item in the list.
Definition SList.h:431
ITEM * get(void) noexcept
Removes the first item in the list.
Definition SList.h:235
void insertBefore(ITEM &before, ITEM &item) noexcept
Insert the "item" ITEM into the list ahead of the "before" ITEM element.
Definition SList.h:358
ITEM * next(const ITEM &item) const noexcept
Return a pointer to the item after the item "item".
Definition SList.h:401
ITEM * first(void) const noexcept
Return a pointer to the first item in the list.
Definition SList.h:389
bool remove(ITEM &item) noexcept
Remove specified ITEM element from the list.
Definition SList.h:309
ITEM * top(void) const noexcept
Return a pointer to the top ITEM item in the stack.
Definition SList.h:419
void insertAfter(ITEM &after, ITEM &item) noexcept
Insert the "item" ITEM into the list behind the "after" ITEM element.
Definition SList.h:344
The 'Cpl' namespace is the root name space for the Colony.
Definition Api16.h:20