mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
circularBuffer.hpp
Go to the documentation of this file.
1/** \file circularBuffer.hpp
2 * \author Jared R. Males
3 * \brief A circular buffer class
4 * \ingroup signal_processing_files
5 *
6 */
7
8//***********************************************************************//
9// Copyright 2020 Jared R. Males (jaredmales@gmail.com)
10//
11// This file is part of mxlib.
12//
13// mxlib is free software: you can redistribute it and/or modify
14// it under the terms of the GNU General Public License as published by
15// the Free Software Foundation, either version 3 of the License, or
16// (at your option) any later version.
17//
18// mxlib is distributed in the hope that it will be useful,
19// but WITHOUT ANY WARRANTY; without even the implied warranty of
20// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21// GNU General Public License for more details.
22//
23// You should have received a copy of the GNU General Public License
24// along with mxlib. If not, see <http://www.gnu.org/licenses/>.
25//***********************************************************************//
26
27#ifndef sigproc_circularBuffer
28#define sigproc_circularBuffer
29
30#include <vector>
31
32namespace mx
33{
34namespace sigproc
35{
36
37/** \defgroup circular_buffer Circular Buffer
38 * \brief A circular buffer class
39 *
40 * Three options for the circular buffer are provided, each inheriting the same underlying interface. The choices
41 * vary the way in which the wrapping is handled at the end of the storage. These are:
42 * - circularBufferMod: uses the mod operator, this is slowest in all situations and is provided for comparison
43 * - circularBufferBranch: uses an if statement (branch) to test for location in memory. Slower in sequential access,
44 * possibly slightly faster in random access.
45 * - circularBufferIndex: uses a pre-populated index array, which adds some memory overhead. Fastest in sequential
46 * access, possibly slightly slower in random access.
47 *
48 * Benchmarks are clear that circularBufferIndex is fastest for sequential acceess, e.g. one element after the other in
49 * sequential order, by something like 30%. For random access circularBufferBranch is possibly slightly faster, but not
50 * enough tests were performed to be conclusive.
51 *
52 * circularBufferMod is always much slower due to use of the `%` operator.
53 *
54 * The memory overhead of circularBufferIndex is `2*maxEntries*sizeof(indexT)`, where `maxEntries` is the maximum length
55 * of the buffer, and indexT is the type used for indexing and sizes.
56 *
57 * \todo perform circular buffer testing on an isolated core, with one test per executable
58 * \todo implement a circular buffer with fixed power-of-two size to test `&` modulo
59 *
60 * \ingroup signal_processing
61 */
62
63/// CRTP base class for all circular buffers, providing the underlying memory management and accessors.
64/** The CRTP derived classes implement a standard interface based on how they handle wrapping from the end
65 * to the beginning of the buffer.
66 *
67 * \tparam _derivedT the child class type
68 * \tparam _storedT the type stored in teh circular buffer
69 * \tparam _indexT the index type, also used for sizes (can be unsigned)
70 *
71 * \ingroup circular_buffer
72 */
73template <typename _derivedT, typename _storedT, typename _indexT>
75{
76 public:
77 typedef _derivedT derivedT; ///< The child class
78
79 typedef _storedT storedT; ///< The type stored in the circular buffer
80 typedef _indexT indexT; ///< The index type, also used for sizes
81
82 protected:
83 std::vector<storedT> m_buffer; ///< The circular buffer storage
84
85 indexT m_maxEntries{ 0 }; ///< The maximum number of entries to allow in the buffer before wrapping
86 indexT m_nextEntry{ 0 }; ///< Index into m_buffer of the next entry. This is the oldest entry in the buffer.
87 indexT m_latest{ 0 }; ///< Index into m_buff of the latest entry. This is the newest entry in the buffer.
88
89 uint64_t m_mono{ 0 }; /**< A monotonic counter, which is incremented for each new entry,
90 and reset on call to maxEntries.*/
91
92 public:
93 /// Default c'tor
95
96 /// Sizing constructor
97 /** Sets the maximum size of the buffer. Note that this will not be the size until
98 * a full set of entries have been added to the buffer.
99 */
100 explicit circularBufferBase( indexT maxEnt /**< [in] the maximum number of entries this buffer will hold*/ );
101
102 /// Set the maximum size of the buffer.
103 /** Note that this will not be the size until
104 * a full set of entries have been added to the buffer.
105 */
106 void maxEntries( indexT maxEnt /**< [in] the maximum number of entries this buffer will hold*/ );
107
108 /// Get the maximum size of the buffer.
109 /** Note that this is not the current size of the buffer until
110 * at least maxEntries have been added. Use size() to check current size.
111 *
112 * \returns the maximum size of the buffer, m_maxEntries
113 */
115
116 /// Get the number of entries.
117 /** Note that this is the current size. It will not
118 * be the same as maxEntries() until at least maxEntries
119 * have been added.
120 */
121 indexT size() const;
122
123 /// Add the next entry to the circular buffer
124 /** Adds the entry (will incur a deep copy) and updates
125 * the wrapping system.
126 */
127 void nextEntry( const storedT &newEnt /**< [in] the new entry to add to the buffer*/ );
128
129 /// Returns the index of the earliest entry
130 /** This is particularly useful for accessing entries with the at() function
131 * over an unchanging sequence if asynchronous updates are possible.
132 */
134
135 /// Returns the index of the latest entry
136 /**
137 */
139
140 uint64_t mono();
141
142 /// Get the entry at a given index
143 /** `idx=0` is the earliest entry in the buffer. `idx=1` is the one after that.
144 * I.e., this counts forward from the oldest data
145 *
146 * \returns a reference to the indicated entry in the circular buffer.
147 */
148 storedT &operator[]( indexT idx /**< [in] the index of the entry to access*/ );
149
150 const storedT &operator[]( indexT idx /**< [in] the index of the entry to access*/ ) const;
151
152 /// Get the entry at a given index relative a fixed reference entry
153 /** `idx=0` is the entry at refEntry. `idx=1` is the one after that.
154 * I.e., this counts forward from the oldest data
155 *
156 * \returns a reference to the indicated entry in the circular buffer.
157 */
158 storedT &at( indexT refEntry, ///< [in] the entry to start counting from
159 indexT idx ///< [in] the index of the entry to access
160 );
161
162 /// Get the entry at a given index relative a fixed start entry, const version
163 /** `idx=0` is the entry at refEntry. `idx=1` is the one after that.
164 * I.e., this counts forward from the oldest data
165 *
166 * \overload
167 *
168 * \returns a const reference to the indicated entry in the circular buffer.
169 */
170 const storedT &at( indexT refEntry, ///< [in] the entry to start counting from
171 indexT idx ///< [in] the index of the entry to access
172 ) const;
173
174 private:
175 derivedT &derived()
176 {
177 return *static_cast<derivedT *>( this );
178 }
179
180 const derivedT &derived() const
181 {
182 return *static_cast<const derivedT *>( this );
183 }
184};
185
186template <class derivedT, typename storedT, typename indexT>
190
191template <class derivedT, typename storedT, typename indexT>
196
197template <class derivedT, typename storedT, typename indexT>
199{
200 m_buffer.clear();
201 m_nextEntry = 0;
202 m_latest = 0;
203
204 m_maxEntries = maxEnt;
205 m_buffer.reserve( m_maxEntries );
206
207 derived().setMaxEntries( maxEnt );
208
209 m_mono = 0;
210}
211
212template <class derivedT, typename storedT, typename indexT>
217
218template <class derivedT, typename storedT, typename indexT>
220{
221 return m_buffer.size();
222}
223
224template <class derivedT, typename storedT, typename indexT>
226{
227 if( m_buffer.size() < m_maxEntries )
228 {
229 m_buffer.push_back( newEnt );
230 m_latest = m_buffer.size() - 1;
231 m_nextEntry = 0;
232 derived().setWrapStartup();
233 }
234 else
235 {
236 m_buffer[m_nextEntry] = newEnt;
237 m_latest = m_nextEntry;
238 ++m_nextEntry;
239 if( m_nextEntry >= m_buffer.size() )
240 m_nextEntry = 0;
241 derived().setWrap();
242 }
243
244 ++m_mono;
245}
246
247template <class derivedT, typename storedT, typename indexT>
252
253template <class derivedT, typename storedT, typename indexT>
258
259template <class derivedT, typename storedT, typename indexT>
261{
262 return m_mono;
263}
264
265template <class derivedT, typename storedT, typename indexT>
267{
268 return derived().at( m_nextEntry, idx );
269}
270
271template <class derivedT, typename storedT, typename indexT>
273{
274 return derived().at( m_nextEntry, idx );
275}
276
277template <class derivedT, typename storedT, typename indexT>
279{
280 return derived().at( refEntry, idx );
281}
282
283template <class derivedT, typename storedT, typename indexT>
285{
286 return derived().at( refEntry, idx );
287}
288
289/// Circular buffer which wraps with an if statement (branching) [faster than mod, less memory than index]
290/**
291 * \ingroup circular_buffer
292 */
293template <typename _storedT, typename _indexT>
294class circularBufferBranch : public circularBufferBase<circularBufferBranch<_storedT, _indexT>, _storedT, _indexT>
295{
296 public:
297 typedef _storedT storedT; ///< The maximum number of entries to allow in the buffer before wrapping
298 typedef _indexT indexT; ///< The index type, also used for sizes
299
300 protected:
301 indexT m_wrapEntry{ 0 };
302
303 public:
304 /// Default c'tor
308
309 /// Sizing constructor
310 /** Sets the maximum size of the buffer. Note that this will not be the size until
311 * a full set of entries have been added to the buffer.
312 */
313 explicit circularBufferBranch( indexT maxEnt /**< [in] the maximum number of entries this buffer will hold*/ )
315 {
316 }
317
318 /// Interface implementation for maxEntries.
319 /** Resets the wrap entry to 0.
320 */
321 void setMaxEntries( indexT maxEnt /**< [in] the maximum number of entries this buffer will hold*/ )
322 {
323 m_wrapEntry = 0;
324 }
325
326 /// Interface implementation for wrapping setup during the startup phase
327 /** This is called before maxEntries have been added.
328 */
330 {
331 m_wrapEntry = this->m_buffer.size();
332 }
333
334 /// Interface implementation for wrapping setup after the startup phase
335 /** This is called after maxEntries have been added.
336 */
337 void setWrap()
338 {
339 m_wrapEntry = this->m_maxEntries - this->m_nextEntry - 1;
340 }
341
342 /// Interface implementation for entry access
343 /** Accesses the idx-th element relative to refEntry, using a branch (if-statement) to wrap
344 *
345 * \returns a reference to the idx-th element
346 */
347 storedT &at( indexT refEntry, ///< [in] the entry to start counting from
348 indexT idx ///< [in] the index of the entry to access
349 )
350 {
351 if( idx > this->m_wrapEntry )
352 return this->m_buffer[refEntry + idx - this->m_buffer.size()];
353 else
354 return this->m_buffer[refEntry + idx];
355 }
356
357 /// Interface implementation for entry access, const version
358 /** Accesses the idx-th element relative to refEntry, using a branch (if-statement) to wrap
359 *
360 * \overload
361 *
362 * \returns a const reference to the idx-th element
363 */
364 const storedT &at( indexT refEntry, ///< [in] the entry to start counting from
365 indexT idx ///< [in] the index of the entry to access
366 ) const
367 {
368 if( idx > this->m_wrapEntry )
369 return this->m_buffer[refEntry + idx - this->m_buffer.size()];
370 else
371 return this->m_buffer[refEntry + idx];
372 }
373};
374
375/// Circular buffer which wraps with the mod opoerator [very slow]
376/**
377 * \ingroup circular_buffer
378 */
379template <typename _storedT, typename _indexT>
380class circularBufferMod : public circularBufferBase<circularBufferMod<_storedT, _indexT>, _storedT, _indexT>
381{
382 public:
383 typedef _storedT storedT; ///< The maximum number of entries to allow in the buffer before wrapping
384 typedef _indexT indexT; ///< The index type, also used for sizes
385
386 /// Default c'tor
390
391 /// Sizing constructor
392 /** Sets the maximum size of the buffer. Note that this will not be the size until
393 * a full set of entries have been added to the buffer.
394 */
395 explicit circularBufferMod( indexT maxEnt /**< [in] the maximum number of entries this buffer will hold*/ )
397 {
398 }
399
400 /// Interface implementation for maxEntries.
401 /** A no-op
402 */
403 void setMaxEntries( indexT maxEnt /**< [in] the maximum number of entries this buffer will hold*/ )
404 {
405 }
406
407 /// Interface implementation for wrapping setup during the startup phase
408 /** This is called before maxEntries have been added.
409 */
411 {
412 }
413
414 /// Interface implementation for wrapping setup after the startup phase
415 /** This is called after maxEntries have been added.
416 */
417 void setWrap()
418 {
419 }
420
421 /// Interface implementation for entry access
422 /** Accesses the idx-th element relative to refEntry, using the mod operator to wrap
423 *
424 * \returns a reference to the idx-th element
425 */
426 storedT &at( indexT refEntry, ///< [in] the entry to start counting from
427 indexT idx ///< [in] the index of the entry to access
428 )
429 {
430 return this->m_buffer[( refEntry + idx ) % this->m_buffer.size()];
431 }
432
433 /// Interface implementation for entry access, const version
434 /** Accesses the idx-th element relative to refEntry, using the mod operator to wrap
435 *
436 * \overload
437 *
438 * \returns a const reference to the idx-th element
439 */
440 const storedT &at( indexT refEntry, ///< [in] the entry to start counting from
441 indexT idx ///< [in] the index of the entry to access
442 ) const
443 {
444 return this->m_buffer[( refEntry + idx ) % this->m_buffer.size()];
445 }
446};
447
448/// Circular buffer which wraps with a pre-populated indices array [generally fastest]
449/**
450 * \ingroup circular_buffer
451 */
452template <typename _storedT, typename _indexT>
453class circularBufferIndex : public circularBufferBase<circularBufferIndex<_storedT, _indexT>, _storedT, _indexT>
454{
455 public:
456 typedef _storedT storedT; ///< The maximum number of entries to allow in the buffer before wrapping
457 typedef _indexT indexT; ///< The index type, also used for sizes
458
459 protected:
460 std::vector<size_t> m_indices; ///< Vector of indices for fast indexing into parent's m_buffer
461
462 public:
463 /// Default c'tor
467
468 /// Sizing constructor
469 /** Sets the maximum size of the buffer. Note that this will not be the size until
470 * a full set of entries have been added to the buffer.
471 */
472 explicit circularBufferIndex( indexT maxEnt /**< [in] the maximum number of entries this buffer will hold*/ )
474 {
475 }
476
477 /// Interface implementation for maxEntries.
478 /** Resizes and populates the indices array.
479 */
480 void setMaxEntries( indexT maxEnt /**< [in] the maximum number of entries this buffer will hold*/ )
481 {
482 m_indices.resize( 2 * this->m_maxEntries );
483 for( size_t i = 0; i < this->m_maxEntries; ++i )
484 {
485 m_indices[i] = i;
486 m_indices[this->m_maxEntries + i] = i;
487 }
488 }
489
490 /// Interface implementation for wrapping setup during the startup phase
491 /** This is called before maxEntries have been added.
492 */
494 {
495 }
496
497 /// Interface implementation for wrapping setup after the startup phase
498 /** This is called after maxEntries have been added.
499 */
500 void setWrap()
501 {
502 }
503
504 /// Interface implementation for entry access
505 /** Accesses the idx-th element relative to refEntry, using the pre-populated indices to wrap
506 *
507 * \returns a reference to the idx-th element
508 */
509 storedT &at( indexT refEntry, ///< [in] the entry to start counting from
510 indexT idx ///< [in] the index of the entry to access
511 )
512 {
513 return this->m_buffer[m_indices[refEntry + idx]];
514 }
515
516 /// Interface implementation for entry access, const version
517 /** Accesses the idx-th element relative to refEntry, using the pre-populated indices to wrap
518 *
519 * \overload
520 *
521 * \returns a const reference to the idx-th element
522 */
523 const storedT &at( indexT refEntry, ///< [in] the entry to start counting from
524 indexT idx ///< [in] the index of the entry to access
525 ) const
526 {
527 return this->m_buffer[m_indices[refEntry + idx]];
528 }
529};
530
531} // namespace sigproc
532} // namespace mx
533
534#endif // sigproc_circularBuffer
CRTP base class for all circular buffers, providing the underlying memory management and accessors.
storedT & at(indexT refEntry, indexT idx)
Get the entry at a given index relative a fixed reference entry.
storedT & operator[](indexT idx)
Get the entry at a given index.
circularBufferBase(indexT maxEnt)
Sizing constructor.
indexT maxEntries()
Get the maximum size of the buffer.
std::vector< storedT > m_buffer
The circular buffer storage.
void maxEntries(indexT maxEnt)
Set the maximum size of the buffer.
indexT size() const
Get the number of entries.
const storedT & at(indexT refEntry, indexT idx) const
Get the entry at a given index relative a fixed start entry, const version.
indexT earliest()
Returns the index of the earliest entry.
indexT latest()
Returns the index of the latest entry.
const storedT & operator[](indexT idx) const
indexT m_maxEntries
The maximum number of entries to allow in the buffer before wrapping.
_derivedT derivedT
The child class.
_storedT storedT
The type stored in the circular buffer.
_indexT indexT
The index type, also used for sizes.
indexT m_latest
Index into m_buff of the latest entry. This is the newest entry in the buffer.
indexT m_nextEntry
Index into m_buffer of the next entry. This is the oldest entry in the buffer.
void nextEntry(const storedT &newEnt)
Add the next entry to the circular buffer.
Circular buffer which wraps with an if statement (branching) [faster than mod, less memory than index...
_storedT storedT
The maximum number of entries to allow in the buffer before wrapping.
void setMaxEntries(indexT maxEnt)
Interface implementation for maxEntries.
const storedT & at(indexT refEntry, indexT idx) const
Interface implementation for entry access, const version.
void setWrapStartup()
Interface implementation for wrapping setup during the startup phase.
storedT & at(indexT refEntry, indexT idx)
Interface implementation for entry access.
void setWrap()
Interface implementation for wrapping setup after the startup phase.
circularBufferBranch(indexT maxEnt)
Sizing constructor.
_indexT indexT
The index type, also used for sizes.
Circular buffer which wraps with a pre-populated indices array [generally fastest].
std::vector< size_t > m_indices
Vector of indices for fast indexing into parent's m_buffer.
const storedT & at(indexT refEntry, indexT idx) const
Interface implementation for entry access, const version.
_indexT indexT
The index type, also used for sizes.
circularBufferIndex(indexT maxEnt)
Sizing constructor.
void setWrap()
Interface implementation for wrapping setup after the startup phase.
_storedT storedT
The maximum number of entries to allow in the buffer before wrapping.
void setWrapStartup()
Interface implementation for wrapping setup during the startup phase.
storedT & at(indexT refEntry, indexT idx)
Interface implementation for entry access.
void setMaxEntries(indexT maxEnt)
Interface implementation for maxEntries.
Circular buffer which wraps with the mod opoerator [very slow].
_storedT storedT
The maximum number of entries to allow in the buffer before wrapping.
void setWrapStartup()
Interface implementation for wrapping setup during the startup phase.
_indexT indexT
The index type, also used for sizes.
const storedT & at(indexT refEntry, indexT idx) const
Interface implementation for entry access, const version.
circularBufferMod(indexT maxEnt)
Sizing constructor.
void setWrap()
Interface implementation for wrapping setup after the startup phase.
storedT & at(indexT refEntry, indexT idx)
Interface implementation for entry access.
void setMaxEntries(indexT maxEnt)
Interface implementation for maxEntries.
The mxlib c++ namespace.
Definition mxError.hpp:106