RFSM is a framework for describing, simulating and generating code from reactive finite state machines. A reactive finite state machine is a finite state machine (FSM) for which transitions between states are always triggered by a single event. When the triggering event occurs a set of boolean expressions, called guards, may be used to decide whether the transition is taken or not and actions, such as updating an output or local variable or emitting an event, can also be performed.
As a simple example, consider the following model of a calibrated pulse generator. This model has
two inputs (h
, of type event
, and e
, of type bool
) and one output (s
, of type
bool
). Input h
is supposed to be periodic, with period T
. Input e
is sampled at each
occurrence of the h
event and when a 1
is read, the s
output is set to 1
for a duration
D=n.T
.
The model has two states (named E0
and E1
). The initial state is E0
.
The s
output is 0
when the machine is in state E0
and 1
when it is in state E1
.
A local variable, k
, of type int
is used to count the occurrences of the event h
when in state
E1
. Each transition has a label of the form
<ev>.<conds> / <acts>
where
<ev>
is the event triggering the transition (alwaysh
here)<conds>
is a (possibly empty) list of enabling boolean expressions ("guards") involving the inputs and local variables<acts>
is a (possibly empty) list of actions involving the outputs and local variables
For example, the transition from state E0
to state E1
, labeled h.(e=1)/k:=1
is taken whenever
input e
is equal to 1
when h
occurs and, when this happens, local variable k
is set to 1.
Here's a formulation of this model in RFSM :
fsm model gensig <n: int> (
in h: event,
in e: bool,
out s: bool)
{
states: E0 where s=0, E1 where s=1;
vars: k: int<1:n>;
trans:
| E0 -> E1 on h when e=1 with k:=1
| E1 -> E1 on h when k<n with k:=k+1
| E1 -> E0 on h when k=n;
itrans:
| -> E0;
}
Within the model body ({ ... }
), sections states
, vars
, trans
and itrans
respectively
describe the states, the local variables, the transitions and the initial transition. For each
transition, the triggering event, the enabling guards and associated actions are respectively
specified after the on
, when
and with
keywords. The duration n
of the pulse is here specified as
a parameter of the model.
This model can be simulated by writing, for example, the following RFSM program :
input H : event = periodic (10,10,80)
input E : bool = value_changes (0:0, 25:1, 35:0)
output S : bool
fsm g = gensig<3>(H,E,S)
This program declares two global inputs, H
and E
and one global output S
, attaches stimuli
to the input and instanciates the previously declared gensig
model (setting the value of the n
parameter to 3 here and binding the global IOs to the model IOs).
Invoking the rfmsc
compiler with the -sim
option produces .vcd
file, which can
be viewed with the Gtkwave trace viewer for example, as shown
below
Invoking the rfmsc
compiler with the -ctask
, -systemc
and -vhdl
option can generate C,
SystemC and VHDL code for the model (and the testbench) for simulation and implementation on a
target platform (micro-controlers or FPGAs for instance). The code generated from the example above
can be viewed here.
RFSM can also be used to describe multi-FSM models. Here's a description of a simple modulo-8 counter as three concurrent modulo-2 counters :
fsm model cntmod2(
in h: event,
out s: bool,
out r: event)
{
states: E0 where s=0, E1 where s=1;
trans:
| E0 -> E1 on h
| E1 -> E0 on h with r;
itrans:
| -> E0;
}
input H: event = periodic(10,10,100)
output S0, S1, S2: bool
output R2: event
shared R0, R1: event
fsm C0 = cntmod2(H,S0,R0)
fsm C1 = cntmod2(R0,S1,R1)
fsm C2 = cntmod2(R1,S2,R2)
We here have three instances of the cntmod2
model. These instances are synchronized using two
internal, shared events named R0
and R1
.
A graphical view of global model (obtained by invoking the rfsmc
compiler with the -dot
option
and displaying this file with Graphviz) is given below
Simulation of this model produces the following trace :
The latest stable version is provided as a ready-to-install OPAM package. Just type
opam install rfsm
This will install
- the compiler
rfsmc
and Makefile generatorrfsmmake
in~/.opam/<switch>/bin
- a platform definition file
~/.opam/<switch>/share/rfsm/platform
- VHDL and SystemC support libraries in
~/.opam/<switch>/share/rfsm/lib
- a collection of examples in
~/.opam/<switch>/share/rfsm/examples
To test the compiler on an example
- Make a copy the example source directory. Ex:
cp -r ~/.opam/<switch>/share/rfsm/examples/single/gensig /tmp/gensig
- Go to the copied directory:
cd /tmp/gensig
- Invoke the Makefile generator:
rfsmmake main.pro
The generated Makefile
contains the required set of rules to
- generate and view the graphical representation of the system (
make dot
) - simulate the behavior the system and view the execution traces (
make sim
) - generate code describing the system in C (
make ctask.code
), SystemC (make systemc.code
) and VHDL (make vhdl.code
)
Viewing the graphical representations (.dot
files) and the execution traces (.vcd
files) is
carried out by calling external programs called $DOTVIEWER
and $VCDVIEWER
in the Makefile.
Default values are provided in the file ~/.opam/<switch>/share/rfsm/platform
but these values
will probably to be adjusted according to your system.
The generated SystemC (resp. VHDL) code is written in
sub-directory ./systemc
(resp. ./vhdl
). Also generated in these directories is a dedicated
Makefile
for compiling and running the generated code and viewing the results. This Makefile
is derived from a template
located in directory ~/.opam/<switch>/share/rfsm/templates/
. These templates will also probably have to be
adjusted to suit your local SystemC
or VHDL
installation.
The user and reference manuals can be found here and here.
The implementation of RFSM relies on the host+guest design pattern described
here. It is fairly easy to build one's own compiler
implementing a variant of the "standard" RFSM language and supporting dedicated expression
sub-languages and type systems. The process is described in the reference
manual. Several examples
of variant languages are provided in the distribution (under directory src/guests/others
).
The so-called host RFSM library API supporting this mechanism is documented here.
A Graphical User Interface to the RFSM compiler - but restricted to mono-FSM models is provided by the RfsmLight application.