Skip to content

Latest commit

 

History

History
182 lines (138 loc) · 7.56 KB

README.md

File metadata and controls

182 lines (138 loc) · 7.56 KB

DESCRIPTION

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 (always h 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 :

INSTALLATION

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 generator rfsmmake 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

Running the examples

To test the compiler on an example

  1. Make a copy the example source directory. Ex: cp -r ~/.opam/<switch>/share/rfsm/examples/single/gensig /tmp/gensig
  2. Go to the copied directory: cd /tmp/gensig
  3. 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.

DOCUMENTATION

The user and reference manuals can be found here and here.

BUILDING A CUSTOM COMPILER

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.

RELATED TOOLS

A Graphical User Interface to the RFSM compiler - but restricted to mono-FSM models is provided by the RfsmLight application.