QP/C++ 6.9.3
qxk_sema.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 namespace QP {
56 
57 Q_DEFINE_THIS_MODULE("qxk_sema")
58 
59 //****************************************************************************
77 void QXSemaphore::init(std::uint_fast16_t const count,
78  std::uint_fast16_t const max_count) noexcept
79 {
81  Q_REQUIRE_ID(100, max_count > 0U);
82 
83  m_count = static_cast<std::uint16_t>(count);
84  m_max_count = static_cast<std::uint16_t>(max_count);
85  m_waitSet.setEmpty();
86 }
87 
88 //****************************************************************************
109 bool QXSemaphore::wait(std::uint_fast16_t const nTicks) noexcept {
110  bool signaled = true; // assume that the semaphore will be signaled
112 
113  QF_CRIT_E_();
114  QXThread * const curr = QXK_PTR_CAST_(QXThread*, QXK_attr_.curr);
115 
123  && (m_max_count > 0U)
124  && (curr != nullptr)
125  && (curr->m_temp.obj == nullptr)); // NOT blocked
127  Q_REQUIRE_ID(201, QXK_attr_.lockHolder != curr->m_prio);
128 
129  if (m_count > 0U) {
130  --m_count;
131  }
132  else {
133  std::uint_fast8_t const p =
134  static_cast<std::uint_fast8_t>(curr->m_dynPrio);
135 
136  // remember the blocking object (this semaphore)
137  curr->m_temp.obj = QXK_PTR_CAST_(QMState*, this);
138  curr->teArm_(static_cast<enum_t>(QXK_SEMA_SIG), nTicks);
139 
140  // remove this curr prio from the ready set (will block)
141  // and insert to the waiting set on this semaphore
142  m_waitSet.insert(p); // add to waiting-set
143  QXK_attr_.readySet.rmove(p); // remove from ready-set
144 
145  // schedule the next thread if multitasking started
146  static_cast<void>(QXK_sched_());
147  QF_CRIT_X_();
148  QF_CRIT_EXIT_NOP(); // BLOCK here !!!
149 
150  QF_CRIT_E_(); // AFTER unblocking...
151  // the blocking object must be this semaphore
152  Q_ASSERT_ID(240, curr->m_temp.obj == QXK_PTR_CAST_(QMState*, this));
153 
154  // did the blocking time-out? (signal of zero means that it did)
155  if (curr->m_timeEvt.sig == 0U) {
156  if (m_waitSet.hasElement(p)) { // still waiting?
157  m_waitSet.rmove(p); // remove the unblocked thread
158  signaled = false; // the semaphore was NOT signaled
159  // semaphore NOT taken: do NOT decrement the count
160  }
161  else { // semaphore was both signaled and timed out
162  --m_count; // semaphore signaled: decrement the count
163  }
164  }
165  else { // blocking did NOT time out
166  // the thread must NOT be waiting on this semaphore
167  Q_ASSERT_ID(250, !m_waitSet.hasElement(p));
168 
169  --m_count; // semaphore signaled: decrement the count
170  }
171  curr->m_temp.obj = nullptr; // clear blocked obj.
172  }
173  QF_CRIT_X_();
174 
175  return signaled;
176 }
177 
178 //****************************************************************************
191 bool QXSemaphore::tryWait(void) noexcept {
192  bool isAvailable;
194 
196  Q_REQUIRE_ID(300, m_max_count > 0U);
197 
198  QF_CRIT_E_();
199  // is the semaphore available?
200  if (m_count > 0U) {
201  --m_count;
202  isAvailable = true;
203  }
204  else { // the semaphore is NOT available (would block)
205  isAvailable = false;
206  }
207  QF_CRIT_X_();
208 
209  return isAvailable;
210 }
211 
212 //****************************************************************************
230 bool QXSemaphore::signal(void) noexcept {
231  bool signaled = true; // assume that the semaphore will be signaled
233 
235  Q_REQUIRE_ID(400, m_max_count > 0U);
236 
237  QF_CRIT_E_();
238  if (m_count < m_max_count) {
239 
240  ++m_count; // increment the semaphore count
241 
242  if (m_waitSet.notEmpty()) {
243 
244  // find the highest-priority thread waiting on this semaphore
245  std::uint_fast8_t const p = m_waitSet.findMax();
246  QXThread * const thr = QXK_PTR_CAST_(QXThread*, QF::active_[p]);
247 
248  // assert that:
249  // - the thread must be registered in QF;
250  // - the thread must be extended; and
251  // - must be blocked on this semaphore;
252  //
253  Q_ASSERT_ID(410, (thr != nullptr)
254  && (thr->m_osObject != nullptr)
255  && (thr->m_temp.obj == QXK_PTR_CAST_(QMState*, this)));
256 
257  // disarm the internal time event
258  static_cast<void>(thr->teDisarm_());
259 
260  // make the thread ready to run and remove from the wait-list
262  m_waitSet.rmove(p);
263 
264  if (!QXK_ISR_CONTEXT_()) { // not inside ISR?
265  static_cast<void>(QXK_sched_()); // schedule the next thread
266  }
267  }
268  }
269  else {
270  signaled = false; // semaphore NOT signaled
271  }
272  QF_CRIT_X_();
273 
274  return signaled;
275 }
276 
277 } // namespace QP
278 
unsigned int uint16_t
exact-width 16-bit unsigned int
Definition: 16bit/stdint.h:30
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
std::uint8_t m_prio
QF priority (1..QF_MAX_ACTIVE) of this active object.
Definition: qf.hpp:185
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
Counting Semaphore of the QXK preemptive kernel.
Definition: qxthread.hpp:153
bool wait(std::uint_fast16_t const nTicks=QXTHREAD_NO_TIMEOUT) noexcept
wait (block) on the semaphore
Definition: qxk_sema.cpp:109
bool signal(void) noexcept
signal (unblock) the semaphore
Definition: qxk_sema.cpp:230
bool tryWait(void) noexcept
try wait on the semaphore (non-blocking)
Definition: qxk_sema.cpp:191
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
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
Dummy definitions of the QS macros that avoid code generation from the QS instrumentation.
Internal (package scope) QS/C++ interface.
QS/C++ port to a 32-bit CPU, generic compiler.
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.
#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