QP/C++ 6.9.3
qxk_mutex.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 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 namespace QP {
56 
57 Q_DEFINE_THIS_MODULE("qxk_mutex")
58 
59 //****************************************************************************
84 void QXMutex::init(std::uint_fast8_t const ceiling) noexcept {
86 
87  QF_CRIT_E_();
92  Q_REQUIRE_ID(100,
93  (ceiling <= QF_MAX_ACTIVE)
94  && ((ceiling == 0U)
95  || (QF::active_[ceiling] == nullptr)));
96 
97  m_ceiling = static_cast<std::uint8_t>(ceiling);
98  m_lockNest = 0U;
99  m_holderPrio = 0U;
100  QF::bzero(&m_waitSet, sizeof(m_waitSet));
101 
102  if (ceiling != 0U) {
103  // reserve the ceiling priority level for this mutex
104  QF::active_[ceiling] = QXK_PTR_CAST_(QActive*, this);
105  }
106  QF_CRIT_X_();
107 }
108 
109 //****************************************************************************
129 bool QXMutex::lock(std::uint_fast16_t const nTicks) noexcept {
130  bool locked = true; // assume that the mutex will be locked
131  QXThread *curr;
133 
134  QF_CRIT_E_();
136 
146  && (curr != nullptr)
147  && ((m_ceiling == 0U)
148  || (curr->m_prio < m_ceiling))
149  && (m_ceiling <= QF_MAX_ACTIVE)
150  && (curr->m_temp.obj == nullptr)); // not blocked
152  Q_REQUIRE_ID(201, QXK_attr_.lockHolder != curr->m_prio);
153 
154  // is the mutex available?
155  if (m_lockNest == 0U) {
156  m_lockNest = 1U;
157 
158  if (m_ceiling != 0U) {
159  // the priority slot must be occupied by this mutex
160  Q_ASSERT_ID(210, QF::active_[m_ceiling]
161  == QXK_PTR_CAST_(QActive*, this));
162 
163  // boost the dynamic priority of this thread to the ceiling
165  static_cast<std::uint_fast8_t>(curr->m_dynPrio));
166  curr->m_dynPrio = m_ceiling;
168  static_cast<std::uint_fast8_t>(curr->m_dynPrio));
169  QF::active_[m_ceiling] = curr;
170  }
171 
172  // make the curr thread the new mutex holder
173  m_holderPrio = static_cast<std::uint8_t>(curr->m_prio);
174 
176  QS_TIME_PRE_(); // timestamp
177  // start prio & current ceiling
178  QS_2U8_PRE_(curr->m_prio, m_ceiling);
180  }
181  // is the mutex locked by this thread already (nested locking)?
182  else if (m_holderPrio == curr->m_prio) {
183 
184  // the nesting level must not exceed the dynamic range of uint8_t
185  Q_ASSERT_ID(220, m_lockNest < 0xFFU);
186 
187  ++m_lockNest;
188  }
189  else { // the mutex is alredy locked by a different thread
190 
191  // the ceiling holder priority must be valid
192  Q_ASSERT_ID(230, 0U < m_holderPrio);
193  Q_ASSERT_ID(231, m_holderPrio <= QF_MAX_ACTIVE);
194 
195  if (m_ceiling != 0U) {
196  // the prio slot must be occupied by the thr. holding the mutex
197  Q_ASSERT_ID(240, QF::active_[m_ceiling]
198  == QF::active_[m_holderPrio]);
199  }
200 
201  // remove the curr dynamic prio from the ready set (block)
202  // and insert it to the waiting set on this mutex
203  std::uint_fast8_t const p =
204  static_cast<std::uint_fast8_t>(curr->m_dynPrio);
206  m_waitSet.insert(p);
207 
208  // store the blocking object (this mutex)
209  curr->m_temp.obj = QXK_PTR_CAST_(QMState*, this);
210  curr->teArm_(static_cast<enum_t>(QXK_SEMA_SIG), nTicks);
211 
212  // schedule the next thread if multitasking started
213  static_cast<void>(QXK_sched_());
214  QF_CRIT_X_();
215  QF_CRIT_EXIT_NOP(); // BLOCK here !!!
216 
217  QF_CRIT_E_(); // AFTER unblocking...
218  // the blocking object must be this mutex
219  Q_ASSERT_ID(240, curr->m_temp.obj == QXK_PTR_CAST_(QMState *, this));
220 
221  // did the blocking time-out? (signal of zero means that it did)
222  if (curr->m_timeEvt.sig == 0U) {
223  if (m_waitSet.hasElement(p)) { // still waiting?
224  m_waitSet.rmove(p); // remove the unblocked thread
225  locked = false; // the mutex was NOT locked
226  }
227  }
228  else { // blocking did NOT time out
229  // the thread must NOT be waiting on this mutex
230  Q_ASSERT_ID(250, !m_waitSet.hasElement(p));
231  }
232  curr->m_temp.obj = nullptr; // clear blocking obj.
233  }
234  QF_CRIT_X_();
235 
236  return locked;
237 }
238 
239 //****************************************************************************
257 bool QXMutex::tryLock(void) noexcept {
258  QActive *curr;
260 
261  QF_CRIT_E_();
262  curr = QXK_attr_.curr;
263  if (curr == nullptr) { // called from a basic thread?
264  curr = QF::active_[QXK_attr_.actPrio];
265  }
266 
274  && (curr != nullptr)
275  && ((m_ceiling == 0U)
276  || (curr->m_prio < m_ceiling))
277  && (m_ceiling <= QF_MAX_ACTIVE));
279  Q_REQUIRE_ID(301, QXK_attr_.lockHolder != curr->m_prio);
280 
281  // is the mutex available?
282  if (m_lockNest == 0U) {
283  m_lockNest = 1U;
284 
285  if (m_ceiling != 0U) {
286  // the priority slot must be set to this mutex
287  Q_ASSERT_ID(310, QF::active_[m_ceiling]
288  == QXK_PTR_CAST_(QActive *, this));
289 
290  // boost the dynamic priority of this thread to the ceiling
292  static_cast<std::uint_fast8_t>(curr->m_dynPrio));
293  curr->m_dynPrio = m_ceiling;
295  static_cast<std::uint_fast8_t>(curr->m_dynPrio));
296  QF::active_[m_ceiling] = curr;
297  }
298 
299  // make curr thread the new mutex holder
300  m_holderPrio = curr->m_prio;
301 
303  QS_TIME_PRE_(); // timestamp
304  // start prio & current ceiling
305  QS_2U8_PRE_(curr->m_prio, m_ceiling);
307  }
308  // is the mutex held by this thread already (nested locking)?
309  else if (m_holderPrio == curr->m_prio) {
310  // the nesting level must not exceed the dynamic range of uint8_t
311  Q_ASSERT_ID(320, m_lockNest < 0xFFU);
312 
313  ++m_lockNest;
314  }
315  else { // the mutex is alredy locked by a different thread
316  if (m_ceiling != 0U) {
317  // the prio slot must be claimed by the mutex holder
318  Q_ASSERT_ID(330, QF::active_[m_ceiling]
319  == QF::active_[m_holderPrio]);
320  }
321  curr = nullptr; // means that mutex is NOT available
322  }
323  QF_CRIT_X_();
324 
325  return curr != nullptr;
326 }
327 
328 //****************************************************************************
345 void QXMutex::unlock(void) noexcept {
346  QActive *curr;
348 
349  QF_CRIT_E_();
350  curr = static_cast<QActive *>(QXK_attr_.curr);
351  if (curr == nullptr) { // called from a basic thread?
352  curr = QF::active_[QXK_attr_.actPrio];
353  }
354 
363  && (curr != nullptr)
364  && ((m_ceiling == 0U)
365  || (curr->m_dynPrio == m_ceiling))
366  && (m_ceiling <= QF_MAX_ACTIVE));
368  Q_REQUIRE_ID(401, m_lockNest > 0U);
370  Q_REQUIRE_ID(402, m_holderPrio == curr->m_prio);
371 
372  // is this the last nesting level?
373  if (m_lockNest == 1U) {
374 
375  if (m_ceiling != 0U) {
376  // restore the holding thread's priority to the original
378  static_cast<std::uint_fast8_t>(curr->m_dynPrio));
379  curr->m_dynPrio = curr->m_prio;
381  static_cast<std::uint_fast8_t>(curr->m_dynPrio));
382 
383  // put the mutex at the priority ceiling slot
384  QF::active_[m_ceiling] = QXK_PTR_CAST_(QActive*, this);
385  }
386 
387  // the mutex no longer held by a thread
388  m_holderPrio = 0U;
389 
391  QS_TIME_PRE_(); // timestamp
392  // start prio & the mutex ceiling
393  QS_2U8_PRE_(curr->m_prio, m_ceiling);
395 
396  // are any other threads waiting for this mutex?
397  if (m_waitSet.notEmpty()) {
398 
399  // find the highest-priority thread waiting on this mutex
400  std::uint_fast8_t const p = m_waitSet.findMax();
401  QXThread * const thr = QXK_PTR_CAST_(QXThread*, QF::active_[p]);
402 
403  // the waiting thread must:
404  // - the ceiling must not be used; or if used
405  // - the thread must have priority below the ceiling
406  // - be registered in QF
407  // - have the dynamic priority the same as initial priority
408  // - be blocked on this mutex
409  Q_ASSERT_ID(410,
410  ((m_ceiling == 0U)
411  || (p < static_cast<std::uint_fast8_t>(m_ceiling)))
412  && (thr != nullptr)
413  && (thr->m_dynPrio == thr->m_prio)
414  && (thr->m_temp.obj == QXK_PTR_CAST_(QMState*, this)));
415 
416  // disarm the internal time event
417  static_cast<void>(thr->teDisarm_());
418 
419  if (m_ceiling != 0U) {
420  // boost the dynamic priority of this thread to the ceiling
421  thr->m_dynPrio = m_ceiling;
422  QF::active_[m_ceiling] = thr;
423  }
424 
425  // make the thread the new mutex holder
426  m_holderPrio = static_cast<std::uint8_t>(thr->m_prio);
427 
428  // make the thread ready to run (at the ceiling prio)
429  // and remove from the waiting list
430  QXK_attr_.readySet.insert(thr->m_dynPrio);
431  m_waitSet.rmove(p);
432 
434  QS_TIME_PRE_(); // timestamp
435  // start priority & ceiling priority
436  QS_2U8_PRE_(thr->m_prio, m_ceiling);
438  }
439  else { // no threads are waiting for this mutex
440  m_lockNest = 0U;
441 
442  if (m_ceiling != 0U) {
443  // put the mutex at the priority ceiling slot
444  QF::active_[m_ceiling] = QXK_PTR_CAST_(QActive*, this);
445  }
446  }
447 
448  // schedule the next thread if multitasking started
449  if (QXK_sched_() != 0U) {
450  QXK_activate_(); // activate a basic thread
451  }
452  }
453  else { // releasing the mutex
454  --m_lockNest; // release one level
455  }
456  QF_CRIT_X_();
457 }
458 
459 } // namespace QP
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 void bzero(void *const start, std::uint_fast16_t const len) noexcept
Clear a specified region of memory to zero.
Definition: qf_act.cpp:129
static QActive * active_[QF_MAX_ACTIVE+1U]
array of registered active objects
Definition: qf.hpp:580
QHsmAttr m_temp
temporary: transition chain, target state, etc.
Definition: qep.hpp:271
Priority Ceiling Mutex the QXK preemptive kernel.
Definition: qxthread.hpp:210
bool tryLock(void) noexcept
try to lock the QXK priority-ceiling mutex QP::QXMutex
Definition: qxk_mutex.cpp:257
void unlock(void) noexcept
unlock the QXK priority-ceiling mutex QP::QXMutex
Definition: qxk_mutex.cpp:345
bool lock(std::uint_fast16_t const nTicks=QXTHREAD_NO_TIMEOUT) noexcept
lock the QXK priority-ceiling mutex QP::QXMutex
Definition: qxk_mutex.cpp:129
Extended (blocking) thread of the QXK preemptive kernel.
Definition: qxthread.hpp:66
QTimeEvt m_timeEvt
time event to handle blocking timeouts
Definition: qxthread.hpp:131
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
@ QS_MUTEX_UNLOCK
a mutex was unlocked
Definition: qs.hpp:137
@ QS_MUTEX_LOCK
a mutex was locked
Definition: qs.hpp:136
QMState const * obj
pointer to QMState object
Definition: qep.hpp:242
@ QXK_SEMA_SIG
Definition: qxk_pkg.hpp:48
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_REQUIRE_ID(id_, test_)
Assertion for checking preconditions with user-specified assertion-id.
Definition: qassert.h:279
int enum_t
alias for enumerations used for event signals
Definition: qep.hpp:82
#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_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 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_END_NOCRIT_PRE_()
Definition: qs_dummy.hpp:99
#define QS_2U8_PRE_(data1_, data2_)
Definition: qs_dummy.hpp:101
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
void QXK_activate_(void)
QXK activator activates the next active object. The activated AO preempts.
Definition: qxk.cpp:423
std::uint_fast8_t QXK_sched_(void) noexcept
QXK scheduler finds the highest-priority thread ready to run.
Definition: qxk.cpp:346
std::uint8_t volatile actPrio
prio of the active basic thread
Definition: qxk.hpp:77
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.
#define QXK_PTR_CAST_(type_, ptr_)
intertnal macro to encapsulate casting of pointers for MISRA deviations
Definition: qxk_pkg.hpp:76
QSignal sig
signal of the event instance
Definition: qep.hpp:210
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