QP/C++ 6.9.3
qf_qeq.cpp
Go to the documentation of this file.
1 
38 #define QP_IMPL // this is QP implementation
39 #include "qf_port.hpp" // QF port
40 #include "qf_pkg.hpp" // QF package-scope interface
41 #include "qassert.h" // QP embedded systems-friendly assertions
42 #ifdef Q_SPY // QS software tracing enabled?
43  #include "qs_port.hpp" // QS port
44  #include "qs_pkg.hpp" // QS facilities for pre-defined trace records
45 #else
46  #include "qs_dummy.hpp" // disable the QS software tracing
47 #endif // Q_SPY
48 
49 namespace QP {
50 
51 Q_DEFINE_THIS_MODULE("qf_qeq")
52 
53 
54 //****************************************************************************
58 QEQueue::QEQueue(void) noexcept
59  : m_frontEvt(nullptr),
60  m_ring(nullptr),
61  m_end(0U),
62  m_head(0U),
63  m_tail(0U),
64  m_nFree(0U),
65  m_nMin(0U)
66 {}
67 
68 //****************************************************************************
84 void QEQueue::init(QEvt const *qSto[],
85  std::uint_fast16_t const qLen) noexcept
86 {
87  m_frontEvt = nullptr; // no events in the queue
88  m_ring = &qSto[0];
89  m_end = static_cast<QEQueueCtr>(qLen);
90  if (qLen > 0U) {
91  m_head = 0U;
92  m_tail = 0U;
93  }
94  m_nFree = static_cast<QEQueueCtr>(qLen + 1U); //+1 for frontEvt
95  m_nMin = m_nFree;
96 }
97 
98 //****************************************************************************
122 bool QEQueue::post(QEvt const * const e,
123  std::uint_fast16_t const margin,
124  std::uint_fast8_t const qs_id) noexcept
125 {
126  bool status;
128 
130  Q_REQUIRE_ID(200, e != nullptr);
131 
132  QF_CRIT_E_();
133  QEQueueCtr nFree = m_nFree; // temporary to avoid UB for volatile access
134 
135  // margin available?
136  if (((margin == QF_NO_MARGIN) && (nFree > 0U))
137  || (nFree > static_cast<QEQueueCtr>(margin)))
138  {
139  // is it a dynamic event?
140  if (e->poolId_ != 0U) {
141  QF_EVT_REF_CTR_INC_(e); // increment the reference counter
142  }
143 
144  --nFree; // one free entry just used up
145  m_nFree = nFree; // update the volatile
146  if (m_nMin > nFree) {
147  m_nMin = nFree; // update minimum so far
148  }
149 
151  QS_TIME_PRE_(); // timestamp
152  QS_SIG_PRE_(e->sig); // the signal of this event
153  QS_OBJ_PRE_(this); // this queue object
154  QS_2U8_PRE_(e->poolId_, e->refCtr_);// pool Id & refCtr of the evt
155  QS_EQC_PRE_(nFree); // number of free entries
156  QS_EQC_PRE_(m_nMin); // min number of free entries
158 
159  // is the queue empty?
160  if (m_frontEvt == nullptr) {
161  m_frontEvt = e; // deliver event directly
162  }
163  // queue is not empty, leave event in the ring-buffer
164  else {
165  // insert event into the ring buffer (FIFO)
166  QF_PTR_AT_(m_ring, m_head) = e; // insert e into buffer
167 
168  // need to wrap?
169  if (m_head == 0U) {
170  m_head = m_end; // wrap around
171  }
172  --m_head;
173  }
174  status = true; // event posted successfully
175  }
176  else {
179  Q_ASSERT_CRIT_(210, margin != QF_NO_MARGIN);
180 
182  QS_TIME_PRE_(); // timestamp
183  QS_SIG_PRE_(e->sig); // the signal of this event
184  QS_OBJ_PRE_(this); // this queue object
185  QS_2U8_PRE_(e->poolId_, e->refCtr_);// pool Id & refCtr of the evt
186  QS_EQC_PRE_(nFree); // number of free entries
187  QS_EQC_PRE_(margin); // margin requested
189 
190  status = false; // event not posted
191  }
192  QF_CRIT_X_();
193 
194  static_cast<void>(qs_id); // unused parameter, if Q_SPY not defined
195 
196  return status;
197 }
198 
199 //****************************************************************************
221 void QEQueue::postLIFO(QEvt const * const e,
222  std::uint_fast8_t const qs_id) noexcept
223 {
225 
226  QF_CRIT_E_();
227  QEQueueCtr nFree = m_nFree; // temporary to avoid UB for volatile access
228 
230  Q_REQUIRE_CRIT_(300, nFree != 0U);
231 
232  // is it a dynamic event?
233  if (e->poolId_ != 0U) {
234  QF_EVT_REF_CTR_INC_(e); // increment the reference counter
235  }
236 
237  --nFree; // one free entry just used up
238  m_nFree = nFree; // update the volatile
239  if (m_nMin > nFree) {
240  m_nMin = nFree; // update minimum so far
241  }
242 
244  QS_TIME_PRE_(); // timestamp
245  QS_SIG_PRE_(e->sig); // the signal of this event
246  QS_OBJ_PRE_(this); // this queue object
247  QS_2U8_PRE_(e->poolId_, e->refCtr_); // pool Id & refCtr of the evt
248  QS_EQC_PRE_(nFree); // number of free entries
249  QS_EQC_PRE_(m_nMin); // min number of free entries
251 
252  QEvt const * const frontEvt = m_frontEvt; // read volatile into temporary
253  m_frontEvt = e; // deliver event directly to the front of the queue
254 
255  // was the queue not empty?
256  if (frontEvt != nullptr) {
257  ++m_tail;
258  if (m_tail == m_end) { // need to wrap the tail?
259  m_tail = 0U; // wrap around
260  }
261  QF_PTR_AT_(m_ring, m_tail) = frontEvt; // buffer the old front evt
262  }
263 
264  static_cast<void>(qs_id); // unused parameter, if Q_SPY not defined
265 
266  QF_CRIT_X_();
267 }
268 
269 //****************************************************************************
287 QEvt const *QEQueue::get(std::uint_fast8_t const qs_id) noexcept {
288  QEvt const *e;
290 
291  QF_CRIT_E_();
292  e = m_frontEvt; // always remove the event from the front location
293 
294  // is the queue not empty?
295  if (e != nullptr) {
296  QEQueueCtr const nFree = m_nFree + 1U;
297  m_nFree = nFree; // upate the number of free
298 
299  // any events in the the ring buffer?
300  if (nFree <= m_end) {
301  m_frontEvt = QF_PTR_AT_(m_ring, m_tail); // remove from the tail
302  if (m_tail == 0U) { // need to wrap?
303  m_tail = m_end; // wrap around
304  }
305  --m_tail;
306 
308  QS_TIME_PRE_(); // timestamp
309  QS_SIG_PRE_(e->sig); // the signal of this event
310  QS_OBJ_PRE_(this); // this queue object
311  QS_2U8_PRE_(e->poolId_, e->refCtr_);
312  QS_EQC_PRE_(nFree); // # free entries
314  }
315  else {
316  m_frontEvt = nullptr; // queue becomes empty
317 
318  // all entries in the queue must be free (+1 for fronEvt)
319  Q_ASSERT_CRIT_(410, nFree == (m_end + 1U));
320 
322  QS_TIME_PRE_(); // timestamp
323  QS_SIG_PRE_(e->sig); // the signal of this event
324  QS_OBJ_PRE_(this); // this queue object
325  QS_2U8_PRE_(e->poolId_, e->refCtr_);
327  }
328  }
329  QF_CRIT_X_();
330 
331  static_cast<void>(qs_id); // unused parameter, if Q_SPY not defined
332 
333  return e;
334 }
335 
336 } // namespace QP
337 
unsigned int uint_fast16_t
fast at-least 16-bit unsigned int
Definition: 16bit/stdint.h:38
unsigned int uint_fast8_t
fast at-least 8-bit unsigned int
Definition: 16bit/stdint.h:36
Native QF Event Queue class.
Definition: qequeue.hpp:123
QEvt const * get(std::uint_fast8_t const qs_id) noexcept
"raw" thread-safe QF event queue implementation for the Last-In-First-Out (LIFO) event posting.
Definition: qf_qeq.cpp:287
void init(QEvt const *qSto[], std::uint_fast16_t const qLen) noexcept
Initializes the native QF event queue.
Definition: qf_qeq.cpp:84
void postLIFO(QEvt const *const e, std::uint_fast8_t const qs_id) noexcept
"raw" thread-safe QF event queue implementation for the First-In-First-Out (FIFO) event posting....
Definition: qf_qeq.cpp:221
bool post(QEvt const *const e, std::uint_fast16_t const margin, std::uint_fast8_t const qs_id) noexcept
"raw" thread-safe QF event queue implementation for the event posting (FIFO). You can call this funct...
Definition: qf_qeq.cpp:122
namespace associated with the QP/C++ framework
Definition: struct.dox:1
std::uint8_t QEQueueCtr
The data type to store the ring-buffer counters based on the macro QF_EQUEUE_CTR_SIZE.
Definition: qequeue.hpp:72
std::uint_fast16_t const QF_NO_MARGIN
special value of margin that causes asserting failure in case event allocation or event posting fails
Definition: qf.hpp:627
@ QS_QF_EQUEUE_GET_LAST
get the last event from the queue
Definition: qs.hpp:92
@ QS_QF_EQUEUE_POST_ATTEMPT
attempt to post an evt to QEQueue failed
Definition: qs.hpp:130
@ QS_QF_EQUEUE_POST_LIFO
an event was posted (LIFO) to a raw queue
Definition: qs.hpp:90
@ QS_QF_EQUEUE_GET
get an event and queue still not empty
Definition: qs.hpp:91
@ QS_QF_EQUEUE_POST
an event was posted (FIFO) to a raw queue
Definition: qs.hpp:89
void QF_EVT_REF_CTR_INC_(QEvt const *const e) noexcept
increment the refCtr_ of an event e
Definition: qf_pkg.hpp:150
Customizable and memory-efficient assertions for embedded systems.
#define Q_DEFINE_THIS_MODULE(name_)
Define the user-specified module name for assertions in this file.
Definition: qassert.h:120
#define Q_REQUIRE_ID(id_, test_)
Assertion for checking preconditions with user-specified assertion-id.
Definition: qassert.h:279
Internal (package scope) QF/C++ interface.
#define QF_CRIT_STAT_
This is an internal macro for defining the critical section status type.
Definition: qf_pkg.hpp:56
#define QF_PTR_AT_(base_, i_)
access element at index i_ from the base pointer base_
Definition: qf_pkg.hpp:171
#define Q_ASSERT_CRIT_(id_, test_)
Definition: qf_pkg.hpp:94
#define QF_CRIT_X_()
This is an internal macro for exiting a critical section.
Definition: qf_pkg.hpp:77
#define Q_REQUIRE_CRIT_(id_, test_)
Definition: qf_pkg.hpp:101
#define QF_CRIT_E_()
This is an internal macro for entering a critical section.
Definition: qf_pkg.hpp:66
#define QS_TIME_PRE_()
Definition: qs.hpp:266
Dummy definitions of the QS macros that avoid code generation from the QS instrumentation.
#define QS_BEGIN_NOCRIT_PRE_(rec_, qs_id_)
Definition: qs_dummy.hpp:98
#define QS_OBJ_PRE_(obj_)
Definition: qs_dummy.hpp:107
#define QS_END_NOCRIT_PRE_()
Definition: qs_dummy.hpp:99
#define QS_2U8_PRE_(data1_, data2_)
Definition: qs_dummy.hpp:101
#define QS_SIG_PRE_(sig_)
Definition: qs_dummy.hpp:105
#define QS_EQC_PRE_(ctr_)
Definition: qs_dummy.hpp:109
Internal (package scope) QS/C++ interface.
QS/C++ port to a 32-bit CPU, generic compiler.
QEvt base class.
Definition: qep.hpp:209
QSignal sig
signal of the event instance
Definition: qep.hpp:210
std::uint8_t volatile refCtr_
reference counter
Definition: qep.hpp:212
std::uint8_t poolId_
pool ID (0 for static event)
Definition: qep.hpp:211