QP/C++ 6.9.3
qep_msm.cpp
Go to the documentation of this file.
1 
39 #define QP_IMPL // this is QP implementation
40 #include "qep_port.hpp" // QEP port
41 #ifdef Q_SPY // QS software tracing enabled?
42  #include "qs_port.hpp" // QS port
43  #include "qs_pkg.hpp" // QS facilities for pre-defined trace records
44 #else
45  #include "qs_dummy.hpp" // disable the QS software tracing
46 #endif // Q_SPY
47 #include "qassert.h" // QP embedded systems-friendly assertions
48 
53 #define QEP_ACT_PTR_INC_(act_) (++(act_))
54 
55 namespace QP {
56 
57 Q_DEFINE_THIS_MODULE("qep_msm")
58 
59 //****************************************************************************
60 QMState const QMsm::msm_top_s = {
61  nullptr,
62  nullptr,
63  nullptr,
64  nullptr,
65  nullptr
66 };
67 
68 //****************************************************************************
83 QMsm::QMsm(QStateHandler const initial) noexcept
84  : QHsm(initial)
85 {
86  m_state.obj = &msm_top_s;
87  m_temp.fun = initial;
88 }
89 
90 //****************************************************************************
101 void QMsm::init(void const * const e, std::uint_fast8_t const qs_id) {
103 
106  Q_REQUIRE_ID(200, (m_temp.fun != nullptr)
107  && (m_state.obj == &msm_top_s));
108 
109  // execute the top-most initial tran.
110  QState r = (*m_temp.fun)(this, Q_EVT_CAST(QEvt));
111 
112  // initial tran. must be taken
113  Q_ASSERT_ID(210, r == Q_RET_TRAN_INIT);
114 
116  QS_OBJ_PRE_(this); // this state machine object
117  QS_FUN_PRE_(m_state.obj->stateHandler); // source handler
118  QS_FUN_PRE_(m_temp.tatbl->target->stateHandler); // target handler
119  QS_END_PRE_()
120 
121  // set state to the last tran. target
123 
124  // drill down into the state hierarchy with initial transitions...
125  do {
126  r = execTatbl_(m_temp.tatbl, qs_id); // execute the tran-action table
127  } while (r >= Q_RET_TRAN_INIT);
128 
130  QS_TIME_PRE_(); // time stamp
131  QS_OBJ_PRE_(this); // this state machine object
132  QS_FUN_PRE_(m_state.obj->stateHandler); // the new current state
133  QS_END_PRE_()
134 
135  static_cast<void>(qs_id); // unused parameter (if Q_SPY not defined)
136 }
137 
138 //****************************************************************************
149 void QMsm::dispatch(QEvt const * const e, std::uint_fast8_t const qs_id) {
150  QMState const *s = m_state.obj; // store the current state
151  QMState const *t = s;
152  QState r;
154 
156  Q_REQUIRE_ID(300, s != nullptr);
157 
159  QS_TIME_PRE_(); // time stamp
160  QS_SIG_PRE_(e->sig); // the signal of the event
161  QS_OBJ_PRE_(this); // this state machine object
162  QS_FUN_PRE_(s->stateHandler); // the current state handler
163  QS_END_PRE_()
164 
165  // scan the state hierarchy up to the top state...
166  do {
167  r = (*t->stateHandler)(this, e); // call state handler function
168 
169  // event handled? (the most frequent case)
170  if (r >= Q_RET_HANDLED) {
171  break; // done scanning the state hierarchy
172  }
173  // event unhandled and passed to the superstate?
174  else if (r == Q_RET_SUPER) {
175  t = t->superstate; // advance to the superstate
176  }
177  // event unhandled and passed to a submachine superstate?
178  else if (r == Q_RET_SUPER_SUB) {
179  t = m_temp.obj; // current host state of the submachie
180  }
181  // event unhandled due to a guard?
182  else if (r == Q_RET_UNHANDLED) {
183 
185  QS_SIG_PRE_(e->sig); // the signal of the event
186  QS_OBJ_PRE_(this); // this state machine object
187  QS_FUN_PRE_(t->stateHandler); // the current state
188  QS_END_PRE_()
189 
190  t = t->superstate; // advance to the superstate
191  }
192  else {
193  // no other return value should be produced
194  Q_ERROR_ID(310);
195  }
196  } while (t != nullptr);
197 
198  // any kind of transition taken?
199  if (r >= Q_RET_TRAN) {
200 #ifdef Q_SPY
201  QMState const * const ts = t; // transition source for QS tracing
202 
203  // the transition source state must not be nullptr
204  Q_ASSERT_ID(320, ts != nullptr);
205 #endif // Q_SPY
206 
207  do {
208  // save the transition-action table before it gets clobbered
209  QMTranActTable const * const tatbl = m_temp.tatbl;
210  QHsmAttr tmp; // temporary to save intermediate values
211 
212  // was TRAN, TRAN_INIT, or TRAN_EP taken?
213  if (r <= Q_RET_TRAN_EP) {
214  exitToTranSource_(s, t, qs_id);
215  r = execTatbl_(tatbl, qs_id);
216  s = m_state.obj;
217  }
218  // was a transition segment to history taken?
219  else if (r == Q_RET_TRAN_HIST) {
220  tmp.obj = m_state.obj; // save history
221  m_state.obj = s; // restore the original state
222  exitToTranSource_(s, t, qs_id);
223  static_cast<void>(execTatbl_(tatbl, qs_id));
224  r = enterHistory_(tmp.obj, qs_id);
225  s = m_state.obj;
226  }
227  // was a transition segment to an exit point taken?
228  else if (r == Q_RET_TRAN_XP) {
229  tmp.act = m_state.act; // save XP action
230  m_state.obj = s; // restore the original state
231  r = (*tmp.act)(this); // execute the XP action
232  if (r == Q_RET_TRAN) { // XP -> TRAN ?
233 #ifdef Q_SPY
234  tmp.tatbl = m_temp.tatbl; // save m_temp
235 #endif // Q_SPY
236  exitToTranSource_(s, t, qs_id);
237  // take the tran-to-XP segment inside submachine
238  static_cast<void>(execTatbl_(tatbl, qs_id));
239  s = m_state.obj;
240 #ifdef Q_SPY
241  m_temp.tatbl = tmp.tatbl; // restore m_temp
242 #endif // Q_SPY
243  }
244  else if (r == Q_RET_TRAN_HIST) { // XP -> HIST ?
245  tmp.obj = m_state.obj; // save the history
246  m_state.obj = s; // restore the original state
247 #ifdef Q_SPY
248  s = m_temp.obj; // save m_temp
249 #endif // Q_SPY
250  exitToTranSource_(m_state.obj, t, qs_id);
251  // take the tran-to-XP segment inside submachine
252  static_cast<void>(execTatbl_(tatbl, qs_id));
253 #ifdef Q_SPY
254  m_temp.obj = s; // restore me->temp
255 #endif // Q_SPY
256  s = m_state.obj;
257  m_state.obj = tmp.obj; // restore the history
258  }
259  else {
260  // TRAN_XP must NOT be followed by any other tran type
261  Q_ASSERT_ID(330, r < Q_RET_TRAN);
262  }
263  }
264  else {
265  // no other return value should be produced
266  Q_ERROR_ID(340);
267  }
268 
269  t = s; // set target to the current state
270 
271  } while (r >= Q_RET_TRAN);
272 
273  QS_BEGIN_PRE_(QS_QEP_TRAN, qs_id)
274  QS_TIME_PRE_(); // time stamp
275  QS_SIG_PRE_(e->sig); // the signal of the event
276  QS_OBJ_PRE_(this); // this state machine object
277  QS_FUN_PRE_(ts->stateHandler); // the transition source
278  QS_FUN_PRE_(s->stateHandler); // the new active state
279  QS_END_PRE_()
280  }
281 
282 #ifdef Q_SPY
283  // was the event handled?
284  else if (r == Q_RET_HANDLED) {
285  // internal tran. source can't be nullptr
286  Q_ASSERT_ID(340, t != nullptr);
287 
289  QS_TIME_PRE_(); // time stamp
290  QS_SIG_PRE_(e->sig); // the signal of the event
291  QS_OBJ_PRE_(this); // this state machine object
292  QS_FUN_PRE_(t->stateHandler); // the source state
293  QS_END_PRE_()
294 
295  }
296  // event bubbled to the 'top' state?
297  else if (t == nullptr) {
298 
300  QS_TIME_PRE_(); // time stamp
301  QS_SIG_PRE_(e->sig); // the signal of the event
302  QS_OBJ_PRE_(this); // this state machine object
303  QS_FUN_PRE_(s->stateHandler); // the current state
304  QS_END_PRE_()
305 
306  }
307 #endif // Q_SPY
308 
309  else {
310  // empty
311  }
312 
313  static_cast<void>(qs_id); // unused parameter (if Q_SPY not defined)
314 }
315 
316 //****************************************************************************
317 #ifdef Q_SPY
322  return m_state.obj->stateHandler;
323  }
324 #endif
325 
326 //****************************************************************************
341  std::uint_fast8_t const qs_id)
342 {
343  QActionHandler const *a;
344  QState r = Q_RET_NULL;
346 
348  Q_REQUIRE_ID(400, tatbl != nullptr);
349 
350  for (a = &tatbl->act[0]; *a != nullptr; QEP_ACT_PTR_INC_(a)) {
351  r = (*(*a))(this); // call the action through the 'a' pointer
352 #ifdef Q_SPY
353  if (r == Q_RET_ENTRY) {
354 
356  QS_OBJ_PRE_(this); // this state machine object
357  QS_FUN_PRE_(m_temp.obj->stateHandler); // entered state handler
358  QS_END_PRE_()
359  }
360  else if (r == Q_RET_EXIT) {
361 
363  QS_OBJ_PRE_(this); // this state machine object
364  QS_FUN_PRE_(m_temp.obj->stateHandler); // exited state handler
365  QS_END_PRE_()
366  }
367  else if (r == Q_RET_TRAN_INIT) {
368 
370  QS_OBJ_PRE_(this); // this state machine object
371  QS_FUN_PRE_(tatbl->target->stateHandler); // source
373  QS_END_PRE_()
374  }
375  else if (r == Q_RET_TRAN_EP) {
376 
378  QS_OBJ_PRE_(this); // this state machine object
379  QS_FUN_PRE_(tatbl->target->stateHandler); // source
381  QS_END_PRE_()
382  }
383  else if (r == Q_RET_TRAN_XP) {
384 
386  QS_OBJ_PRE_(this); // this state machine object
387  QS_FUN_PRE_(tatbl->target->stateHandler); // source
389  QS_END_PRE_()
390  }
391  else {
392  // empty
393  }
394 #endif // Q_SPY
395  }
396 
397  static_cast<void>(qs_id); // unused parameter (if Q_SPY not defined)
398 
399  m_state.obj = (r >= Q_RET_TRAN)
400  ? m_temp.tatbl->target
401  : tatbl->target;
402  return r;
403 }
404 
405 //****************************************************************************
416  QMState const * const ts,
417  std::uint_fast8_t const qs_id)
418 {
419  // exit states from the current state to the tran. source state
420  while (s != ts) {
421  // exit action provided in state 's'?
422  if (s->exitAction != nullptr) {
423  // execute the exit action
424  static_cast<void>((*s->exitAction)(this));
425 
428  QS_OBJ_PRE_(this); // this state machine object
429  QS_FUN_PRE_(s->stateHandler); // the exited state handler
430  QS_END_PRE_()
431  }
432 
433  s = s->superstate; // advance to the superstate
434 
435  // reached the top of a submachine?
436  if (s == nullptr) {
437  s = m_temp.obj; // the superstate from QM_SM_EXIT()
438  Q_ASSERT_ID(510, s != nullptr);
439  }
440  }
441  static_cast<void>(qs_id); // unused parameter (if Q_SPY not defined)
442 }
443 
444 //****************************************************************************
456 QState QMsm::enterHistory_(QMState const * const hist,
457  std::uint_fast8_t const qs_id)
458 {
459  QMState const *s = hist;
460  QMState const *ts = m_state.obj; // transition source
461  QMState const *epath[MAX_ENTRY_DEPTH_];
462  QState r;
463  std::uint_fast8_t i = 0U; // entry path index
465 
467  QS_OBJ_PRE_(this); // this state machine object
468  QS_FUN_PRE_(ts->stateHandler); // source state handler
469  QS_FUN_PRE_(hist->stateHandler); // target state handler
470  QS_END_PRE_()
471 
472  while (s != ts) {
473  if (s->entryAction != nullptr) {
474  epath[i] = s;
475  ++i;
476  Q_ASSERT_ID(620,
477  i <= static_cast<std::uint_fast8_t>(Q_DIM(epath)));
478  }
479  s = s->superstate;
480  if (s == nullptr) {
481  ts = s; // force exit from the for-loop
482  }
483  }
484 
485  // retrace the entry path in reverse (desired) order...
486  while (i > 0U) {
487  --i;
488  // run entry action in epath[i]
489  static_cast<void>((*epath[i]->entryAction)(this));
490 
492  QS_OBJ_PRE_(this);
493  QS_FUN_PRE_(epath[i]->stateHandler); // entered state handler
494  QS_END_PRE_()
495  }
496 
497  m_state.obj = hist; // set current state to the transition target
498 
499  // initial tran. present?
500  if (hist->initAction != nullptr) {
501  r = (*hist->initAction)(this); // execute the transition action
502  }
503  else {
504  r = Q_RET_NULL;
505  }
506 
507  static_cast<void>(qs_id); // unused parameter (if Q_SPY not defined)
508  return r;
509 }
510 
511 //****************************************************************************
524 bool QMsm::isInState(QMState const * const st) const noexcept {
525  bool inState = false; // assume that this MSM is not in 'state'
526 
527  for (QMState const *s = m_state.obj;
528  s != nullptr;
529  s = s->superstate)
530  {
531  if (s == st) {
532  inState = true; // match found, return 'true'
533  break;
534  }
535  }
536  return inState;
537 }
538 
539 //****************************************************************************
554 QMState const *QMsm::childStateObj(QMState const * const parent)
555  const noexcept
556 {
557  QMState const *child = m_state.obj;
558  bool isFound = false; // start with the child not found
559 
560  for (QMState const *s = m_state.obj;
561  s != nullptr;
562  s = s->superstate)
563  {
564  if (s == parent) {
565  isFound = true; // child is found
566  break;
567  }
568  else {
569  child = s;
570  }
571  }
572 
574  Q_ENSURE_ID(810, isFound);
575 #ifdef Q_NASSERT
576  // avoid compiler warning about unused variable
577  static_cast<void>(isFound);
578 #endif
579 
580  return child; // return the child
581 }
582 
583 } // namespace QP
584 
unsigned int uint_fast8_t
fast at-least 8-bit unsigned int
Definition: 16bit/stdint.h:36
Hierarchical State Machine base class.
Definition: qep.hpp:269
static constexpr QState Q_RET_SUPER_SUB
event passed to submachine superstate
Definition: qep.hpp:317
static constexpr QState Q_RET_HANDLED
event handled (internal transition)
Definition: qep.hpp:323
friend class QMsm
Definition: qep.hpp:469
static constexpr QState Q_RET_SUPER
event passed to the superstate to handle
Definition: qep.hpp:314
static constexpr QState Q_RET_UNHANDLED
event unhandled due to a guard evaluating to 'false'
Definition: qep.hpp:320
static constexpr QState Q_RET_TRAN_EP
entry-point transition into a submachine
Definition: qep.hpp:344
static constexpr QState Q_RET_TRAN_INIT
initial transition taken
Definition: qep.hpp:341
static constexpr QState Q_RET_NULL
return value without any effect
Definition: qep.hpp:335
static constexpr QState Q_RET_TRAN
regular transition taken
Definition: qep.hpp:338
QHsmAttr m_state
current active state (state-variable)
Definition: qep.hpp:270
static constexpr QState Q_RET_EXIT
state exit action executed
Definition: qep.hpp:332
static constexpr QState Q_RET_TRAN_HIST
transition to history of a given state
Definition: qep.hpp:347
QHsmAttr m_temp
temporary: transition chain, target state, etc.
Definition: qep.hpp:271
static constexpr QState Q_RET_TRAN_XP
exit-point transition out of a submachine
Definition: qep.hpp:350
static constexpr QState Q_RET_ENTRY
state entry action executed
Definition: qep.hpp:329
void exitToTranSource_(QMState const *s, QMState const *const ts, std::uint_fast8_t const qs_id)
Internal helper function to exit current state to transition source.
Definition: qep_msm.cpp:415
static QMState const msm_top_s
the top state object for the QMsm
Definition: qep.hpp:569
QStateHandler getStateHandler() noexcept override
Get the current state handler of the QMsm.
Definition: qep_msm.cpp:321
QState execTatbl_(QMTranActTable const *const tatbl, std::uint_fast8_t const qs_id)
Internal helper function to execute a transition-action table.
Definition: qep_msm.cpp:340
void dispatch(QEvt const *const e, std::uint_fast8_t const qs_id) override
Dispatches an event to a HSM.
Definition: qep_msm.cpp:149
static constexpr std::int_fast8_t MAX_ENTRY_DEPTH_
maximum depth of implemented entry levels for transitions to history
Definition: qep.hpp:566
QMState const * childStateObj(QMState const *const parent) const noexcept
Obtain the current active child state of a given parent (read only)
Definition: qep_msm.cpp:554
QState enterHistory_(QMState const *const hist, std::uint_fast8_t const qs_id)
Internal helper function to enter state history.
Definition: qep_msm.cpp:456
void init(void const *const e, std::uint_fast8_t const qs_id) override
Performs the second step of SM initialization by triggering the top-most initial transition.
Definition: qep_msm.cpp:101
bool isInState(QMState const *const st) const noexcept
Tests if a given state is part of the active state configuration.
Definition: qep_msm.cpp:524
namespace associated with the QP/C++ framework
Definition: struct.dox:1
std::uint_fast8_t QState
Type returned from state-handler functions.
Definition: qep.hpp:223
@ QS_QEP_STATE_INIT
an initial transition was taken in a state
Definition: qs.hpp:69
@ QS_QEP_TRAN_HIST
a tran to history was taken
Definition: qs.hpp:145
@ QS_QEP_TRAN_EP
a tran to entry point into a submachine
Definition: qs.hpp:146
@ QS_QEP_STATE_EXIT
a state was exited
Definition: qs.hpp:68
@ QS_QEP_INIT_TRAN
the top-most initial transition was taken
Definition: qs.hpp:70
@ QS_QEP_INTERN_TRAN
an internal transition was taken
Definition: qs.hpp:71
@ QS_QEP_TRAN_XP
a tran to exit point out of a submachine
Definition: qs.hpp:147
@ QS_QEP_STATE_ENTRY
a state was entered
Definition: qs.hpp:67
@ QS_QEP_UNHANDLED
an event was unhandled due to a guard
Definition: qs.hpp:75
@ QS_QEP_TRAN
a regular transition was taken
Definition: qs.hpp:72
@ QS_QEP_DISPATCH
an event was dispatched (begin of RTC step)
Definition: qs.hpp:74
@ QS_QEP_IGNORED
an event was ignored (silently discarded)
Definition: qs.hpp:73
QActionHandler const act[1]
Definition: qep.hpp:595
QMState const * obj
pointer to QMState object
Definition: qep.hpp:242
QMTranActTable const * tatbl
transition-action table
Definition: qep.hpp:243
QActionHandler const entryAction
entry action handler function
Definition: qep.hpp:587
QMState const * superstate
superstate of this state
Definition: qep.hpp:585
QState(*)(void *const me) QActionHandler
Pointer to an action-handler function.
Definition: qep.hpp:229
QMState const * target
Definition: qep.hpp:594
QActionHandler const initAction
init action handler function
Definition: qep.hpp:589
QState(*)(void *const me, QEvt const *const e) QStateHandler
Pointer to state-handler function.
Definition: qep.hpp:226
QActionHandler const exitAction
exit action handler function
Definition: qep.hpp:588
QStateHandler const stateHandler
state handler function
Definition: qep.hpp:586
QStateHandler fun
pointer to a state handler function
Definition: qep.hpp:239
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
Transition-Action Table for the QP::QMsm State Machine.
Definition: qep.hpp:593
Attribute of for the QHsm class (Hierarchical State Machine).
Definition: qep.hpp:238
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_DIM(array_)
Helper macro to calculate static dimension of a 1-dim array_.
Definition: qassert.h:337
#define Q_ERROR_ID(id_)
Assertion with user-specified assertion-id for a wrong path.
Definition: qassert.h:211
#define Q_EVT_CAST(class_)
Perform downcast of an event onto a subclass of QEvt class_.
Definition: qep.hpp:108
#define QEP_ACT_PTR_INC_(act_)
Internal macro to increment the given action table act_.
Definition: qep_msm.cpp:53
QEP/C++ port, generic C++11 compiler.
#define QS_CRIT_STAT_
This is an internal macro for defining the critical section status type.
Definition: qs.hpp:783
#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_PRE_(rec_, qs_id_)
Definition: qs_dummy.hpp:96
#define QS_OBJ_PRE_(obj_)
Definition: qs_dummy.hpp:107
#define QS_FUN_PRE_(fun_)
Definition: qs_dummy.hpp:108
#define QS_SIG_PRE_(sig_)
Definition: qs_dummy.hpp:105
#define QS_END_PRE_()
Definition: qs_dummy.hpp:97
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