QP/C++ 6.9.3
qxk_xthr.cpp
Go to the documentation of this file.
1 
39 #define QP_IMPL // this is QP implementation
40 #include "qf_port.hpp" // QF port
41 #include "qxk_pkg.hpp" // QXK package-scope internal interface
42 #include "qassert.h" // QP embedded systems-friendly assertions
43 #ifdef Q_SPY // QS software tracing enabled?
44  #include "qs_port.hpp" // QS port
45  #include "qs_pkg.hpp" // QS facilities for pre-defined trace records
46 #else
47  #include "qs_dummy.hpp" // disable the QS software tracing
48 #endif // Q_SPY
49 
50 // protection against including this source file in a wrong project
51 #ifndef QXK_HPP
52  #error "Source file included in a project NOT based on the QXK kernel"
53 #endif // QXK_HPP
54 
55 Q_DEFINE_THIS_MODULE("qxk_xthr")
56 
57 #define QXTHREAD_CAST_(ptr_) (static_cast<QP::QXThread *>(ptr_))
69 
70 namespace QP {
71 
72 //****************************************************************************
74  std::uint_fast8_t const tickRate) noexcept
75  : QActive(Q_STATE_CAST(handler)),
76  m_timeEvt(this, static_cast<enum_t>(QXK_DELAY_SIG),
77  static_cast<std::uint_fast8_t>(tickRate))
78 {
79  m_state.act = nullptr; // mark as extended thread
80 }
81 
82 //****************************************************************************
83 // QXThread virtual function implementations...
84 void QXThread::init(void const * const e,
85  std::uint_fast8_t const qs_id) noexcept
86 {
87  static_cast<void>(e); // unused parameter
88  static_cast<void>(qs_id); // unused parameter
89  Q_ERROR_ID(110);
90 }
91 
92 //****************************************************************************
93 void QXThread::dispatch(QEvt const * const e,
94  std::uint_fast8_t const qs_id) noexcept
95 {
96  static_cast<void>(e); // unused parameter
97  static_cast<void>(qs_id); // unused parameter
98  Q_ERROR_ID(120);
99 }
100 
101 //****************************************************************************
123  QEvt const * * const qSto, std::uint_fast16_t const qLen,
124  void * const stkSto, std::uint_fast16_t const stkSize,
125  void const * const par)
126 {
127  static_cast<void>(par); // unused parameter
128 
135  && (prio <= QF_MAX_ACTIVE)
136  && (stkSto != nullptr)
137  && (stkSize != 0U)
138  && (m_state.act == nullptr));
139 
140  // is storage for the queue buffer provided?
141  if (qSto != nullptr) {
142  m_eQueue.init(qSto, qLen);
143  }
144 
145  // extended threads provide their thread function in place of
146  // the top-most initial transition 'm_temp.act'
147  QXK_stackInit_(this, m_temp.thr, stkSto, stkSize);
148 
149  m_prio = static_cast<std::uint8_t>(prio);
150  m_dynPrio = static_cast<std::uint8_t>(prio);
151 
152  // the new thread is not blocked on any object
153  m_temp.obj = nullptr;
154 
155  QF::add_(this); // make QF aware of this extended thread
156 
158  QF_CRIT_E_();
159  // extended-thread becomes ready immediately
160  QXK_attr_.readySet.insert(static_cast<std::uint_fast8_t>(m_dynPrio));
161 
162  // see if this thread needs to be scheduled in case QXK is running
163  static_cast<void>(QXK_sched_());
164  QF_CRIT_X_();
165 }
166 
167 #ifdef Q_SPY
168 //****************************************************************************
200 bool QXThread::post_(QEvt const * const e,
201  std::uint_fast16_t const margin,
202  void const * const sender) noexcept
203 #else
204 bool QXThread::post_(QEvt const * const e,
205  std::uint_fast16_t const margin) noexcept
206 #endif
207 {
208  bool status;
211 
212  // is it the private time event?
213  if (e == &m_timeEvt) {
214  QF_CRIT_E_();
215  // the private time event is disarmed and not in any queue,
216  // so it is safe to change its signal. The signal of 0 means
217  // that the time event has expired.
218  m_timeEvt.sig = 0U;
219 
220  unblock_();
221  QF_CRIT_X_();
222 
223  status = true;
224  }
225  // is the event queue provided?
226  else if (m_eQueue.m_end != 0U) {
227 
229  Q_REQUIRE_ID(300, e != nullptr);
230 
231  QF_CRIT_E_();
232  QEQueueCtr nFree = m_eQueue.m_nFree; // get volatile into temporary
233 
234  // test-probe#1 for faking queue overflow
236  nFree = 0U;
237  )
238 
239  if (margin == QF_NO_MARGIN) {
240  if (nFree > 0U) {
241  status = true; // can post
242  }
243  else {
244  status = false; // cannot post
245  Q_ERROR_CRIT_(310); // must be able to post the event
246  }
247  }
248  else if (nFree > static_cast<QEQueueCtr>(margin)) {
249  status = true; // can post
250  }
251  else {
252  status = false; // cannot post, but don't assert
253  }
254 
255  // is it a dynamic event?
256  if (e->poolId_ != 0U) {
257  QF_EVT_REF_CTR_INC_(e); // increment the reference counter
258  }
259 
260  if (status) { // can post the event?
261 
262  --nFree; // one free entry just used up
263  m_eQueue.m_nFree = nFree; // update the volatile
264  if (m_eQueue.m_nMin > nFree) {
265  m_eQueue.m_nMin = nFree; // update minimum so far
266  }
267 
269  QS_TIME_PRE_(); // timestamp
270  QS_OBJ_PRE_(sender); // the sender object
271  QS_SIG_PRE_(e->sig); // the signal of the event
272  QS_OBJ_PRE_(this); // this active object
273  QS_2U8_PRE_(e->poolId_, e->refCtr_); // poolID & refCtr
274  QS_EQC_PRE_(nFree); // number of free entries
275  QS_EQC_PRE_(m_eQueue.m_nMin); // min number of free entries
277 
278  // queue empty?
279  if (m_eQueue.m_frontEvt == nullptr) {
280  m_eQueue.m_frontEvt = e; // deliver event directly
281 
282  // is this thread blocked on the queue?
283  if (m_temp.obj == QXK_PTR_CAST_(QMState*, &m_eQueue)) {
284  static_cast<void>(teDisarm_());
286  static_cast<std::uint_fast8_t>(m_dynPrio));
287  if (!QXK_ISR_CONTEXT_()) {
288  static_cast<void>(QXK_sched_());
289  }
290  }
291  }
292  // queue is not empty, insert event into the ring-buffer
293  else {
294  // insert event into the ring buffer (FIFO)
295  QF_PTR_AT_(m_eQueue.m_ring, m_eQueue.m_head) = e;
296 
297  // need to wrap the head couner?
298  if (m_eQueue.m_head == 0U) {
299  m_eQueue.m_head = m_eQueue.m_end; // wrap around
300  }
301  --m_eQueue.m_head; // advance the head (counter clockwise)
302  }
303 
304  QF_CRIT_X_();
305  }
306  else { // cannot post the event
307 
309  QS_TIME_PRE_(); // timestamp
310  QS_OBJ_PRE_(sender); // the sender object
311  QS_SIG_PRE_(e->sig); // the signal of the event
312  QS_OBJ_PRE_(this); // this active object (recipient)
313  QS_2U8_PRE_(e->poolId_, e->refCtr_); // poolID & ref Count
314  QS_EQC_PRE_(nFree); // number of free entries
315  QS_EQC_PRE_(margin); // margin
317 
318  QF_CRIT_X_();
319 
320  QF::gc(e); // recycle the event to avoid a leak
321  }
322  }
323  else { // the queue is not available
324  QF::gc(e); // make sure the event is not leaked
325  status = false;
326  Q_ERROR_ID(320); // this extended thread cannot accept events
327  }
328 
329  return status;
330 }
331 
332 //****************************************************************************
341 void QXThread::postLIFO(QEvt const * const e) noexcept {
342  static_cast<void>(e); // unused parameter
343  Q_ERROR_ID(410);
344 }
345 
346 //****************************************************************************
366 QEvt const *QXThread::queueGet(std::uint_fast16_t const nTicks) noexcept {
367  QEQueueCtr nFree;
368  QEvt const *e;
370 
371  QF_CRIT_E_();
372  QXThread * const thr = QXTHREAD_CAST_(QXK_attr_.curr);
373 
379  && (thr != nullptr)
380  && (thr->m_temp.obj == nullptr));
382  Q_REQUIRE_ID(501, QXK_attr_.lockHolder != thr->m_prio);
383 
384  // is the queue empty? -- block and wait for event(s)
385  if (thr->m_eQueue.m_frontEvt == nullptr) {
386 
387  // remember the blocking object (the thread's queue)
388  thr->m_temp.obj = QXK_PTR_CAST_(QMState*, &thr->m_eQueue);
389 
390  thr->teArm_(static_cast<enum_t>(QXK_QUEUE_SIG), nTicks);
392  static_cast<std::uint_fast8_t>(thr->m_dynPrio));
393  static_cast<void>(QXK_sched_());
394  QF_CRIT_X_();
395  QF_CRIT_EXIT_NOP(); // BLOCK here
396 
397  QF_CRIT_E_();
398  // the blocking object must be this queue
399  Q_ASSERT_ID(510, thr->m_temp.obj ==
400  QXK_PTR_CAST_(QMState *, &thr->m_eQueue));
401  thr->m_temp.obj = nullptr; // clear
402  }
403 
404  // is the queue not empty?
405  if (thr->m_eQueue.m_frontEvt != nullptr) {
406  e = thr->m_eQueue.m_frontEvt; // always remove from the front
407  // volatile into tmp
408  nFree= thr->m_eQueue.m_nFree + 1U;
409  thr->m_eQueue.m_nFree = nFree; // update the number of free
410 
411  // any events in the ring buffer?
412  if (nFree <= thr->m_eQueue.m_end) {
413 
414  // remove event from the tail
415  thr->m_eQueue.m_frontEvt =
416  QF_PTR_AT_(thr->m_eQueue.m_ring, thr->m_eQueue.m_tail);
417  if (thr->m_eQueue.m_tail == 0U) {
418  thr->m_eQueue.m_tail = thr->m_eQueue.m_end; // wrap
419  }
420  --thr->m_eQueue.m_tail;
421 
423  QS_TIME_PRE_(); // timestamp
424  QS_SIG_PRE_(e->sig); // the signal of this event
425  QS_OBJ_PRE_(&thr); // this active object
426  QS_2U8_PRE_(e->poolId_, e->refCtr_); // poolID & ref Count
427  QS_EQC_PRE_(nFree); // number of free entries
429  }
430  else {
431  thr->m_eQueue.m_frontEvt = nullptr; // the queue becomes empty
432 
433  // all entries in the queue must be free (+1 for fronEvt)
434  Q_ASSERT_ID(520, nFree == (thr->m_eQueue.m_end + 1U));
435 
437  QS_TIME_PRE_(); // timestamp
438  QS_SIG_PRE_(e->sig); // the signal of this event
439  QS_OBJ_PRE_(&thr); // this active object
440  QS_2U8_PRE_(e->poolId_, e->refCtr_); // poolID & ref Count
442  }
443  }
444  else { // the queue is still empty -- the timeout must have fired
445  e = nullptr;
446  }
447  QF_CRIT_X_();
448 
449  return e;
450 }
451 
452 //****************************************************************************
459 void QXThread::block_(void) const noexcept {
461  Q_REQUIRE_ID(600, (QXK_attr_.lockHolder != m_prio));
462  QXK_attr_.readySet.rmove(static_cast<std::uint_fast8_t>(m_dynPrio));
463  static_cast<void>(QXK_sched_());
464 }
465 
466 //****************************************************************************
473 void QXThread::unblock_(void) const noexcept {
474  QXK_attr_.readySet.insert(static_cast<std::uint_fast8_t>(m_dynPrio));
475 
476  if ((!QXK_ISR_CONTEXT_()) // not inside ISR?
477  && (QF::active_[0] != nullptr)) // kernel started?
478  {
479  static_cast<void>(QXK_sched_());
480  }
481 }
482 
483 //****************************************************************************
491 void QXThread::teArm_(enum_t const sig,
492  std::uint_fast16_t const nTicks) noexcept
493 {
495  Q_REQUIRE_ID(700, m_timeEvt.m_ctr == 0U);
496 
497  m_timeEvt.sig = static_cast<QSignal>(sig);
498 
499  if (nTicks != QXTHREAD_NO_TIMEOUT) {
500  m_timeEvt.m_ctr = static_cast<QTimeEvtCtr>(nTicks);
501  m_timeEvt.m_interval = 0U;
502 
503  // is the time event unlinked?
504  // NOTE: For the duration of a single clock tick of the specified tick
505  // rate a time event can be disarmed and yet still linked in the list,
506  // because un-linking is performed exclusively in QF::tickX().
507  if ((m_timeEvt.refCtr_ & TE_IS_LINKED) == 0U) {
508  std::uint_fast8_t const tickRate =
509  static_cast<std::uint_fast8_t>(m_timeEvt.refCtr_);
510  m_timeEvt.refCtr_ |= TE_IS_LINKED; // mark as linked
511 
512  // The time event is initially inserted into the separate
513  // "freshly armed" list based on QF::timeEvtHead_[tickRate].act.
514  // Only later, inside QF::tickX() function, the "freshly armed"
515  // list is appended to the main list of armed time events based on
516  // QF_timeEvtHead_[tickRate].next. Again, this is to keep any
517  // changes to the main list exclusively inside QF::tickX().
518  m_timeEvt.m_next
519  = QXK_PTR_CAST_(QTimeEvt*, QF::timeEvtHead_[tickRate].m_act);
520  QF::timeEvtHead_[tickRate].m_act = &m_timeEvt;
521  }
522  }
523 }
524 
525 //****************************************************************************
532 bool QXThread::teDisarm_(void) noexcept {
533  bool wasArmed;
534  // is the time evt running?
535  if (m_timeEvt.m_ctr != 0U) {
536  wasArmed = true;
537  // schedule removal from list
538  m_timeEvt.m_ctr = 0U;
539  }
540  // the time event was already automatically disarmed
541  else {
542  wasArmed = false;
543  }
544  return wasArmed;
545 }
546 
547 //****************************************************************************
549 bool QXThread::delay(std::uint_fast16_t const nTicks) noexcept {
551 
552  QF_CRIT_E_();
553  QXThread * const thr = QXK_PTR_CAST_(QXThread*, QXK_attr_.curr);
554 
560  && (thr != nullptr)
561  && (thr->m_temp.obj == nullptr));
563  Q_REQUIRE_ID(801, QXK_attr_.lockHolder != thr->m_prio);
564 
565  // remember the blocking object
566  thr->m_temp.obj = QXK_PTR_CAST_(QMState*, &thr->m_timeEvt);
567  thr->teArm_(static_cast<enum_t>(QXK_DELAY_SIG), nTicks);
568  thr->block_();
569  QF_CRIT_X_();
570  QF_CRIT_EXIT_NOP(); // BLOCK here
571 
572  QF_CRIT_E_();
573  // the blocking object must be the time event
574  Q_ENSURE_ID(890, thr->m_temp.obj
575  == QXK_PTR_CAST_(QMState*, &thr->m_timeEvt));
576  thr->m_temp.obj = nullptr; // clear
577  QF_CRIT_X_();
578 
579  // signal of zero means that the time event was posted without
580  // being canceled.
581  return (thr->m_timeEvt.sig == 0U);
582 }
583 
584 //****************************************************************************
586 bool QXThread::delayCancel(void) noexcept {
587  bool wasArmed;
589 
590  QF_CRIT_E_();
591  if (m_temp.obj == QXK_PTR_CAST_(QMState*, &m_timeEvt)) {
592  wasArmed = teDisarm_();
593  unblock_();
594  }
595  else {
596  wasArmed = false;
597  }
598  QF_CRIT_X_();
599 
600  return wasArmed;
601 }
602 
603 } // namespace QP
604 
605 
606 //****************************************************************************
607 extern "C" {
608 
618 void QXK_threadRet_(void) noexcept {
620 
621  QF_CRIT_E_();
622  QP::QXThread * const thr = QXTHREAD_CAST_(QXK_attr_.curr);
623 
628  && (thr != nullptr));
630  Q_REQUIRE_ID(901, QXK_attr_.lockHolder != thr->m_prio);
631 
632  std::uint_fast8_t const p =
633  static_cast<std::uint_fast8_t>(thr->m_prio);
634 
635  // remove this thread from the QF
636  QP::QF::active_[p] = nullptr;
638  static_cast<void>(QXK_sched_());
639  QF_CRIT_X_();
640 }
641 
642 } // extern "C"
643 
unsigned int uint_fast16_t
fast at-least 16-bit unsigned int
Definition: 16bit/stdint.h:38
unsigned char uint8_t
exact-width 8-bit unsigned int
Definition: 16bit/stdint.h:29
unsigned int uint_fast8_t
fast at-least 8-bit unsigned int
Definition: 16bit/stdint.h:36
QActive active object (based on QP::QHsm implementation)
Definition: qf.hpp:144
std::uint8_t m_prio
QF priority (1..QF_MAX_ACTIVE) of this active object.
Definition: qf.hpp:185
static QTimeEvt timeEvtHead_[QF_MAX_TICK_RATE]
heads of linked lists of time events, one for every clock tick rate
Definition: qf.hpp:615
static void gc(QEvt const *const e) noexcept
Recycle a dynamic event.
Definition: qf_dyn.cpp:219
static void add_(QActive *const a) noexcept
Register an active object to be managed by the framework.
Definition: qf_act.cpp:79
static QActive * active_[QF_MAX_ACTIVE+1U]
array of registered active objects
Definition: qf.hpp:580
QHsmAttr m_state
current active state (state-variable)
Definition: qep.hpp:270
QHsmAttr m_temp
temporary: transition chain, target state, etc.
Definition: qep.hpp:271
friend class QXThread
Definition: qep.hpp:475
Time Event class.
Definition: qf.hpp:404
void *volatile m_act
the active object that receives the time events
Definition: qf.hpp:414
Extended (blocking) thread of the QXK preemptive kernel.
Definition: qxthread.hpp:66
void start(std::uint_fast8_t const prio, QEvt const **const qSto, std::uint_fast16_t const qLen, void *const stkSto, std::uint_fast16_t const stkSize, void const *const par) override
Starts execution of an extended thread and registers the thread with the framework.
Definition: qxk_xthr.cpp:122
void block_(void) const noexcept
Definition: qxk_xthr.cpp:459
void init(void const *const e, std::uint_fast8_t const qs_id) noexcept override
Executes the top-most initial transition in HSM.
Definition: qxk_xthr.cpp:84
void postLIFO(QEvt const *const e) noexcept override
Posts an event directly to the event queue of the active object using the Last-In-First-Out (LIFO) po...
Definition: qxk_xthr.cpp:341
static bool delay(std::uint_fast16_t const nTicks) noexcept
delay (block) the current extended thread for a specified # ticks
Definition: qxk_xthr.cpp:549
bool delayCancel(void) noexcept
cancel the delay
Definition: qxk_xthr.cpp:586
bool post_(QEvt const *const e, std::uint_fast16_t const margin, void const *const sender) noexcept override
Definition: qxk_xthr.cpp:200
void dispatch(QEvt const *const e, std::uint_fast8_t const qs_id) noexcept override
Dispatches an event to HSM.
Definition: qxk_xthr.cpp:93
QTimeEvt m_timeEvt
time event to handle blocking timeouts
Definition: qxthread.hpp:131
void unblock_(void) const noexcept
Definition: qxk_xthr.cpp:473
static QEvt const * queueGet(std::uint_fast16_t const nTicks=QXTHREAD_NO_TIMEOUT) noexcept
obtain a message from the private message queue (block if no messages)
Definition: qxk_xthr.cpp:366
void teArm_(enum_t const sig, std::uint_fast16_t const nTicks) noexcept
Definition: qxk_xthr.cpp:491
bool teDisarm_(void) noexcept
Definition: qxk_xthr.cpp:532
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_ACTIVE_GET
AO got an event and its queue is not empty.
Definition: qs.hpp:84
@ QS_QF_ACTIVE_POST_ATTEMPT
attempt to post an evt to AO failed
Definition: qs.hpp:127
@ QS_QF_ACTIVE_POST
an event was posted (FIFO) directly to AO
Definition: qs.hpp:82
@ QS_QF_ACTIVE_GET_LAST
AO got an event and its queue is empty.
Definition: qs.hpp:85
void(*)(QXThread *const me) QXThreadHandler
Pointer to a thread-handler function.
Definition: qep.hpp:232
void QF_EVT_REF_CTR_INC_(QEvt const *const e) noexcept
increment the refCtr_ of an event e
Definition: qf_pkg.hpp:150
QMState const * obj
pointer to QMState object
Definition: qep.hpp:242
static constexpr std::uint_fast16_t QXTHREAD_NO_TIMEOUT
no-timeout sepcification when blocking on queues or semaphores
Definition: qxthread.hpp:45
@ QXK_QUEUE_SIG
Definition: qxk_pkg.hpp:47
@ QXK_DELAY_SIG
Definition: qxk_pkg.hpp:46
constexpr std::uint8_t TE_IS_LINKED
Definition: qf_pkg.hpp:132
std::uint8_t QTimeEvtCtr
Definition: qf.hpp:99
QXThreadHandler thr
pointer to an thread-handler function
Definition: qep.hpp:241
std::uint16_t QSignal
QSignal represents the signal of an event.
Definition: qep.hpp:146
QActionHandler act
pointer to an action-handler function
Definition: qep.hpp:240
State object for the QP::QMsm class (QM State Machine).
Definition: qep.hpp:584
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_ASSERT_ID(id_, test_)
General purpose assertion with user-specified assertion-id.
Definition: qassert.h:155
#define Q_ENSURE_ID(id_, test_)
Assertion for checking postconditions with user-specified assertion-id.
Definition: qassert.h:298
#define Q_REQUIRE_ID(id_, test_)
Assertion for checking preconditions with user-specified assertion-id.
Definition: qassert.h:279
#define Q_ERROR_ID(id_)
Assertion with user-specified assertion-id for a wrong path.
Definition: qassert.h:211
int enum_t
alias for enumerations used for event signals
Definition: qep.hpp:82
#define Q_STATE_CAST(handler_)
Macro to perform casting to QStateHandler.
Definition: qep.hpp:642
#define QF_CRIT_EXIT_NOP()
No-operation for exiting a critical section.
Definition: qf.hpp:674
#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 QF_CRIT_X_()
This is an internal macro for exiting a critical section.
Definition: qf_pkg.hpp:77
#define QF_CRIT_E_()
This is an internal macro for entering a critical section.
Definition: qf_pkg.hpp:66
#define Q_ERROR_CRIT_(id_)
Definition: qf_pkg.hpp:103
#define QS_TIME_PRE_()
Definition: qs.hpp:266
#define QS_TEST_PROBE_DEF(fun_)
QS macro to define the Test-Probe for a given fun_.
Definition: qs.hpp:1068
#define QS_TEST_PROBE_ID(id_, code_)
QS macro to apply a Test-Probe.
Definition: qs.hpp:1077
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.
#define QF_MAX_ACTIVE
The maximum number of active objects in the application.
Definition: qxk/qf_port.hpp:57
std::uint8_t volatile lockHolder
prio of the lock holder
Definition: qxk.hpp:79
#define QXK_ISR_CONTEXT_()
Internal port-specific macro that reports the execution context.
Definition: qxk.hpp:178
std::uint_fast8_t QXK_sched_(void) noexcept
QXK scheduler finds the highest-priority thread ready to run.
Definition: qxk.cpp:346
QP::QPSet readySet
ready-set of all threads
Definition: qxk.hpp:82
QP::QActive *volatile curr
currently executing thread
Definition: qxk.hpp:75
QXK_Attr QXK_attr_
global attributes of the QXK kernel
Definition: qxk.cpp:60
Internal (package scope) QXK/C++ interface.
void QXK_stackInit_(void *thr, QP::QXThreadHandler const handler, void *const stkSto, std::uint_fast16_t const stkSize) noexcept
initialize the private stack of a given AO
#define QXK_PTR_CAST_(type_, ptr_)
intertnal macro to encapsulate casting of pointers for MISRA deviations
Definition: qxk_pkg.hpp:76
void QXK_threadRet_(void) noexcept
called when a thread function returns
Definition: qxk_xthr.cpp:618
#define QXTHREAD_CAST_(ptr_)
intertnal macro to encapsulate casting of pointers for MISRA deviations *‍/
Definition: qxk_xthr.cpp:68
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
void insert(std::uint_fast8_t const n) noexcept
insert element n into the set, n = 1..QF_MAX_ACTIVE
Definition: qpset.hpp:101
void rmove(std::uint_fast8_t const n) noexcept
remove element n from the set, n = 1..QF_MAX_ACTIVE
Definition: qpset.hpp:109