2012-08-14 18:07:04 -04:00
|
|
|
/*****************************************************************************
|
|
|
|
* Product: Time Bomb Example with QEP
|
|
|
|
* Last Updated for Version: 4.5.00
|
|
|
|
* Date of the Last Update: May 18, 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
|
2013-10-16 16:44:03 -04:00
|
|
|
* by the Free Software Foundation, either version 3 of the License, or
|
2012-08-14 18:07:04 -04:00
|
|
|
* (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:
|
|
|
|
* Quantum Leaps Web sites: http://www.quantum-leaps.com
|
|
|
|
* http://www.state-machine.com
|
|
|
|
* e-mail: info@quantum-leaps.com
|
|
|
|
*****************************************************************************/
|
|
|
|
#include "qep_port.h" /* the port of the QEP event processor */
|
|
|
|
#include "bsp.h" /* board support package */
|
|
|
|
|
|
|
|
enum BombSignals { /* all signals for the Bomb FSM */
|
|
|
|
UP_SIG = Q_USER_SIG,
|
|
|
|
DOWN_SIG,
|
|
|
|
ARM_SIG,
|
|
|
|
TICK_SIG
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct TickEvtTag {
|
|
|
|
QEvt super; /* derive from the QEvt structure */
|
|
|
|
uint8_t fine_time; /* the fine 1/10 s counter */
|
|
|
|
} TickEvt;
|
|
|
|
|
|
|
|
typedef struct Bomb4Tag {
|
|
|
|
QFsm super; /* derive from QFsm */
|
|
|
|
uint8_t timeout; /* number of seconds till explosion */
|
|
|
|
uint8_t code; /* currently entered code to disarm the bomb */
|
|
|
|
uint8_t defuse; /* secret defuse code to disarm the bomb */
|
|
|
|
} Bomb4;
|
|
|
|
|
|
|
|
void Bomb4_ctor(Bomb4 *me, uint8_t defuse);
|
|
|
|
|
|
|
|
static QState Bomb4_initial(Bomb4 *me, QEvt const *e);
|
|
|
|
static QState Bomb4_setting(Bomb4 *me, QEvt const *e);
|
|
|
|
static QState Bomb4_timing (Bomb4 *me, QEvt const *e);
|
|
|
|
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/* the initial value of the timeout */
|
|
|
|
#define INIT_TIMEOUT 10
|
|
|
|
|
|
|
|
/*..........................................................................*/
|
|
|
|
void Bomb4_ctor(Bomb4 *me, uint8_t defuse) {
|
|
|
|
QFsm_ctor(&me->super, (QStateHandler)&Bomb4_initial);/* superclass ctor */
|
|
|
|
me->defuse = defuse; /* the defuse code is assigned at instantiation */
|
|
|
|
}
|
|
|
|
/*..........................................................................*/
|
|
|
|
QState Bomb4_initial(Bomb4 *me, QEvt const *e) {
|
|
|
|
(void)e;
|
|
|
|
me->timeout = INIT_TIMEOUT;
|
|
|
|
return Q_TRAN(&Bomb4_setting);
|
|
|
|
}
|
|
|
|
/*..........................................................................*/
|
|
|
|
QState Bomb4_setting(Bomb4 *me, QEvt const *e) {
|
|
|
|
switch (e->sig) {
|
|
|
|
case UP_SIG: {
|
|
|
|
if (me->timeout < 60) {
|
|
|
|
++me->timeout;
|
|
|
|
BSP_display(me->timeout);
|
|
|
|
}
|
|
|
|
return Q_HANDLED();
|
|
|
|
}
|
|
|
|
case DOWN_SIG: {
|
|
|
|
if (me->timeout > 1) {
|
|
|
|
--me->timeout;
|
|
|
|
BSP_display(me->timeout);
|
|
|
|
}
|
|
|
|
return Q_HANDLED();
|
|
|
|
}
|
|
|
|
case ARM_SIG: {
|
|
|
|
return Q_TRAN(&Bomb4_timing); /* transition to "timing" */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Q_IGNORED();
|
|
|
|
}
|
|
|
|
/*..........................................................................*/
|
|
|
|
QState Bomb4_timing(Bomb4 *me, QEvt const *e) {
|
|
|
|
switch (e->sig) {
|
|
|
|
case Q_ENTRY_SIG: {
|
|
|
|
me->code = 0; /* clear the defuse code */
|
|
|
|
return Q_HANDLED();
|
|
|
|
}
|
|
|
|
case UP_SIG: {
|
|
|
|
me->code <<= 1;
|
|
|
|
me->code |= 1;
|
|
|
|
return Q_HANDLED();
|
|
|
|
}
|
|
|
|
case DOWN_SIG: {
|
|
|
|
me->code <<= 1;
|
|
|
|
return Q_HANDLED();
|
|
|
|
}
|
|
|
|
case ARM_SIG: {
|
|
|
|
if (me->code == me->defuse) {
|
|
|
|
return Q_TRAN(&Bomb4_setting);
|
|
|
|
}
|
|
|
|
return Q_HANDLED();
|
|
|
|
}
|
|
|
|
case TICK_SIG: {
|
|
|
|
if (((TickEvt const *)e)->fine_time == 0) {
|
|
|
|
--me->timeout;
|
|
|
|
BSP_display(me->timeout);
|
|
|
|
if (me->timeout == 0) {
|
|
|
|
BSP_boom(); /* destroy the bomb */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Q_HANDLED();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Q_IGNORED();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Test harness ------------------------------------------------------------*/
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <conio.h>
|
|
|
|
#include <dos.h>
|
|
|
|
#include <stdlib.h> /* for _exit() */
|
|
|
|
|
|
|
|
static Bomb4 l_bomb; /* time bomb FSM */
|
|
|
|
|
|
|
|
/*..........................................................................*/
|
|
|
|
int main() {
|
|
|
|
|
|
|
|
Bomb4_ctor(&l_bomb, 0x0D); /* the secret defuse code, 1101 binary */
|
|
|
|
|
|
|
|
printf("Time Bomb (QEP QFsm class)\n"
|
|
|
|
"Press 'u' for UP event\n"
|
|
|
|
"Press 'd' for DOWN event\n"
|
|
|
|
"Press 'a' for ARM event\n"
|
|
|
|
"Press <Esc> to quit.\n");
|
|
|
|
|
|
|
|
QFsm_init((QFsm *)&l_bomb, (QEvt *)0); /* take the initial transition */
|
|
|
|
|
|
|
|
for (;;) { /* event loop */
|
|
|
|
static TickEvt tick_evt = { TICK_SIG, 0, 0, 0 };
|
|
|
|
|
|
|
|
delay(100); /* 100 ms delay */
|
|
|
|
|
|
|
|
if (++tick_evt.fine_time == 10) {
|
|
|
|
tick_evt.fine_time = 0;
|
|
|
|
}
|
|
|
|
printf("T(%1d)%c", tick_evt.fine_time,
|
|
|
|
(tick_evt.fine_time == 0) ? '\n' : ' ');
|
|
|
|
|
|
|
|
QFsm_dispatch((QFsm *)&l_bomb, (QEvt *)&tick_evt);
|
|
|
|
|
|
|
|
if (kbhit()) {
|
|
|
|
static QEvt const up_evt = { UP_SIG, 0, 0 };
|
|
|
|
static QEvt const down_evt = { DOWN_SIG, 0, 0 };
|
|
|
|
static QEvt const arm_evt = { ARM_SIG, 0, 0 };
|
|
|
|
QEvt const *e = (QEvt *)0;
|
|
|
|
|
|
|
|
switch (getch()) {
|
|
|
|
case 'u': { /* UP event */
|
|
|
|
printf("\nUP : ");
|
|
|
|
e = &up_evt; /* generate the UP event */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'd': { /* DOWN event */
|
|
|
|
printf("\nDOWN: ");
|
|
|
|
e = &down_evt; /* generate the DOWN event */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'a': { /* ARM event */
|
|
|
|
printf("\nARM : ");
|
|
|
|
e = &arm_evt; /* generate the ARM event */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case '\33': { /* <Esc> key */
|
|
|
|
printf("\nESC : Bye! Bye!");
|
|
|
|
fflush(stdout);
|
|
|
|
_exit(0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (e != (QEvt *)0) { /* keyboard event available? */
|
|
|
|
QFsm_dispatch((QFsm *)&l_bomb, e); /* dispatch the event */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|