GM6000 Digital Heater Controller Branch: main
SDX-1330
RingBuffer.h
Go to the documentation of this file.
1#ifndef Cpl_Container_RingBuffer_h_
2#define Cpl_Container_RingBuffer_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 <stdint.h>
16
17
18///
19namespace Cpl {
20///
21namespace Container {
22
23
24/** This template class implements a Ring Buffer. The size of the
25 ring buffer is limited by number of bits in platform's 'unsigned'
26 data type.
27
28 Thread/ISR Safety Notes:
29 - The only mutable (once the Ring buffer is created) data members of the
30 class are the head/tail pointers and the buffer contents.
31 - The add() operation only modifies the tail pointer (but does read the
32 head pointer)
33 - The remove() operation only modifies the head pointer (but does read
34 the tail pointer).
35 - The clearTheBuffer() operation modifies both the head and tail pointers
36 - All other methods only read data members
37 - The implementation ASSUMES there is a single producer and a single
38 consumer of the RingBuffer. And that ONLY the producer/consumer
39 are the invoking operations on the RingBuffer.
40
41 Template Args:
42 ITEM:= Type of the data stored in the Ring Buffer
43 */
44template <class ITEM>
46{
47private:
48 /// Points to the first item in the buffer.
49 ITEM* m_headPtr;
50
51 /// Points to the last item in the buffer.
52 ITEM* m_tailPtr;
53
54 /// Number of element in the allocate memory
55 const unsigned m_memoryNumElements;
56
57 /// End of the Memory element storage
58 ITEM* const m_endOfMemPtr;
59
60 /// Memory for the Elements
61 ITEM* const m_elements;
62
63
64public:
65 /** Constructor. The application is responsible for providing the memory
66 for the Ring Buffer. The argument ''numElements' is the number of
67 items that will fit in the memory allocated by 'memoryForElements' - it
68 is NOT the number of bytes of 'memoryForElements'.
69
70 Note: The maximum number of element that can actually be stored is
71 numElements - 1 (one element/index/slot is consumed/used to
72 represents the empty buffer state).
73 */
74 RingBuffer( unsigned numElements, ITEM memoryForElements[] ) noexcept;
75
76
77
78public:
79 /** Removes the first item in the Buffer. The contents of the
80 removed item will be copied into the 'dst' argument. The method
81 return true if the operation was successful; else false is
82 returned, i.e. the Ring buffer is/was empty.
83 */
84 bool remove( ITEM& dst ) noexcept;
85
86
87 /** The contents of 'item' will be copied into the Ring Buffer as the
88 'last' item in the buffer. Return true if the operation was
89 successful; else false is returned, i.e. the Buffer was full prior to
90 the attempted add().
91 */
92 bool add( const ITEM& item ) noexcept;
93
94
95 /** Returns a pointer to the first item in the Buffer.
96 The returned item remains in the buffer. Returns 0
97 if the Buffer is empty.
98 */
99 ITEM* peekHead( void ) const noexcept;
100
101
102 /** Returns a pointer to the last item in the Buffer.
103 The returned item remains in the Buffer. Returns 0
104 if the Buffer is empty.
105 */
106 ITEM* peekTail( void ) const noexcept;
107
108
109public:
110 /** This method returns true if the Ring Buffer is empty
111 */
112 bool isEmpty( void ) const noexcept;
113
114 /** This method returns true if the Ring Buffer is full
115 */
116 bool isFull( void ) const noexcept;
117
118
119 /** This method returns the current number of items in
120 the Ring Buffer
121 */
122 unsigned getNumItems( void ) const noexcept;
123
124
125 /** This method returns the maximum number of items that
126 can be stored in the Ring buffer.
127 */
128 unsigned getMaxItems( void ) const noexcept;
129
130
131
132public:
133 /** Empties the Ring Buffer. All references to the item(s) in the
134 buffer are lost.
135 */
136 void clearTheBuffer() noexcept;
137
138public:
139 /** This method returns a pointer to the next item to be removed. In addition
140 it returns the number of elements that can be removed as linear/flat
141 buffer (i.e. without wrapping around raw buffer memory)
142
143 If the Ring buffer is empty, a null pointer is returned
144 */
146
147 /** This method 'removes' N elements - that were removed using the
148 pointer returned from peekNextRemoveItems - from the ring buffer.
149 Basically it updates the head pointer to reflect items removed using
150 direct memory access.
151
152 'numElementsToRemove' be less than or equal to the 'dstNumFlatElements'
153 returned from peekNextRemoveItems().
154
155 CAUTION: IF YOU DON'T UNDERSTAND THE USE CASE FOR THIS METHOD - THEN
156 DON'T USE IT. If this method is used improperly, it WILL
157 CORRUPT the Ring Buffer!
158 */
160
161public:
162 /** This method returns a pointer to the next item to be added. In addition
163 it returns the number of elements that can be added as linear/flat
164 buffer (i.e. without wrapping around raw buffer memory)
165
166 If the Ring buffer is full, a null pointer is returned
167 */
169
170 /** This method 'adds' N elements - that were populated using the
171 pointer returned from peekNextAddItems - to the ring buffer. Basically
172 its updates the tail pointer to reflect items added using direct
173 memory access.
174
175 'numElementsAdded' be less than or equal to the 'dstNumFlatElements'
176 returned from peekNextAddItems().
177
178 CAUTION: IF YOU DON'T UNDERSTAND THE USE CASE FOR THIS METHOD - THEN
179 DON'T USE IT. If this method is used improperly, it WILL
180 CORRUPT the Ring Buffer!
181 */
183
184private:
185 /// Prevent access to the copy constructor -->Containers can not be copied!
187
188 /// Prevent access to the assignment operator -->Containers can not be copied!
190
191};
192
193/////////////////////////////////////////////////////////////////////////////
194// INLINE IMPLEMENTAION
195/////////////////////////////////////////////////////////////////////////////
196
197
200 : m_headPtr( memoryForElements )
201 , m_tailPtr( memoryForElements )
202 , m_memoryNumElements( maxElements )
203 , m_endOfMemPtr( memoryForElements + maxElements )
204 , m_elements( memoryForElements )
205{
206}
207
208
209template <class ITEM>
211{
212 m_headPtr = m_tailPtr = m_elements;
213}
214
215
216template <class ITEM>
217inline bool RingBuffer<ITEM>::add( const ITEM& item ) noexcept
218{
219 if ( isFull() )
220 {
221 return false;
222 }
223
224 *m_tailPtr = item;
225 if ( ++m_tailPtr >= m_endOfMemPtr )
226 {
227 m_tailPtr = m_elements;
228 }
229
230 return true;
231}
232
233template <class ITEM>
234inline bool RingBuffer<ITEM>::remove( ITEM& dst ) noexcept
235{
236 if ( isEmpty() )
237 {
238 return false;
239 }
240
241 dst = *m_headPtr;
242 if ( ++m_headPtr >= m_endOfMemPtr )
243 {
244 m_headPtr = m_elements;
245 }
246 return true;
247}
248
249
250template <class ITEM>
252{
253 if ( isEmpty() )
254 {
255 return nullptr;
256 }
257
258 unsigned totalNumElements = getNumItems();
259 dstNumFlatElements = m_endOfMemPtr - m_headPtr;
261 {
263 }
264
265 return m_headPtr;
266}
267
268template <class ITEM>
270{
271 // By the defined semantics - I simply update the head pointer
272 m_headPtr += numElementsToRemove;
273 if ( m_headPtr >= m_endOfMemPtr )
274 {
275 m_headPtr = m_elements;
276 }
277}
278
279template <class ITEM>
281{
282 if ( isFull() )
283 {
285 return nullptr;
286 }
287
289 dstNumFlatElements = m_endOfMemPtr - m_tailPtr;
291 {
293 }
294
295 return m_tailPtr;
296}
297
298template <class ITEM>
299inline void RingBuffer<ITEM>::addElements( unsigned numElementsAdded ) noexcept
300{
301 // By the defined semantics - I simply update the tail pointer
302 m_tailPtr += numElementsAdded;
303 if ( m_tailPtr >= m_endOfMemPtr )
304 {
305 m_tailPtr = m_elements;
306 }
307}
308
309template <class ITEM>
310inline ITEM* RingBuffer<ITEM>::peekHead( void ) const noexcept
311{
312 if ( isEmpty() )
313 {
314 return 0;
315 }
316
317 return m_headPtr;
318}
319
320template <class ITEM>
321inline ITEM* RingBuffer<ITEM>::peekTail( void ) const noexcept
322{
323 if ( isEmpty() )
324 {
325 return 0;
326 }
327
328 ITEM* prevElem = m_tailPtr - 1;
329 if ( prevElem < m_elements )
330 {
331 prevElem = m_endOfMemPtr - 1;
332 }
333
334 return prevElem;
335}
336
337
338template <class ITEM>
339inline bool RingBuffer<ITEM>::isEmpty( void ) const noexcept
340{
341 return m_headPtr == m_tailPtr;
342}
343
344template <class ITEM>
345inline bool RingBuffer<ITEM>::isFull( void ) const noexcept
346{
347 ITEM* nextElem = m_tailPtr + 1;
348 if ( nextElem >= m_endOfMemPtr )
349 {
350 nextElem = m_elements;
351 }
352 return nextElem == m_headPtr;
353}
354
355template <class ITEM>
356inline unsigned RingBuffer<ITEM>::getNumItems( void ) const noexcept
357{
358 unsigned headIdx = (unsigned) (m_headPtr - m_elements);
359 unsigned tailIdx = (unsigned) (m_tailPtr - m_elements);
360 if ( tailIdx < headIdx )
361 {
362 tailIdx += m_memoryNumElements;
363 }
364 return tailIdx - headIdx;
365}
366
367template <class ITEM>
368inline unsigned RingBuffer<ITEM>::getMaxItems( void ) const noexcept
369{
370 return m_memoryNumElements - 1; // One elem/slot is reserved for the empty-list condition
371}
372
373
374
375}; // end namespaces
376};
377#endif // end header latch
This template class implements a Ring Buffer.
Definition RingBuffer.h:46
ITEM * peekTail(void) const noexcept
Returns a pointer to the last item in the Buffer.
Definition RingBuffer.h:321
RingBuffer(unsigned numElements, ITEM memoryForElements[]) noexcept
Constructor.
Definition RingBuffer.h:199
bool isFull(void) const noexcept
This method returns true if the Ring Buffer is full.
Definition RingBuffer.h:345
ITEM * peekHead(void) const noexcept
Returns a pointer to the first item in the Buffer.
Definition RingBuffer.h:310
void clearTheBuffer() noexcept
Empties the Ring Buffer.
Definition RingBuffer.h:210
void removeElements(unsigned numElementsToRemove) noexcept
This method 'removes' N elements - that were removed using the pointer returned from peekNextRemoveIt...
Definition RingBuffer.h:269
ITEM * peekNextAddItems(unsigned &dstNumFlatElements) noexcept
This method returns a pointer to the next item to be added.
Definition RingBuffer.h:280
unsigned getNumItems(void) const noexcept
This method returns the current number of items in the Ring Buffer.
Definition RingBuffer.h:356
ITEM * peekNextRemoveItems(unsigned &dstNumFlatElements) noexcept
This method returns a pointer to the next item to be removed.
Definition RingBuffer.h:251
unsigned getMaxItems(void) const noexcept
This method returns the maximum number of items that can be stored in the Ring buffer.
Definition RingBuffer.h:368
bool add(const ITEM &item) noexcept
The contents of 'item' will be copied into the Ring Buffer as the 'last' item in the buffer.
Definition RingBuffer.h:217
bool isEmpty(void) const noexcept
This method returns true if the Ring Buffer is empty.
Definition RingBuffer.h:339
bool remove(ITEM &dst) noexcept
Removes the first item in the Buffer.
Definition RingBuffer.h:234
void addElements(unsigned numElementsAdded) noexcept
This method 'adds' N elements - that were populated using the pointer returned from peekNextAddItems ...
Definition RingBuffer.h:299
This template class implements a THREAD SAFE Ring Buffer.
Definition RingBufferMT.h:33
The 'Cpl' namespace is the root name space for the Colony.
Definition Api16.h:20