2014-04-13 21:35:34 -04:00
|
|
|
/// \file
|
|
|
|
/// \ingroup qep
|
|
|
|
/// \brief QEP/C++ platform-independent public interface.
|
|
|
|
/// \cond
|
|
|
|
///***************************************************************************
|
|
|
|
/// Product: QEP/C++
|
2014-09-22 11:48:11 -04:00
|
|
|
/// Last updated for version 5.3.1
|
|
|
|
/// Last updated on 2014-09-18
|
2014-04-13 21:35:34 -04:00
|
|
|
///
|
|
|
|
/// Q u a n t u m L e a P s
|
|
|
|
/// ---------------------------
|
|
|
|
/// innovating embedded systems
|
|
|
|
///
|
|
|
|
/// Copyright (C) Quantum Leaps, www.state-machine.com.
|
|
|
|
///
|
|
|
|
/// This program is open source software: you can redistribute it and/or
|
|
|
|
/// modify it under the terms of the GNU General Public License as published
|
|
|
|
/// by the Free Software Foundation, either version 3 of the License, or
|
|
|
|
/// (at your option) any later version.
|
|
|
|
///
|
|
|
|
/// Alternatively, this program may be distributed and modified under the
|
|
|
|
/// terms of Quantum Leaps commercial licenses, which expressly supersede
|
|
|
|
/// the GNU General Public License and are specifically designed for
|
|
|
|
/// licensees interested in retaining the proprietary status of their code.
|
|
|
|
///
|
|
|
|
/// This program is distributed in the hope that it will be useful,
|
|
|
|
/// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
/// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
/// GNU General Public License for more details.
|
|
|
|
///
|
|
|
|
/// You should have received a copy of the GNU General Public License
|
|
|
|
/// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
///
|
|
|
|
/// Contact information:
|
|
|
|
/// Web: www.state-machine.com
|
|
|
|
/// Email: info@state-machine.com
|
|
|
|
///***************************************************************************
|
|
|
|
/// \endcond
|
|
|
|
|
2012-08-14 18:00:48 -04:00
|
|
|
#ifndef qep_h
|
|
|
|
#define qep_h
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//****************************************************************************
|
|
|
|
//! The current QP version number
|
|
|
|
///
|
|
|
|
/// version of the QP as a decimal constant XYZ, where X is a 1-digit
|
|
|
|
/// major version number, Y is a 1-digit minor version number, and Z is
|
|
|
|
/// a 1-digit release number.
|
|
|
|
///
|
2014-09-22 11:48:11 -04:00
|
|
|
#define QP_VERSION 531
|
2014-04-13 21:35:34 -04:00
|
|
|
|
|
|
|
//! The current QP version string
|
2014-09-22 11:48:11 -04:00
|
|
|
#define QP_VERSION_STR "5.3.1"
|
2014-04-13 21:35:34 -04:00
|
|
|
|
2014-09-22 11:48:11 -04:00
|
|
|
//! Tamperproof current QP release (5.3.1) and date (14-09-18)
|
|
|
|
#define QP_RELEASE 0xAC019C8CU
|
2014-04-13 21:35:34 -04:00
|
|
|
|
|
|
|
#ifndef Q_ROM
|
|
|
|
//! Macro to specify compiler-specific directive for placing a
|
|
|
|
//! constant object in ROM.
|
|
|
|
/// \description
|
|
|
|
/// Many compilers for Harvard-architecture MCUs provide non-standard
|
|
|
|
/// extensions to support placement of objects in different memories.
|
|
|
|
/// In order to conserve the precious RAM, QP uses the Q_ROM macro for
|
|
|
|
/// all constant objects that can be allocated in ROM.
|
|
|
|
///
|
|
|
|
/// \note
|
|
|
|
/// To override the following empty definition, you need to define the
|
|
|
|
/// Q_ROM macro in the qep_port.h header file. Some examples of valid
|
|
|
|
/// Q_ROM macro definitions are: __code (IAR 8051 compiler), code (Keil
|
|
|
|
/// Cx51 compiler), PROGMEM (gcc for AVR), __flash (IAR for AVR).
|
|
|
|
#define Q_ROM
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef Q_ROM_BYTE
|
|
|
|
//! Macro to access a byte allocated in ROM
|
|
|
|
/// \description
|
|
|
|
/// Some compilers for Harvard-architecture MCUs, such as gcc for AVR, do
|
|
|
|
/// not generate correct code for accessing data allocated in the program
|
|
|
|
/// space (ROM). The workaround for such compilers is to explictly add
|
|
|
|
/// assembly code to access each data element allocated in the program
|
|
|
|
/// space. The macro Q_ROM_BYTE() retrieves a byte from the given ROM
|
|
|
|
/// address.
|
|
|
|
///
|
|
|
|
/// \note
|
|
|
|
/// The Q_ROM_BYTE() macro should be defined for the compilers that
|
|
|
|
/// cannot handle correctly data allocated in ROM (such as the gcc).
|
|
|
|
/// If the macro is left undefined, the default definition simply returns
|
|
|
|
/// the argument and lets the compiler generate the correct code.
|
|
|
|
#define Q_ROM_BYTE(rom_var_) (rom_var_)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef Q_SIGNAL_SIZE
|
|
|
|
//! The size (in bytes) of the signal of an event. Valid values:
|
|
|
|
//! 1, 2, or 4; default 1
|
|
|
|
/// \description
|
|
|
|
/// This macro can be defined in the QEP port file (qep_port.h) to
|
|
|
|
/// configure the QP::QSignal type. When the macro is not defined, the
|
|
|
|
/// default of 1 byte is chosen.
|
|
|
|
#define Q_SIGNAL_SIZE 2
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//****************************************************************************
|
|
|
|
//! helper macro to calculate static dimension of a 1-dim array \a array_
|
|
|
|
#define Q_DIM(array_) (sizeof(array_) / sizeof((array_)[0]))
|
|
|
|
|
|
|
|
//****************************************************************************
|
|
|
|
// typedefs for basic numerical types; MISRA-C++ 2008 rule 3-9-2(req).
|
|
|
|
|
|
|
|
//! typedef for character strings.
|
|
|
|
/// \description
|
|
|
|
/// This typedef specifies character type for exclusive use in character
|
|
|
|
/// strings. Use of this type, rather than plain 'char', is in compliance
|
|
|
|
/// with the MISRA-C 2004 Rules 6.1(req), 6.3(adv).
|
|
|
|
typedef char char_t;
|
|
|
|
|
|
|
|
//! typedef for line numbers in assertions and return from QF_run()
|
|
|
|
typedef int int_t;
|
|
|
|
|
|
|
|
//! typedef for enumerations used for event signals
|
|
|
|
typedef int enum_t;
|
|
|
|
|
|
|
|
//! typedef for 32-bit IEEE 754 floating point numbers
|
|
|
|
/// \note
|
|
|
|
/// QP does not use floating-point types anywhere in the internal
|
|
|
|
/// implementation, except in QS software tracing, where utilities for
|
|
|
|
/// output of floating-point numbers are provided for application-level
|
|
|
|
/// trace records.
|
|
|
|
///
|
|
|
|
typedef float float32_t;
|
|
|
|
|
|
|
|
//! typedef for 64-bit IEEE 754 floating point numbers
|
|
|
|
/// \note
|
|
|
|
/// QP does not use floating-point types anywhere in the internal
|
|
|
|
/// implementation, except in QS software tracing, where utilities for
|
|
|
|
/// output of floating-point numbers are provided for application-level
|
|
|
|
/// trace records.
|
|
|
|
///
|
|
|
|
typedef double float64_t;
|
|
|
|
|
|
|
|
|
|
|
|
//! Perform downcast of an event onto a subclass of QEvt \a class_
|
|
|
|
/// \description
|
|
|
|
/// This macro encapsulates the downcast of QEvt pointers, which violates
|
|
|
|
/// MISRA-C 2004 rule 11.4(advisory). This macro helps to localize this
|
|
|
|
/// deviation.
|
|
|
|
///
|
|
|
|
#define Q_EVT_CAST(class_) (static_cast<class_ const *>(e))
|
|
|
|
|
|
|
|
//! Perform cast from unsigned integer \a uint_ to pointer of type \a type_.
|
|
|
|
/// \description
|
|
|
|
/// This macro encapsulates the cast to (type_ *), which QP ports or
|
|
|
|
/// application might use to access embedded hardware registers.
|
|
|
|
/// Such uses can trigger PC-Lint "Note 923: cast from int to pointer"
|
|
|
|
/// and this macro helps to encapsulate this deviation.
|
2012-08-14 18:00:48 -04:00
|
|
|
///
|
2014-04-13 21:35:34 -04:00
|
|
|
#define Q_UINT2PTR_CAST(type_, uint_) (reinterpret_cast<type_ *>(uint_))
|
|
|
|
|
|
|
|
//! Initializer of static constant QEvt instances
|
|
|
|
/// \description
|
|
|
|
/// This macro encapsulates the ugly casting of enumerated signals
|
|
|
|
/// to QSignal and constants for QEvt.poolID and QEvt.refCtr_.
|
|
|
|
///
|
|
|
|
#define QEVT_INITIALIZER(sig_) { static_cast<QP::QSignal>(sig_), \
|
|
|
|
static_cast<uint8_t>(0), static_cast<uint8_t>(0) }
|
2012-08-14 18:00:48 -04:00
|
|
|
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//****************************************************************************
|
2013-10-10 20:01:51 -04:00
|
|
|
namespace QP {
|
2014-04-13 21:35:34 -04:00
|
|
|
|
|
|
|
#if (Q_SIGNAL_SIZE == 1)
|
|
|
|
typedef uint8_t QSignal;
|
|
|
|
#elif (Q_SIGNAL_SIZE == 2)
|
|
|
|
//! QSignal represents the signal of an event.
|
|
|
|
/// \description
|
|
|
|
/// The relationship between an event and a signal is as follows. A signal
|
|
|
|
/// in UML is the specification of an asynchronous stimulus that triggers
|
|
|
|
/// reactions, and as such is an essential part of an event. (The signal
|
|
|
|
/// conveys the type of the occurrence--what happened?) However, an event
|
|
|
|
/// can also contain additional quantitative information about the
|
|
|
|
/// occurrence in form of event parameters.
|
|
|
|
typedef uint16_t QSignal;
|
|
|
|
#elif (Q_SIGNAL_SIZE == 4)
|
|
|
|
typedef uint32_t QSignal;
|
|
|
|
#else
|
|
|
|
#error "Q_SIGNAL_SIZE defined incorrectly, expected 1, 2, or 4"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef Q_EVT_CTOR // Provide the constructor for the QEvt class?
|
|
|
|
|
|
|
|
//************************************************************************
|
|
|
|
class QEvt {
|
|
|
|
public:
|
|
|
|
QSignal sig; //!< signal of the event instance
|
|
|
|
|
|
|
|
//! the constructor
|
|
|
|
QEvt(QSignal const s) // poolId_/refCtr_ intentionally uninitialized
|
|
|
|
: sig(s) {}
|
|
|
|
|
|
|
|
#ifdef Q_EVT_VIRTUAL
|
|
|
|
// virtual destructor
|
|
|
|
virtual ~QEvt() {}
|
|
|
|
#endif // Q_EVT_VIRTUAL
|
|
|
|
|
|
|
|
private:
|
|
|
|
uint8_t poolId_; //!< pool ID (0 for static event)
|
|
|
|
uint8_t volatile refCtr_; //!< reference counter
|
|
|
|
|
|
|
|
friend class QF;
|
|
|
|
friend class QActive;
|
|
|
|
friend class QTimeEvt;
|
|
|
|
friend class QEQueue;
|
|
|
|
friend uint8_t QF_EVT_POOL_ID_ (QEvt const * const e);
|
|
|
|
friend uint8_t QF_EVT_REF_CTR_ (QEvt const * const e);
|
|
|
|
friend void QF_EVT_REF_CTR_INC_(QEvt const * const e);
|
|
|
|
friend void QF_EVT_REF_CTR_DEC_(QEvt const * const e);
|
|
|
|
};
|
|
|
|
|
|
|
|
#else // QEvt is a POD (Plain Old Datatype)
|
|
|
|
|
|
|
|
//************************************************************************
|
|
|
|
//! QEvt base class.
|
|
|
|
/// \description
|
|
|
|
/// QEvt represents events without parameters and serves as the
|
|
|
|
/// base class for derivation of events with parameters.
|
|
|
|
///
|
|
|
|
/// \usage
|
|
|
|
/// The following example illustrates how to add an event parameter by
|
|
|
|
/// inheriting from the QEvt class.
|
|
|
|
/// \include qep_qevt.cpp
|
|
|
|
struct QEvt {
|
|
|
|
QSignal sig; //!< signal of the event instance
|
|
|
|
uint8_t poolId_; //!< pool ID (0 for static event)
|
|
|
|
uint8_t volatile refCtr_; //!< reference counter
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif // Q_EVT_CTOR
|
|
|
|
|
|
|
|
|
|
|
|
//! Type returned from state-handler functions
|
|
|
|
typedef uint_fast8_t QState;
|
|
|
|
|
|
|
|
//! pointer to state-handler function
|
2013-10-10 20:01:51 -04:00
|
|
|
typedef QState (*QStateHandler)(void * const me, QEvt const * const e);
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! pointer to an action-handler function
|
2013-10-10 20:01:51 -04:00
|
|
|
typedef QState (*QActionHandler)(void * const me);
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! State object for the QMsm class (Meta State Machine).
|
|
|
|
/// \description
|
2013-10-10 20:01:51 -04:00
|
|
|
/// This class groups together the attributes of a QMsm state, such as the
|
|
|
|
/// parent state (state nesting), the associated state handler function and
|
|
|
|
/// the exit action handler function. These attributes are used inside the
|
2014-04-13 21:35:34 -04:00
|
|
|
/// QP::QMsm::dispatch() and QP::QMsm::init() functions.
|
2013-10-10 20:01:51 -04:00
|
|
|
///
|
2014-04-13 21:35:34 -04:00
|
|
|
/// \attention
|
|
|
|
/// The QMStateObj class is only intended for the QM code generator and
|
|
|
|
/// should not be used in hand-crafted code.
|
2013-10-10 20:01:51 -04:00
|
|
|
struct QMState {
|
2014-04-13 21:35:34 -04:00
|
|
|
QMState const *superstate; //!< superstate of this state
|
|
|
|
QStateHandler const stateHandler; //!< state handler function
|
|
|
|
QActionHandler const entryAction; //!< entry action handler function
|
|
|
|
QActionHandler const exitAction; //!< exit action handler function
|
|
|
|
QActionHandler const initAction; //!< init action handler function
|
2013-10-10 20:01:51 -04:00
|
|
|
};
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Transition-Action Table for the Meta State Machine.
|
|
|
|
struct QMTranActTable {
|
|
|
|
QMState const *target;
|
|
|
|
QActionHandler const act[1];
|
|
|
|
};
|
|
|
|
|
|
|
|
//! Attribute of for the QMsm class (Meta State Machine).
|
|
|
|
/// \description
|
2013-10-10 20:01:51 -04:00
|
|
|
/// This union represents possible values stored in the 'state' and 'temp'
|
|
|
|
/// attributes of the QMsm class.
|
|
|
|
union QMAttr {
|
2014-04-13 21:35:34 -04:00
|
|
|
QMState const *obj; //!< pointer to QMState object
|
|
|
|
QMTranActTable const *tatbl; //!< transition-action table
|
|
|
|
QStateHandler fun; //!< pointer to a state handler function
|
|
|
|
QActionHandler act; //!< pointer to an action-handler function
|
2012-08-14 18:00:48 -04:00
|
|
|
};
|
|
|
|
|
2013-10-10 20:01:51 -04:00
|
|
|
//****************************************************************************
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! event passed to the superstate to handle
|
2013-10-10 20:01:51 -04:00
|
|
|
QState const Q_RET_SUPER = static_cast<QState>(0);
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! event passed to submachine superstate
|
|
|
|
QState const Q_RET_SUPER_SUB = static_cast<QState>(1);
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! event unhandled due to a guard evaluating to FALSE
|
2013-10-10 20:01:51 -04:00
|
|
|
QState const Q_RET_UNHANDLED = static_cast<QState>(2);
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! event handled (internal transition)
|
|
|
|
QState const Q_RET_HANDLED = static_cast<QState>(3);
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! event silently ignored (bubbled up to top)
|
|
|
|
QState const Q_RET_IGNORED = static_cast<QState>(4);
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! state entry action executed
|
2013-10-10 20:01:51 -04:00
|
|
|
QState const Q_RET_ENTRY = static_cast<QState>(5);
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! state exit action executed
|
2013-10-10 20:01:51 -04:00
|
|
|
QState const Q_RET_EXIT = static_cast<QState>(6);
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! return value without any effect
|
|
|
|
QState const Q_RET_NULL = static_cast<QState>(7);
|
|
|
|
|
|
|
|
//! regular transition taken
|
|
|
|
QState const Q_RET_TRAN = static_cast<QState>(8);
|
|
|
|
|
|
|
|
//! initial transition taken
|
|
|
|
QState const Q_RET_TRAN_INIT = static_cast<QState>(9);
|
|
|
|
|
|
|
|
//! event handled (transition to history)
|
|
|
|
QState const Q_RET_TRAN_HIST = static_cast<QState>(10);
|
|
|
|
|
|
|
|
//! entry-point transition into a submachine
|
|
|
|
QState const Q_RET_TRAN_EP = static_cast<QState>(11);
|
|
|
|
|
|
|
|
//! exit-point transition out of a submachine
|
|
|
|
QState const Q_RET_TRAN_XP = static_cast<QState>(12);
|
|
|
|
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2013-10-10 20:01:51 -04:00
|
|
|
//****************************************************************************
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Meta State Machine
|
|
|
|
/// \description
|
2013-10-10 20:01:51 -04:00
|
|
|
/// QMsm represents the most fundamental State Machine in QP. The application-
|
|
|
|
/// level state machines derived directly from QMsm typically require the use
|
|
|
|
/// of the QM modeling tool, but are the fastest and need the least run-time
|
2014-04-13 21:35:34 -04:00
|
|
|
/// support (the smallest event-processor taking up the least code space).\n
|
|
|
|
/// \n
|
2013-10-10 20:01:51 -04:00
|
|
|
/// QMsm is also the base class for the QFsm and QHsm state machines, which
|
|
|
|
/// can be coded and maintained by hand (as well as by QM), but aren't as fast
|
|
|
|
/// and requrie significantly more run-time code (0.5-1KB) to execute.
|
2012-08-14 18:00:48 -04:00
|
|
|
///
|
2014-04-13 21:35:34 -04:00
|
|
|
/// \note
|
|
|
|
/// QMsm is not intended to be instantiated directly, but rather serves as
|
|
|
|
/// the base class for derivation of state machines in the application code.
|
2012-08-14 18:00:48 -04:00
|
|
|
///
|
2014-04-13 21:35:34 -04:00
|
|
|
/// \usage
|
2012-08-14 18:00:48 -04:00
|
|
|
/// The following example illustrates how to derive a state machine class
|
2013-10-10 20:01:51 -04:00
|
|
|
/// from QMsm. Please note that the QMsm member 'super' is defined as the
|
|
|
|
/// _first_ member of the derived struct.
|
|
|
|
/// \include qep_qmsm.cpp
|
|
|
|
class QMsm {
|
2014-04-13 21:35:34 -04:00
|
|
|
QMAttr m_state; //!< current active state (state-variable)
|
|
|
|
QMAttr m_temp; //!< temporary: transition chain, target state, etc.
|
2012-08-14 18:00:48 -04:00
|
|
|
|
|
|
|
public:
|
2014-04-13 21:35:34 -04:00
|
|
|
//! virtual destructor
|
2013-10-10 20:01:51 -04:00
|
|
|
virtual ~QMsm();
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Executes the top-most initial transition in QP::QMsm.
|
2013-10-10 20:01:51 -04:00
|
|
|
virtual void init(QEvt const * const e);
|
2013-12-30 17:41:15 -05:00
|
|
|
virtual void init(void) { this->init(static_cast<QEvt const *>(0)); }
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Dispatches an event to QMsm
|
2013-10-10 20:01:51 -04:00
|
|
|
virtual void dispatch(QEvt const * const e);
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Tests if a given state is part of the active state configuration
|
|
|
|
bool isInState(QMState const *state) const;
|
|
|
|
|
|
|
|
//! Return the current active state (state object)
|
|
|
|
QMState const *stateObj(void) const {
|
|
|
|
return m_state.obj;
|
|
|
|
}
|
|
|
|
|
2012-08-14 18:00:48 -04:00
|
|
|
protected:
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Protected constructor of QMsm.
|
|
|
|
QMsm(QStateHandler const initial);
|
|
|
|
|
|
|
|
//! internal helper function to record a regular state transition
|
|
|
|
QState qm_tran_(QMTranActTable const * const tatbl) {
|
|
|
|
m_temp.tatbl = tatbl;
|
|
|
|
return Q_RET_TRAN;
|
2012-08-14 18:00:48 -04:00
|
|
|
}
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Internal helper function to record a regular state transition
|
|
|
|
QState qm_tran_hist_(QMState const * const hist,
|
|
|
|
QMTranActTable const * const tatbl)
|
2013-10-10 20:01:51 -04:00
|
|
|
{
|
2014-04-13 21:35:34 -04:00
|
|
|
m_state.obj = hist;
|
|
|
|
m_temp.tatbl = tatbl;
|
|
|
|
return Q_RET_TRAN_HIST;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Internal helper function to record an initial state transition
|
|
|
|
QState qm_tran_init_(QMTranActTable const * const tatbl) {
|
|
|
|
m_temp.tatbl = tatbl;
|
|
|
|
return Q_RET_TRAN_INIT;
|
2013-10-10 20:01:51 -04:00
|
|
|
}
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Internal helper function to record an transition to an entry point
|
|
|
|
//! to a submachine state
|
|
|
|
QState qm_tran_ep_(QMTranActTable const * const tatbl) {
|
|
|
|
m_temp.tatbl = tatbl;
|
|
|
|
return Q_RET_TRAN_EP;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Internal helper function to record an transition to an exit point
|
|
|
|
//! from a submachine state
|
|
|
|
QState qm_tran_xp_(QActionHandler const xp,
|
|
|
|
QMTranActTable const *const tatbl)
|
2013-10-10 20:01:51 -04:00
|
|
|
{
|
2014-04-13 21:35:34 -04:00
|
|
|
m_state.act = xp;
|
|
|
|
m_temp.tatbl = tatbl;
|
|
|
|
return Q_RET_TRAN_XP;
|
2013-10-10 20:01:51 -04:00
|
|
|
}
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Internal helper function to record a state entry
|
2013-10-10 20:01:51 -04:00
|
|
|
QState qm_entry_(QMState const * const state) {
|
|
|
|
m_temp.obj = state;
|
2014-04-13 21:35:34 -04:00
|
|
|
return Q_RET_ENTRY;
|
2013-10-10 20:01:51 -04:00
|
|
|
}
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Internal helper function to record a state exit
|
2013-10-10 20:01:51 -04:00
|
|
|
QState qm_exit_(QMState const * const state) {
|
|
|
|
m_temp.obj = state;
|
2014-04-13 21:35:34 -04:00
|
|
|
return Q_RET_EXIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Internal helper function to call in a QM action-handler when
|
|
|
|
//! it passes the event to the host submachine state to handle an event.
|
|
|
|
QState qm_super_sub_(QMState const * const state) {
|
|
|
|
m_temp.obj = state;
|
|
|
|
return Q_RET_SUPER_SUB;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Internal helper function to call in a QM action-handler when
|
|
|
|
//! it handles an event.
|
|
|
|
static QState QM_HANDLED(void) {
|
|
|
|
return Q_RET_HANDLED;
|
2012-08-14 18:00:48 -04:00
|
|
|
}
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Macro to call in a QM action-handler when it does not handle
|
|
|
|
//! an event due to a guard condition evaluating to false.
|
|
|
|
static QState QM_UNHANDLED(void) {
|
|
|
|
return Q_RET_UNHANDLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Internal helper function to call in a QM action-handler when
|
|
|
|
//! it passes the event to the superstate for processing.
|
|
|
|
static QState QM_SUPER(void) {
|
|
|
|
return Q_RET_SUPER;
|
|
|
|
}
|
2013-10-10 20:01:51 -04:00
|
|
|
|
2013-12-30 17:41:15 -05:00
|
|
|
private:
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Internal helper function to execute a transition-action table
|
|
|
|
QState execTatbl_(QMTranActTable const * const tatbl);
|
|
|
|
|
|
|
|
//! Internal helper function to exit current state to transition source
|
|
|
|
void exitToTranSource_(QMState const *s, QMState const * const ts);
|
|
|
|
|
|
|
|
//! Internal helper function to enter state history
|
|
|
|
QState enterHistory_(QMState const * const hist);
|
|
|
|
|
|
|
|
//! maximum depth of implemented entry levels for transitions to history
|
|
|
|
static int_fast8_t const MAX_ENTRY_DEPTH_ = static_cast<int_fast8_t>(4);
|
2013-12-30 17:41:15 -05:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! the top state object for the QMsm
|
|
|
|
static QMState const msm_top_s;
|
|
|
|
|
2014-09-22 11:48:11 -04:00
|
|
|
friend class QFsm;
|
2013-10-10 20:01:51 -04:00
|
|
|
friend class QHsm;
|
|
|
|
friend class QMActive;
|
2012-08-14 18:00:48 -04:00
|
|
|
};
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Top-most state of QMSM is NULL
|
2013-10-10 20:01:51 -04:00
|
|
|
QMState * const QMsm_top = static_cast<QMState *>(0);
|
|
|
|
|
|
|
|
//****************************************************************************
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Hierarchical State Machine base class
|
2012-08-14 18:00:48 -04:00
|
|
|
///
|
2014-04-13 21:35:34 -04:00
|
|
|
/// \description
|
|
|
|
/// QHsm represents a Hierarchical State Machine (HSM) with full support for
|
|
|
|
/// hierarchical nesting of states, entry/exit actions, and initial
|
|
|
|
/// transitions in any composite state. QHsm inherits QMsm without adding
|
|
|
|
/// new attributes, so it takes the same amount of RAM as QMsm.
|
2012-08-14 18:00:48 -04:00
|
|
|
///
|
2014-04-13 21:35:34 -04:00
|
|
|
/// \note
|
|
|
|
/// QHsm is not intended to be instantiated directly, but rather serves as
|
|
|
|
/// the base class for derivation of state machines in the application code.
|
2012-08-14 18:00:48 -04:00
|
|
|
///
|
2014-04-13 21:35:34 -04:00
|
|
|
/// \usage
|
2012-08-14 18:00:48 -04:00
|
|
|
/// The following example illustrates how to derive a state machine class
|
|
|
|
/// from QHsm.
|
|
|
|
/// \include qep_qhsm.cpp
|
2013-10-10 20:01:51 -04:00
|
|
|
class QHsm : public QMsm {
|
2012-08-14 18:00:48 -04:00
|
|
|
public:
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Performs the second step of FSM initialization by triggering
|
2012-08-14 18:00:48 -04:00
|
|
|
/// the top-most initial transition.
|
2013-10-10 20:01:51 -04:00
|
|
|
virtual void init(QEvt const * const e);
|
2013-12-30 17:41:15 -05:00
|
|
|
virtual void init(void) { this->init(static_cast<QEvt const *>(0)); }
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Dispatches an event to a HSM
|
2013-10-10 20:01:51 -04:00
|
|
|
virtual void dispatch(QEvt const * const e);
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Tests if a given state is part of the current active state
|
|
|
|
//! configuration
|
2012-08-14 18:00:48 -04:00
|
|
|
bool isIn(QStateHandler const s);
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Return the current active state (state handler function)
|
2012-08-14 18:00:48 -04:00
|
|
|
QStateHandler state(void) const {
|
2013-10-10 20:01:51 -04:00
|
|
|
return m_state.fun;
|
2012-08-14 18:00:48 -04:00
|
|
|
}
|
|
|
|
|
2013-10-10 20:01:51 -04:00
|
|
|
protected:
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Protected constructor of a HSM.
|
2013-10-10 20:01:51 -04:00
|
|
|
QHsm(QStateHandler const initial);
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! the top-state.
|
2012-08-14 18:00:48 -04:00
|
|
|
static QState top(void * const me, QEvt const * const e);
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! internal helper function to specify the return of a state-handler
|
|
|
|
//! when it handles the event.
|
2013-10-10 20:01:51 -04:00
|
|
|
static QState Q_HANDLED(void) {
|
|
|
|
return Q_RET_HANDLED;
|
|
|
|
}
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! internal helper function to specify the return of a state-handler
|
|
|
|
//! function when it attempts to handle the event but a guard condition
|
|
|
|
//! evaluates to false and there is no other explicit way of handling
|
|
|
|
//! the event.
|
2013-10-10 20:01:51 -04:00
|
|
|
static QState Q_UNHANDLED(void) {
|
|
|
|
return Q_RET_UNHANDLED;
|
|
|
|
}
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! internal helper function to record a state transition
|
2013-10-10 20:01:51 -04:00
|
|
|
QState tran_(QStateHandler const target) {
|
|
|
|
m_temp.fun = target;
|
2012-08-14 18:00:48 -04:00
|
|
|
return Q_RET_TRAN;
|
|
|
|
}
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! internal helper function to record a transition to history
|
|
|
|
QState tran_hist_(QStateHandler const hist) {
|
|
|
|
m_temp.fun = hist;
|
|
|
|
return Q_RET_TRAN_HIST;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! internal helper function to record the superstate
|
2013-10-10 20:01:51 -04:00
|
|
|
QState super_(QStateHandler const superstate) {
|
|
|
|
m_temp.fun = superstate;
|
2012-08-14 18:00:48 -04:00
|
|
|
return Q_RET_SUPER;
|
|
|
|
}
|
|
|
|
|
|
|
|
enum ReservedHsmSignals {
|
2014-04-13 21:35:34 -04:00
|
|
|
Q_ENTRY_SIG = 1, //!< signal for entry actions
|
|
|
|
Q_EXIT_SIG, //!< signal for exit actions
|
|
|
|
Q_INIT_SIG //!< signal for nested initial transitions
|
2012-08-14 18:00:48 -04:00
|
|
|
};
|
2013-10-10 20:01:51 -04:00
|
|
|
|
2013-12-30 17:41:15 -05:00
|
|
|
private:
|
2014-04-13 21:35:34 -04:00
|
|
|
//! maximum depth of state nesting (including the top level), must be >= 3
|
|
|
|
static int_fast8_t const MAX_NEST_DEPTH_ = static_cast<int_fast8_t>(6);
|
2013-12-30 17:41:15 -05:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! internal helper function to take a transition
|
|
|
|
int_fast8_t hsm_tran(QStateHandler (&path)[MAX_NEST_DEPTH_]);
|
2013-12-30 17:41:15 -05:00
|
|
|
|
2013-10-10 20:01:51 -04:00
|
|
|
friend class QMActive;
|
|
|
|
};
|
|
|
|
|
|
|
|
//****************************************************************************
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Finite State Machine base class
|
2013-10-10 20:01:51 -04:00
|
|
|
///
|
2014-04-13 21:35:34 -04:00
|
|
|
/// \description
|
2013-10-10 20:01:51 -04:00
|
|
|
/// QFsm represents a traditional non-hierarchical Finite State Machine (FSM)
|
|
|
|
/// without state hierarchy, but with entry/exit actions.
|
|
|
|
///
|
|
|
|
/// \note QFsm is not intended to be instantiated directly, but rather serves
|
|
|
|
/// as the base class for derivation of state machines in the application
|
|
|
|
/// code.
|
|
|
|
///
|
2014-04-13 21:35:34 -04:00
|
|
|
/// \usage
|
2013-10-10 20:01:51 -04:00
|
|
|
/// The following example illustrates how to derive a state machine class
|
|
|
|
/// from QFsm.
|
|
|
|
/// \include qep_qfsm.cpp
|
|
|
|
class QFsm : public QMsm {
|
|
|
|
public:
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Performs the second step of FSM initialization by triggering
|
|
|
|
//! the top-most initial transition.
|
2013-10-10 20:01:51 -04:00
|
|
|
virtual void init(QEvt const * const e);
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Overloaded init operation (no initialization event)
|
2013-12-30 17:41:15 -05:00
|
|
|
virtual void init(void) { this->init(static_cast<QEvt const *>(0)); }
|
2013-10-10 20:01:51 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Dispatches an event to a FSM
|
2013-10-10 20:01:51 -04:00
|
|
|
virtual void dispatch(QEvt const * const e);
|
|
|
|
|
|
|
|
protected:
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Protected constructor of a FSM.
|
2013-10-10 20:01:51 -04:00
|
|
|
QFsm(QStateHandler const initial);
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Inline function to specify the return of a state-handler when it
|
|
|
|
//! handles the event.
|
2013-10-10 20:01:51 -04:00
|
|
|
static QState Q_HANDLED(void) {
|
|
|
|
return Q_RET_HANDLED;
|
|
|
|
}
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Helper function to specify the return of a state-handler function
|
|
|
|
//! when it attempts to handle the event but a guard condition evaluates
|
|
|
|
//! to false and there is no other explicit way of handling the event.
|
2013-10-10 20:01:51 -04:00
|
|
|
static QState Q_UNHANDLED(void) {
|
|
|
|
return Q_RET_UNHANDLED;
|
|
|
|
}
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Helper function to specify the return of a state-handler function
|
|
|
|
//! when it ignored the event (didn't attempt to handle it in any way).
|
2013-10-10 20:01:51 -04:00
|
|
|
static QState Q_IGNORED(void) {
|
|
|
|
return Q_RET_IGNORED;
|
|
|
|
}
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! internal helper function to record a state transition
|
2013-10-10 20:01:51 -04:00
|
|
|
QState tran_(QStateHandler const target) {
|
|
|
|
m_temp.fun = target;
|
|
|
|
return Q_RET_TRAN;
|
|
|
|
}
|
|
|
|
|
|
|
|
enum ReservedFsmSignals {
|
2014-04-13 21:35:34 -04:00
|
|
|
Q_ENTRY_SIG = 1, //!< signal for entry actions
|
|
|
|
Q_EXIT_SIG //!< signal for exit actions
|
2013-10-10 20:01:51 -04:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
//****************************************************************************
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Provides miscellaneous QEP services.
|
2013-10-10 20:01:51 -04:00
|
|
|
class QEP {
|
|
|
|
public:
|
2014-04-13 21:35:34 -04:00
|
|
|
//! get the current QEP version number string
|
|
|
|
/// \returns
|
|
|
|
/// version of QEP as a constant 5-character string of the
|
2013-10-10 20:01:51 -04:00
|
|
|
/// form X.Y.Z, where X is a 1-digit major version number, Y is a
|
|
|
|
/// 1-digit minor version number, and Z is a 1-digit release number.
|
2013-12-30 17:41:15 -05:00
|
|
|
static char_t const Q_ROM *getVersion(void) {
|
2013-10-10 20:01:51 -04:00
|
|
|
return QP_VERSION_STR;
|
|
|
|
}
|
2012-08-14 18:00:48 -04:00
|
|
|
};
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Offset or the user signals
|
2013-10-10 20:01:51 -04:00
|
|
|
enum_t const Q_USER_SIG = static_cast<enum_t>(4);
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
} // namespace QP
|
|
|
|
|
2013-10-10 20:01:51 -04:00
|
|
|
//****************************************************************************
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Perform cast to QStateHandler.
|
|
|
|
/// \description
|
2013-10-10 20:01:51 -04:00
|
|
|
/// This macro encapsulates the cast of a specific state handler function
|
|
|
|
/// pointer to QStateHandler, which violates MISRA-C 2004 rule 11.4(advisory).
|
|
|
|
/// This macro helps to localize this deviation.
|
|
|
|
#define Q_STATE_CAST(handler_) \
|
|
|
|
(reinterpret_cast<QP::QStateHandler>(handler_))
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Perform cast to QActionHandler.
|
|
|
|
/// \description
|
2013-10-10 20:01:51 -04:00
|
|
|
/// This macro encapsulates the cast of a specific action handler function
|
|
|
|
/// pointer to QActionHandler, which violates MISRA-C2004 rule 11.4(advisory).
|
|
|
|
/// This macro helps to localize this deviation.
|
|
|
|
#define Q_ACTION_CAST(action_) \
|
|
|
|
(reinterpret_cast<QP::QActionHandler>(action_))
|
|
|
|
|
|
|
|
#ifdef Q_SPY
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Macro to call in a QM action-handler when it executes an entry
|
|
|
|
//! action. Applicable only to QMSMs.
|
2013-10-10 20:01:51 -04:00
|
|
|
#define QM_ENTRY(state_) (me->qm_entry_((state_)))
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Macro to call in a QM action-handler when it executes an exit action.
|
|
|
|
//! Applicable only to QMSMs.
|
2013-10-10 20:01:51 -04:00
|
|
|
#define QM_EXIT(state_) (me->qm_exit_((state_)))
|
|
|
|
#else
|
|
|
|
#define QM_ENTRY(dummy) (QP::Q_RET_ENTRY)
|
|
|
|
#define QM_EXIT(dummy) (QP::Q_RET_EXIT)
|
|
|
|
#endif
|
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Macro to call in a QM state-handler when it executes a regular
|
|
|
|
//! transition. Applicable only to QMSMs.
|
|
|
|
#define QM_TRAN(tatbl_) \
|
|
|
|
(me->qm_tran_(reinterpret_cast<QP::QMTranActTable const *>(tatbl_)))
|
2013-10-10 20:01:51 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Macro to call in a QM state-handler when it executes an initial
|
|
|
|
//! transition. Applicable only to QMSMs.
|
|
|
|
#define QM_TRAN_INIT(tatbl_) \
|
|
|
|
(me->qm_tran_init_(reinterpret_cast<QP::QMTranActTable const *>(tatbl_)))
|
2013-10-10 20:01:51 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Macro to call in a QM state-handler when it executes a transition
|
|
|
|
//! to history. Applicable only to QMSMs.
|
|
|
|
#define QM_TRAN_HIST(history_, tatbl_) \
|
|
|
|
(me->qm_tran_hist_((history_), \
|
|
|
|
reinterpret_cast<QP::QMTranActTable const *>(tatbl_)))
|
2013-10-10 20:01:51 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Macro to call in a QM state-handler when it executes an initial
|
|
|
|
//! transition. Applicable only to QMSMs.
|
|
|
|
#define QM_TRAN_EP(tatbl_) \
|
|
|
|
(me->qm_tran_ep_(reinterpret_cast<QP::QMTranActTable const *>(tatbl_)))
|
2013-10-10 20:01:51 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Macro to call in a QM state-handler when it executes a transition
|
|
|
|
//! to exit point. Applicable only to QMSMs.
|
|
|
|
#define QM_TRAN_XP(xp_, tatbl_) \
|
|
|
|
(me->qm_tran_xp_((xp_), \
|
|
|
|
reinterpret_cast<QP::QMTranActTable const *>(tatbl_)))
|
2013-10-10 20:01:51 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Designates the superstate of a given state in an MSM.
|
|
|
|
#define QM_SUPER_SUB(state_) (me->qm_super_sub_((state_)))
|
|
|
|
|
|
|
|
|
|
|
|
//! Designates a target for an initial or regular transition.
|
|
|
|
//! Q_TRAN() can be used both in the FSMs and HSMs.
|
|
|
|
/// \usage
|
2013-10-10 20:01:51 -04:00
|
|
|
/// \include qep_qtran.cpp
|
|
|
|
#define Q_TRAN(target_) (me->tran_(Q_STATE_CAST(target_)))
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
//! Designates a target for an initial or regular transition.
|
|
|
|
//! Q_TRAN() can be used both in the FSMs and HSMs.
|
|
|
|
/// \usage
|
|
|
|
/// \include qep_qtran.cpp
|
|
|
|
#define Q_TRAN_HIST(hist_) (me->tran_hist_((hist_)))
|
|
|
|
|
|
|
|
//! Designates the superstate of a given state in an HSM.
|
|
|
|
/// \usage
|
2013-10-10 20:01:51 -04:00
|
|
|
/// \include qep_qhsm.cpp
|
|
|
|
#define Q_SUPER(state_) (me->super_(Q_STATE_CAST(state_)))
|
2012-08-14 18:00:48 -04:00
|
|
|
|
2014-04-13 21:35:34 -04:00
|
|
|
#endif // qep_h
|