/***************************************************************************** * Product: Reminder state pattern example * Last Updated for Version: 4.5.02 * Date of the Last Update: Jul 21, 2012 * * Q u a n t u m L e a P s * --------------------------- * innovating embedded systems * * Copyright (C) 2002-2012 Quantum Leaps, LLC. All rights reserved. * * 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 2 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 . * * Contact information: * Quantum Leaps Web sites: http://www.quantum-leaps.com * http://www.state-machine.com * e-mail: info@quantum-leaps.com *****************************************************************************/ #include "qp_port.h" #include "bsp.h" #include Q_DEFINE_THIS_FILE /*..........................................................................*/ enum SensorSignals { TIMEOUT_SIG = Q_USER_SIG, /* the periodic timeout signal */ DATA_READY_SIG, /* the invented reminder signal */ TERMINATE_SIG /* terminate the application */ }; /*..........................................................................*/ typedef struct SensorTag { /* the Sensor active object */ QActive super; /* derive from QActive */ QTimeEvt timeEvt; /* private time event generator */ uint16_t pollCtr; uint16_t procCtr; } Sensor; void Sensor_ctor(Sensor * const me); /* hierarchical state machine ... */ static QState Sensor_initial (Sensor * const me, QEvt const * const e); static QState Sensor_polling (Sensor * const me, QEvt const * const e); static QState Sensor_processing(Sensor * const me, QEvt const * const e); static QState Sensor_idle (Sensor * const me, QEvt const * const e); static QState Sensor_busy (Sensor * const me, QEvt const * const e); static QState Sensor_final (Sensor * const me, QEvt const * const e); /*..........................................................................*/ void Sensor_ctor(Sensor * const me) { QActive_ctor(&me->super, (QStateHandler)&Sensor_initial); QTimeEvt_ctor(&me->timeEvt, TIMEOUT_SIG); } /* HSM definition ----------------------------------------------------------*/ QState Sensor_initial(Sensor * const me, QEvt const * const e) { (void)e; /* avoid compiler warning about unused parameter */ me->pollCtr = 0; me->procCtr = 0; return Q_TRAN(&Sensor_polling); } /*..........................................................................*/ QState Sensor_final(Sensor * const me, QEvt const * const e) { QState status; switch (e->sig) { case Q_ENTRY_SIG: { printf("final-ENTRY;\nBye!Bye!\n"); QF_stop(); /* terminate the application */ status = Q_HANDLED(); break; } default: { status = Q_SUPER(&QHsm_top); break; } } return status; } /*..........................................................................*/ QState Sensor_polling(Sensor * const me, QEvt const * const e) { QState status; switch (e->sig) { case Q_ENTRY_SIG: { /* periodic timeout every 1/2 second */ QTimeEvt_postEvery(&me->timeEvt, (QActive *)me, BSP_TICKS_PER_SEC/2); status = Q_HANDLED(); break; } case Q_EXIT_SIG: { QTimeEvt_disarm(&me->timeEvt); status = Q_HANDLED(); break; } case Q_INIT_SIG: { status = Q_TRAN(&Sensor_processing); break; } case TIMEOUT_SIG: { /* NOTE: this constant event is statically pre-allocated. * It can be posted/published as any other event. */ static const QEvt reminderEvt = { DATA_READY_SIG, 0 }; ++me->pollCtr; printf("polling %3d\n", me->pollCtr); if ((me->pollCtr & 0x3) == 0) { /* modulo 4 */ QActive_postFIFO((QActive *)me, &reminderEvt); } status = Q_HANDLED(); break; } case TERMINATE_SIG: { status = Q_TRAN(&Sensor_final); break; } default: { status = Q_SUPER(&QHsm_top); break; } } return status; } /*..........................................................................*/ QState Sensor_processing(Sensor * const me, QEvt const * const e) { QState status; switch (e->sig) { case Q_INIT_SIG: { status = Q_TRAN(&Sensor_idle); break; } default: { status = Q_SUPER(&Sensor_polling); break; } } return status; } /*..........................................................................*/ QState Sensor_idle(Sensor * const me, QEvt const * const e) { QState status; switch (e->sig) { case Q_ENTRY_SIG: { printf("idle-ENTRY;\n"); status = Q_HANDLED(); break; } case DATA_READY_SIG: { status = Q_TRAN(&Sensor_busy); break; } default: { status = Q_SUPER(&Sensor_processing); break; } } return status; } /*..........................................................................*/ QState Sensor_busy(Sensor * const me, QEvt const * const e) { QState status; switch (e->sig) { case Q_ENTRY_SIG: { printf("busy-ENTRY;\n"); status = Q_HANDLED(); break; } case TIMEOUT_SIG: { ++me->procCtr; printf("processing %3d\n", me->procCtr); if ((me->procCtr & 0x1) == 0) { /* modulo 2 */ status = Q_TRAN(&Sensor_idle); } else { status = Q_HANDLED(); } break; } default: { status = Q_SUPER(&Sensor_processing); break; } } return status; } /* test harness ============================================================*/ /* Local-scope objects -----------------------------------------------------*/ static Sensor l_sensor; /* the Sensor active object */ static QEvt const *l_sensorQSto[10]; /* Event queue storage for Sensor */ /*..........................................................................*/ int main(int argc, char *argv[]) { printf("Reminder state pattern\nQEP version: %s\nQF version: %s\n" "Press ESC to quit...\n", QEP_getVersion(), QF_getVersion()); BSP_init(argc, argv); /* initialize the BSP */ QF_init(); /* initialize the framework and the underlying RT kernel */ /* publish-subscribe not used, no call to QF_psInit() */ /* dynamic event allocation not used, no call to QF_poolInit() */ /* instantiate and start the active objects... */ Sensor_ctor(&l_sensor); QActive_start((QActive *)&l_sensor, 1, l_sensorQSto, Q_DIM(l_sensorQSto), (void *)0, 1024, (QEvt *)0); return QF_run(); /* run the QF application */ } /*..........................................................................*/ void BSP_onKeyboardInput(uint8_t key) { switch (key) { case '\033': { /* ESC pressed? */ /* NOTE: this constant event is statically pre-allocated. * It can be posted/published as any other event. */ static QEvt const terminateEvt = { TERMINATE_SIG, 0}; QActive_postFIFO((QActive *)&l_sensor, &terminateEvt); break; } } }