Engine is designed to execute sophisticated, hierarchical state machines with guards. State machines are assembled from individual parts, parsed from a machine definition file written in Engine Machine Language.
Tool, the integrated compiler, transforms these Engine Machine Language files into byte code, ready for execution by Engine. Fast, efficient, and lightweight, Tool is perfect for just-in-time compilation, minimizing overhead and seamlessly integrating with your application.
Engine with Tool is tailor-made for embedded C or C++ applications, providing a powerful and efficient solution for real-time state machine management, ensuring optimal performance with a nostalgic twist.
You might be surprised to hear that Engine was designed for use with deeply embedded systems. To better understand how this works, let's consider a simple example, the proverbial "Toaster Oven":
The implementation of the "Toaster Oven" in this case is fairly straightforward. The "Toaster Oven" has two events that can trigger changes in its state, "On/Off" and "Door." These events turn the oven on or off and open or close the door, depending on the current state of the toaster. The timer does not paused when the door is opened and always restart when transitioning from the Toaster_off to the Toaster_on state.
The "toaster part" includes two actions, toaster_lamp and toaster_heater, which are written in C. For this simple demonstration, these functions simply print messages indicating that the toaster heater or lamp was turned on or off. In a real embedded application, the actual work would be performed here. With the "toaster part" implemented, we can fully define the behavior of the "Toaster Oven" using only Engine Machine Language code.
With a syntax similar to C, C++, or most other programming languages, those with programming experience should have no difficulty reading the toaster's implementation in Engine Machine Language. There are only a few key words, like enter, exit and action which calls actions/functions with an optional parameter. The key word event will initiate a transition to the next state for the specific event. But let's look at the implementation of the "Toaster Oven" before we continue.
The "Toaster Oven" Engine Machine Language code:
decl_name "Toaster Oven"
decl_version 2
decl_variables {
Timer = [timer.default] /* Reload value for the "off" timer when the toaster is started.
The value "timer.default" is read from a registry or config file. */
}
/* Actions can dispatch evets into state machines, their own or globally
to all state machines. We declare a few events here fo use by the
state machines. */
decl_events {
_evt_Door
_evt_OnOff
}
statemachine Toaster_oven {
/* On startup, transition to Toaster_off. It is
assumed the door is closed. */
startstate Toaster_off
state Door_closed {
/* Transtioning from Door_open to any of the Door_closed
substates will call the entry actions defined here. */
enter (toaster_lamp, OFF)
/* For any of the Door_closed substates the Door event
will transition to the Door_open state. */
event (_evt_Door, Door_opened)
/* When the door is no longer closed, the lamp is turned on. */
exit (toaster_lamp, ON)
}
super Door_closed {
/* All states in this scope takes Door_closed as
super state. Entry and Exit actions during state
state transitions will happen according to the UML
specification. Super states can be nested. */
state Toaster_off {
/* Reset the timer. */
enter (state_timer1_sec, 0)
/* Set the timer. */
action (_evt_OnOff, state_timer1_sec, [Timer])
/* Transition to Toaster_on. */
event (_evt_OnOff, Toaster_on)
}
state Toaster_on {
/* On entry, turn the heater on. */
enter (toaster_heater, ON)
/* When the timer expires, transition to Toaster_off. */
event (_state_timer1, Toaster_off)
/* On the OnOff event, transition to Toaster_off. */
event (_evt_OnOff, Toaster_off)
/* On exit, turn the heater off. */
exit (toaster_heater, OFF)
}
}
state Door_opened {
/* When the door is closed, check if the timer is active
and load the result into the accumulator. */
action_ld (_evt_Door, [a], state_timer1_active)
/* If the accumulator is set, transition to Toaster_on. */
event_if (_evt_Door, Toaster_on)
/* If the accumulator is clear, transition to Toaster_off. */
event_nt (_evt_Door, Toaster_off)
}
}
The code snippet above doesn't include the Toaster_controller state machine, which runs in parralel, which implements a controller interface for the toaster. This interface generates the _evt_OnOff and _evt_Door events and allows you to change the toaster's settings such as the timer duration. It displays a menu for these options and prints out the current settings of the toaster. You can find the complete Engine Machine Language code for the "Toaster Oven" example in the toaster.e file.
First, Engine with Tool must be compiled from the sources in the repository. To compile and run the project in a codespace, just start a codespace from the repository using the <> code
button in the code view of the repository. When the codespace is open in the browser, use the following commands in the terminal that opened in the browser to compile and run the project (if the terminal is not open use ctrl + `
to open the terminal):
make
now start the engine to run the toaster with the folloing command:
./build/engine ./test/toaster.e
💡 Use the --help option to display the command line syntax:
./build/engine --help
When you start the "toaster.e" machine, you will be presented with a menu.
The Toaster_controller menu:
'My Toaster' settings:
- Timer: 10s
- Log Level: ALL
- Log Output: ENABLED
Control menu:
[t] Toaster On/Off.
[d] Door Open/Close.
[s] Settings.
[h] Help.
Engine provides extensive logging for debugging state machines. As shown in the menu above, the log level is set to ALL. So lets dispatch a few commands and look at the output. Type "tdd" in the console and press enter. This will turn the toaster on (t) and open the door (d) and close the door again (d). The debug log below should be generated:
💡 In the following debug log, 0x700 is the "Door" event and 0x701 is the "OnOff" event. The text preceded by the
####################
is debug output geberated the actions itself and not the state machine. All other logging is generated by Engine.
The Test Drive log:
562.798: [evt] ---* 0x701 ([a] 1, [e] 0)
562.800: [act] state_timer1_sec [5] 10
562.803: [trn] ---> 'Toaster_off' to 'Toaster_on' () ([a] 1, [r] 0)
562.806: [ext] ---- exit actions (Toaster_off):
562.808: [ent] ---- entry actions (Toaster_on):
562.810: [ent] toaster_heater , 1
562.811: #################### HEATER ON
562.812: [evt] ---* _state_start ([a] 1, [e] 0)
562.817: [evt] ---* 0x700 ([a] 1, [e] 0)
562.819: [trn] ---> 'Toaster_on' to 'Door_opened' () ([a] 1, [r] 0)
562.823: [ext] ---- exit actions (Toaster_on):
562.825: [ext] toaster_heater , 0
562.827: #################### HEATER OFF
562.830: [ext] ---- exit actions (Door_closed):
562.832: [ext] toaster_lamp , 1
562.834: #################### LAMP ON
562.835: [ent] ---- entry actions (Door_opened):
562.837: [evt] ---* _state_start ([a] 1, [e] 0)
562.839: [evt] ---* 0x700 ([a] 1, [e] 0)
562.840: [act] state_timer1_active (ld [0] 1) 0
562.842: [trn] ---> 'Door_opened' to 'Toaster_on' (if) ([a] 1, [r] 0)
562.845: [ext] ---- exit actions (Door_opened):
562.847: [ent] ---- entry actions (Door_closed):
562.848: [ent] toaster_lamp , 0
562.850: #################### LAMP OFF
562.851: [ent] ---- entry actions (Toaster_on):
562.853: [ent] toaster_heater , 1
562.855: #################### HEATER ON
562.857: [evt] ---* _state_start ([a] 1, [e] 0)
572.806: [evt] ---* _state_timer1 ([a] 1, [e] 0)
572.809: [trn] ---> 'Toaster_on' to 'Toaster_off' () ([a] 1, [r] 0)
572.817: [ext] ---- exit actions (Toaster_on):
572.820: [ext] toaster_heater , 0
572.822: #################### HEATER OFF
572.824: [ent] ---- entry actions (Toaster_off):
572.827: [ent] state_timer1_sec , 0
572.828: [evt] ---* _state_start ([a] 1, [e] 0)
The log displays the time in seconds on the left followed by the action as described in the following table:
Header | Description |
---|---|
[ent] |
Entry action. |
[act] |
Internal transitions where an event has trigering an action. |
[ext] |
Exit action. |
[evt] |
Event. If it is a built in event from a part the text will be displayed. Events declared in the "decl_events" section start from 0x700. |
[trn] |
A transition was triggered by the previous event. This will be followed by all the exit and entry actions for all the states and super states as defined by the UML specification. |
####### |
Debug output printed by the toaster part |
According to the log, the toaster turns on and the door opens shortly thereafter. When the door is closed, the toaster returns to 'Toaster_on' state because the timer has not yet expired, as expected.
Additional logging output shows the different registers implemented that can be used by guards for conditional execution of actions or transitions. Registers, as well as variables and indexed variables, are referenced using square brackets for example [a] for the "accumulator", [r] for the "register" and [e] for the "event register". But more about this later.
💡 "toaster.e" requires a configuration file named "toaster.cfg" to be located in the same directory. Alternatively, the location of the configuration file can be specified on the command line, if it has a different name.
The example below features a small OLED display used in a real-world application, a battery-powered Wi-Fi toaster with a three-button capacitive touch input. The video demonstrates the display and how to change the timer on the toaster.
😉 I mean, I'm pretty sure there's no actual battery-powered Wi-Fi toaster out there in the world... or at least, I hope not. That would be way too much technology for a humble toaster. This was just an excuse to show off the awesomeness of Engine!
Before we delve into the syntax of Engine Machine Language, let's first consider the end result, the byte code that is produced when Tool compiles it. With this "end in mind," we can better understand the inner workings of Tool as it translates Engine Machine Language into byte code.
Tool includes a lexer and a parser which work together to create byte code for machines that can be executed by the Engine. This byte code is position-independent, meaning it can also be pre-compiled and linked as a resource to be executed from any memory if desired.
Each state machine is created in a contiguous block of memory where the state machine header is followed by a indexed lookup table containing the offsets to the individual states. A state has a header followed by the state data. State data consists of an array of action or event identifiers followed by a parameter. Depending on the context, the parameter will be interpreted differently. The following image illustrates the memory layout of a simple state machine with two states:
💡 External transitions are shown with solid lines while the internal transitions are shown with dotted lines.
In this example, each state has one entry and exit action, one internal transition and one external transition, but the total number of events, deferred events, entry actions, exit actions and actions (internal transitions) can be up to 255 each.
The state header contains the number of events, deferred events, entry actions, exit actions and actions (internal transitions) in the state data.
Actions can be entry or exit actions or actions of internal transitions.
Param is a 16 bit integer but should be interpreted by the action according to thhe TYP bits.
Bits | Description |
---|---|
Bit_31:30 | Type: 0 - constant 1 - indexed (string index into a registry. This index must be loaded from the string table) 2 - string (must be loaded from the string table) 3 - variable (an identifier in square brackets. This is a 32 bit integer) |
Bit_29:28 | Operation to perform before or after the action: 0 - nothing. 1 - push (acumulator pushed and result save in accumulator) 2 - pop (pop accumulator before the action) 3 - save (result saved in the register) |
The following bits depict how the event bits are interpreted for an internal transition.
Bits | Description |
---|---|
Bit_31 | If set, the value in the comparator is a variable, otherwise it is a constant |
Bit_30:28 | How the comparator is evaluated to determin if the action should be executed: 1 - If event register equal to the comparator. 2 - If the comparator less than the accumulator. 3 - If the comparator greater than the accumulator. 4 - If the comparator is equal to the accumulator. 5 - If the comparator is not equal to the accumulator. 6 - Will load the result of the action into the variable in the comparator. |
Bit_27 | If set, this will terminate the evaluation of the event for further actions. |
Bit_26:16 | Event id to identify if the action that follow in the next 32 bits will be executed. |
Bit_15:0 | Comparator to determine of the action should be executed. This could be a constant or a variable |
The event bits are interpreted differently for an internal transition than for an external transition. The following bits depict an external transition.
Bits | Description |
---|---|
Bit_31 | Previous pin. If set it will toggle pushing the previous state stack. Transitioning to the PREVIOUS state will always be the state where this bit was set. |
Bit_30:28 | Guard condition for the event to trigger a transition: 1 - If accumulator set. 2 - If accumulator NOT set. 3 - If register is set. 4 - If register NOT set. |
Bit_26:16 | Event id to identify if the transition to the next state should occur. |
Bit_15:0 | Next state index. |
Deferred events are saved until after the next transition.
Bits | Description |
---|---|
Bit_26:16 | Event id for events that should be deferred. |
All strings in the machine definition file is copied into the stringtable.
Engine references strings through their index in the stringtable. A string has a 16 bit ID and a 16 bit length field followed by the actual string.
Tool is a compiler taking Engine Machine Language and compiles it to byte code that is executed by Engine.
A machine definition consists of declarations and one or more state machines. A state machine is made up of one or more states that can be organised into super-states and sub-states.
Machine Definition:
<declarations>
statemachine <name_1> {
startstate <state_s11>
state <state_s1> {
}
super <state_s1> {
state <state_s11> {
}
state <state_s12> {
}
super <state_s12> {
state <state_s121> {
}
state <state_s122> {
}
}
}
state <state_s2> {
}
...
...
}
...
...
statemachine <name_x> {
...
...
}
The following declarations can be done outside the scope of the state machines.
💡 All declarations are optional.
Name | Description |
---|---|
decl_name |
Description: Set the name for the assembly of state machines in the machine definition file. |
decl_version |
Set the version for the assembly of state machines in the machine definition file. |
decl_variables |
Declares a list of initialised variables. |
decl_events |
Declares events that can be used as a parameter for an action. |
decl_startup |
Declares a list of initialization shell commands. |
After the declarations one or more state machines can be defined. All states inside of the superstate scope will have the specific state in the "super" as super state. Super states can nest up to defined maximum.
As many as the defined maximum state machines can be declared. Events will be dispatched to the state machines in the order they are declared. Processing of the event is complete once it has been dispatched to all state machines. Only one event will be active at a time.
State Definition:
state <name> {
enter (<action>[op], [param])
action (<event>[op], <action>[op], [param])
action_xx (<event>[op], <comparator>, <action>[op], [param])
event (<event>[op], <state>)
event_xx (<event>[op], <state>)
deferred (<event>)
exit (<action>[op], [param])
}
Declarations inside a state, such as events, actions, etc., can be placed in any order and may be mixed together. However, they will be evaluated in the order in which they were declared.
After transitioning to a state, the entry actions will be executed starting with the least common ancestor (LCA) superstate and progressing down to the current state
If an event is dispatched, the following actions will be taken in the following order:
- Deferred events will be saved for later processing. If the dispatched event was deferred, the event processing will be terminated.
- Actions associated with internal transitions, if defined for the event, will be executed based on a guard. THis will start with the current state and progressing up to the top-most superstate. It is possible for zero or more actions to be executed for a single event. If the termination flag is set, the evaluation of the event for further actions in the current state and superstates will be terminated after the action has been executed.
- Finally, the evaluation process for potential transitions will begin at the current state and progress up to the superstate. An event with a guard is able to trigger a transition. If the resulting transition leads to the IGNORE state, the evaluation of the event for transitions will end and no transition will take place.
If a transition is triggered, exit actions will be executed starting with the current state and progressing up to the LCA superstate.
Parameters may be simple constants with a 16-bit integer value, but registers or variables, which are 32-bit integer values passed to the C implementation of the action, can also be used. Registers and variables are denoted in square brackets.
Register | Description |
---|---|
[a] |
Accumulator. Can be used for arithmetic operations. Can also be pushed and popped from a stack. This is local for each instance of an Engine. |
[e] |
Event register. Can b e used for arithmetic operations. Can also be pushed and popped from a stack. This is local for each instance of an Engine. |
[r] |
General purpose register. It can be used for some guarded transitions. This is local for each instance of an Engine. |
[p] |
General purpose register. By convention used as a second parameter for an action. This is local for each instance of an Engine. |
[<variable>] |
Any variables declared in the "decl_variables" section is global to all instances of Engines. |
[<registry>] |
A string that can be used for registry lookup if implemented by the port. |
For the declarations of the action or events in the state "_xx" demotes guard conditions. For guard conditions, a register is compared to a "comparator" which can be a constant, variable or register. The following guards are defined:
Guard | Description |
---|---|
action |
No guard, the action will always execute. |
action_eq_e |
The action will execute if the event register is equal to the comparator. |
action_lt |
The action will execute if the accumulator is less than the comparator. |
action_gt |
The action will execute if the accumulator is greater than the comparator. |
action_eq |
The action will execute if the accumulator is equal to the comparator. |
action_ne |
The action will execute if the accumulator is not equal to the comparator. |
action_ld |
This is not a guard but the result of the action will be loaded into the comparator if it is a register or a variable. |
event |
No guard. The transition will always trigger |
event_if |
The transition will trigger if the accumulator is set. |
event_if |
The transition will trigger if the accumulator is clear. |
event_if_r |
The transition will trigger if the register is set. |
event_nt_r |
The transition will trigger if the register is clear. |
Optional operators ([op]) can be used with actions and events. The following operators are defined:
Op | Use_With | Name | Description |
---|---|---|---|
+ |
action | Push | After the action was executed, the accumulator will be pushed and the result of the action will be saved in the accumulator. |
- |
action | Pop | Before the action is executed, the accumulator will be popped from the stack. |
= |
action | Save | After the action was executed, the result of the action will be saved in the register. |
^ |
event | Pin | State transitions will no longer be pushed on to the previous state stack. The next transition to the "PREVIOUS" state will transition to this state. On a previous transition the previsous pin will be cleared". |
! |
event | Terminate | No more actions will be executed for this event. |
Engine can be extended with new parts. There are three things a part can export to use from Engine Machina Language, these are: actions, events and constants.
Using the demo application you can list al the part's actions, events and constants using the "--list" command line option.
Demo application actions, events and constants:
Actions:
a_add [a] += parm. return [a].
a_and [a] = [a] && parm. return [a]
a_dec [a]-- (down to parm). return [a].
a_div [a] /= parm. return [a].
a_eq return [a] == parm.
a_get return [a].
a_gt return [a] > parm.
a_inc [a]++ (up to parm). return [a].
a_load [a] = parm. return [a]
a_lt return [a] < parm.
a_mod [a] %= parm. return [a].
a_mov [r] = [a]. return [a]
a_mult [a] *= parm. return [a].
a_not [a] = ![a]. return [a].
a_or [a] = [a] || parm. return [a]
a_pop Pop accumulator.
a_push Push accumulator.
a_sub [a] -= parm. return [a].
a_swap Swap accumulator.
console_events_register Register "statemachine" to receive console events
console_write Write a 'character' the console.
console_writeln Write a "line" to the console.
debug_dump Dump the current state of all state machines.
debug_log_level Set the log level. To be used with the LOG_XXX constants.
debug_log_output ENABLE/DISABLE output logging
debug_log_statemachine A space separated list of statemachine to enable logging for.
e_eq return [e] == parm.
e_gt return [e] > parm.
e_lt return [e] < parm.
get Load and return the value.
nop No Operation
p_add [p] += parm. return [p]
p_load [p] = parm. return [p]
r_clear [r] = 0. retutn [r].
r_inc [r]++. return [r].
r_load [r] = parm. return [r].
r_set [r] = 1. retutn [r].
rand Return rand value % parm.
state_event Fire the event to all state machines
state_event_if Fire the event to all state machines if accumulator set
state_event_local Fire the event to this state machine only
state_event_local_if Fire the event to this state machine only if accumulator set
state_event_local_not Fire the event to this state machine only if accumulator clear
state_event_not Fire the event to all state machines if accumulator clear
state_keepalive1 Set state keep-alive timer (autorepeat milliseconds)
state_keepalive1_sec Set state keep-alive timer (autorepeat seconds)
state_keepalive2 Set state keep-alive timer (autorepeat milliseconds)
state_keepalive2_sec Set state keep-alive timer (autorepeat seconds)
state_timeout Set state timeout timer (milliseconds) (cancelled on the first transition)
state_timeout_sec Set state timeout timer (seconds) (cancelled on the first transition)
state_timer1 Set state timer 1 (milliseconds)
state_timer1_active Return TRUE if timer active
state_timer1_sec Set state timer 1 (seconds)
state_timer2 Set state timer 2 (milliseconds)
state_timer2_active Return TRUE if timer active
state_timer2_sec Set state timer 2 (seconds)
strlen Return the string length.
toaster_heater Turn the heater ON/OFF.
toaster_lamp Turn the lamp ON/OFF.
Events:
_console_char A character was received in [e].
_state_keepalive1 Event for state_keepalive1()
_state_keepalive2 Event for state_keepalive2()
_state_start State start (always the first event after a transition).
_state_timeout Event for state_timeout()
_state_timer1 Event for state_timer1()
_state_timer2 Event for state_timer2()
_toaster_smoke_alert Toaster smoak alert event.
Constatnts:
CURRENT (-3) Current State (use instead of a state name for an event transition)
DISABLE (0) Disable
ENABLE (1) Enable
FALSE (0) False
IGNORE (-4) Ignore State (use instead of a state name for an event transition)
INVALID (-1) Invalid State (use instead of a state name to for an event transition)
LOG_ALL (2046) Log level ALL
LOG_NONE (0) NO logging
LOG_REPORT (6) Log level REPORT
LOG_TRANSITIONS (70) Log TRANSITIONS only
OFF (0) Turn heater/lamp OFF
ON (1) Turn heater/lamp ON
PREVIOUS (-2) Previous State (use instead of a state name for an event transition)
TRUE (1) True
VAL_MAX (32767) Maximum for constant value
VAL_MIN (-32768) Minimum for constant value
The name on the left is the name as it would be used from an Engine Machine Language file.
The C action prototype looks like this:
int32_t action_part_action (PENGINE_T instance, uint32_t parm, uint32_t flags) ;
The instance argument is the instance of the engine, or state machine this action is called from. It can be used with the engine API defined in"engine.h".
The parm argument is passed from the statemhine and should be interpreted according to the flags.
The flags are defined in parts.h and can be one of the following:
Flag | Description |
---|---|
PART_ACTION_FLAG_VALIDATE | Only validate the parm and return ENGINE_OK if it is valid for the action |
PART_ACTION_FLAG_INDEXED | This is an indexed variable and parm should be used to do the lookup in the registry to load a value. The value can be anything the registry supports. The Tool compiler uses parm as an string table index for use with a registry with string keys, so a lookup in the string table should be done first. |
PART_ACTION_FLAG_STRING | Parm should be used to get the string from the strin table using the API exported in "engine.h" |
PART_ACTION_FLAG_VARIABLE | Parm is a register or variable. Parm is already loaded with the value. |
To export this action for use with the Tool compiler also add the following declaration:
ENGINE_ACTION_IMPL (part_action, "Example action.") ;
Now in the machine definition file this new action can be used like this, for example performing an entry action for a state:
enter (part_action, TRUE)
Adding a event can be done with a single declaration in the C code of the part:
ENGINE_EVENT_IMPL (_part_event, "Example event") ;
From the C code of the part, the event can be dispatched to one or more state machines using the following code:
engine_queue_event (engine, ENGINE_EVENT_ID_GET(_part_event), 0);
The first parameter is the instance. If it is NULL the event will be dispatched to all state machines. The last parameter is the event register that can specify the event if required. From the state machine it can now be used as follows:
action (_part_event, part_action, FALSE)
This is a internal transition simply executing the part_action with the parameter FALSE.
Constants are declared as follows in the C code of the part:
ENGINE_CONST_IMPL (1, TRUE, "True") ;
ENGINE_CONST_IMPL (0, FALSE, "False") ;
Engine can be useful for separating the functionality or top-level state machine of your implementation from the rest of the code. This separation can be useful in some situations as it can help to create a clear boundary between different parts of the system.
The ability to use different state machines in a program without the need for recompiling the source code can have benefits for embedded systems, such as the ability to run unit tests more easily. Another benefit is the ability to run different variations of the program by choosing a different state machine as a runtime option.
The way the compiled code is executed, it is likely to be just as fast as native C. The straightforward syntax of Engine Machine Language makes it ideal for development using a text editor. With an editor that supports folding and unfolding of curly brackets, Engine may even be more convenient to use than many visual design tools. Another advantage of this is that there are no code generators involved, and this usually results in a greater level of control and flexibility over the final product.
The idea for the project came from Pico C. The lexer of Pico C was used with a few modifications and the parser changed to parse Engine Machine Language syntax. Pico C (https://gitlab.com/zsaleeba/picoc) is published under the "New BSD License". http://www.opensource.org/licenses/bsd-license.php
To draw the bitfields in the "Engine Byte Code" section I used the wavedrom bitfield library, very easy to use: https://github.com/wavedrom/bitfield. After installing you can just use a command like this to create the bitfield:
npx bit-field -i event.json > event.svg
The mp4 of the real world example was converted to a gif with https://ezgif.com/video-to-gif.
It was a nice surprise to see trains included in the search for the logo, Engine. Given the interest in trains that's common among engineers, http://clipart-library.com/clipart/steam-train-cliparts_6.htm should be a great representation.