Skip to content

Watayut/tsm

 
 

Repository files navigation

Build Status codecov.io Coverity Scan License: MIT

The C++ API documentation has an overview of usage and architecture and a discussion of drawbacks.

What is tsm?

tsm is a state machine framework with support for Hierarchical and Orthogonal State Machines.

Features:

* Hierarchical State machine.
* Thread safe event queue.
* Single Threaded (Moore) or Asynchronous (Mealy) execution.
* Timer driven execution policy.
* Ease of installation and integration - Header only, CMake and Nix support.
* Policy Based - Ability to customize behavior by defining execution policies.

Quick Overview

A Finite State Machine

struct Switch : Hsm<Switch>
{
    Switch()
    {
        setStartState(&off);

        add(on, toggle, off);
        add(off, toggle, on);
    }

    State on, off;

    Event toggle;
};

Clients can interact with this state machine in two ways - Synchronously and Asynchronously.

Single Threaded Execution Policy
using SwitchFsm = SingleThreadedHsm<Switch>;
int main() {
    SwitchFsm s;
    s.startSM();
    s.sendEvent(s.toggle);
    s.step();
    // ...
    s.stopSM();
}
Asynchronous Execution Policy
using SwitchFsm = AsynchronousHsm<Switch>;
int main() {
    SwitchFsm s;
    s.startSM();
    s.sendEvent(s.toggle);
    // ...
    s.stopSM();
}

s.step() is missing in the AsynchronousHsm. The EventQueue blocks waiting for the next event to arrive and processes it as soon as it arrives.

A Hierarchical State Machine

Assume you are trying to model a CdPlayer as a state machine.

struct CdPlayer : public Hsm<CdPlayer>
{
    // Actions
    void PlaySong()
    {
        LOG(INFO) << "Play Song";
    }

    // Guards
    bool PlaySongGuard()
    {
        DLOG(INFO) << "Play Song Guard";
        return true;
    }
    // States
    State Stopped, Paused, Empty, Open;
    PlayingHsm Playing; // <-------- Note

    // Events
    Event play, open_close, stop_event, cd_detected, pause, end_pause;
}

If you noticed, Playing is another state machine hidden among the CdPlayer's states declared above. State Machines are states too. When you have the ability to integrate them in this manner, you get a Hierarchical State Machine aka State Chart aka Harel State Chart. What does Playing look like?

// Playing Hsm
struct Playing : public Hsm<PlayingHsm>
{

    // ... ignoring boiler plate stuff ...

    // States
    State Song1, Song2, Song3;

    // Events
    Event next_song, prev_song;

};

Let's see how the state transition table looks like for the CdPlayer.

State Transition Table
// Goes in the header along with your State Machine class or .cpp file.
CdPlayerHsm::CdPlayerHsm()
{
    setStartState(&Empty);

    // Tell Playing who's the parent. <---------------- Note
    Playing.setParent(this);

    // State Transition Table
    add(Stopped, play, Playing); // <-------- Note
    add(Stopped, open_close, Open);
    add(Stopped, stop_event, Stopped);
    //-------------------------------------------------
    add(Open, open_close, Empty);
    //-------------------------------------------------
    add(Empty, open_close, Open);
    add(Empty, cd_detected, Stopped);
    add(Empty, cd_detected, Playing);
    //-------------------------------------------------
    add(Playing, stop_event, Stopped);
    add(Playing, pause, Paused);
    add(Playing, open_close, Open);
    //-------------------------------------------------
    add(Paused, end_pause, Playing);
    add(Paused, stop_event, Stopped);
    add(Paused, open_close, Open);
}

Playing is just treated just as any other (atomic) state. A transition like add(Stopped, play, Playing);, is read as "When the machine is Stopped and gets a play event, it transitions to the Playing state". Transitions can have actions and guards attached to them.

Actions and Guards

Looking at Playing's state transition table,

PlayingHsm()
{
    setStartState(&Song1);

    add(Song1, next_song, Song2, &PlayingHsm::PlaySong, &PlayingHsm::PlaySongGuard);  // <-------- Note
    add(Song2, next_song, Song3);
    add(Song3, prev_song, Song2);
    add(Song2, prev_song, Song1);
}

we notice PlaySong and PlaySongGuard. Guards of course prevent a transition from taking place. To make such decisions, they must have access to some additional information pertaining to the system.

bool PlaySongGuard()
{
    // purely made up
    if (currentSong_.noiseLevel() > 50) {
        return false;
    }
    return true;
}

Actions perform an action after exiting the current state and before entering the next. Ideally, they are able to perform the action based on some knowledge of the system state. Hence they are implemented as member functions of an Hsm. As seen for SwitchFsm, we can do:

#include <tsm.h>
int main() {
    SingleThreadedHsm<CdPlayer> sm;
    sm.startSM();
    sm.sendEvent(play);
    sm.step(); // take the 'play' event from the event queue and transition to the next state
    // ...
    sm.stopSM();
}

or

AsynchronousHsm<CdPlayer> sm;
sm.sendEvent(play);

Note again that the call to sm.step() is not required for the AsynchronousHsm. Events will be processed in a separate thread.

Start and Stop States

Specify the start state within the Hsm's constructor setStartState(&Song1). This is required. If there is a termination state, that has to be specified as well e.g. setStopState(&Song3).

TimedExecutionPolicy

A whole class of problems can be solved in a much simpler manner with state machines that are driven by timers. Consider the problem of having to model traffic lights at a 2-way crossing. The states are G1(30s), Y1(5s), G2(60s), Y2(5s). When G1 or Y1 are on, the opposite R2 is on etc. The signal stays on for the amount of time indicated in parenthesis before moving on to the next. The added complication is that G2 has a walk signal. If the walk signal is pressed, G2 stays on for only 30s instead of 60s before transitioning to Y2. The trick is to realize that there is only one event for this state machine: The expiry of a timer at say, 1s granularity. Such problems can be modeled by using timer driven state machines. Applications include game engines where a refresh of the game state happens every so many milliseconds, robotics, embedded software and of course traffic lights :). This problem is modeled with a custom "handle" method without a state transition table and a LightState type inherited from the State struct.

struct TrafficLightHsm : public IHsm
{
    // ... with a few details removed...

    // States
    LightState g1, y1, y2;
    G2 g2;      // G2 is derived from LightState, which in turn is inherited from State.

    // Events
    Event timer_event;

    // Walk button
    bool walkPressed;
    uint64_t ticks_;
};

LightState looks like this:

    struct LightState : public State
    {
        LightState(std::string const& name, uint64_t limit, LightState& next)
          : State(name)
          , limit_(limit)
          , next_(next)
        {}

        ~LightState() override = default;

        uint64_t getLimit() const { return limit_; }
        LightState& nextState() { return next_; }

      private:
        const uint64_t limit_;
        LightState& next_;
    };

Event handling, guard/action transition table functionality etc. are all captured here:

    void TrafficLightHsm::handle(Event const&) override
    {
        ++ticks_;
        auto* state = static_cast<LightState*>(this->currentState_);
        bool guard = (this->ticks_ > state->getLimit());
        if (state->id == G2.id) {
            guard |= (walkPressed && (this->ticks_ > G2WALK));
        }

        if (guard) { // ok to transition?
            // disable walkPressed when exiting G2
            if ((state->id == G2.id) && walkPressed) {
                walkPressed = false;
            }

            ticks_ = 0;
            // set the next state
            this->currentState_ = &state->nextState();
        }
    }

To turn it into a timer driven state machine,

#include <tsm.h>
using TrafficLightTimedHsm = tsm::ClockedMooreHsm<TrafficLightHsm,
                                                  tsm::ThreadSleepTimer,
                                                  std::chrono::microseconds>;

using AsyncTrafficLightTimedHsm =
  TimedExecutionPolicy<AsynchronousHsm<TrafficLightHsm>,  // <----- Note use of AsynchronousHsm
                       tsm::ThreadSleepTimer,
                       std::chrono::milliseconds>;
int main() {
    using namespace std::chrono_literals;
    // Painfully named as such to showcase capabilities.
    AsyncTrafficLightTimedHsm t(1ms);
    t.startSM();
    // ...
}

t.start() will start a timer with a period of 1s. At the expiration of this timer, a timer_event will be placed in the event queue.

Policy Based Design

We could just as well have created a timer driven traffic light in this manner:

#include <tsm.h>

using TrafficLightTimedHsm = tsm::ClockedMooreHsm<TrafficLightHsm,
                                                  tsm::ThreadSleepTimer,
                                                  std::chrono::milliseconds>;
int main() {
    using namespace std::chrono_literals;
    // Painfully named as such to showcase capabilities.
    TrafficLightTimedHsm t(1ms);
    t.startSM();
    // ...
}

The only difference is that we are creating a TrafficLightTimedHsm where we have to drive the state machine with t.step() for event processing. That is flexible... and powerful. We can also write policies to

* Persist state changes to log file
* Calculate statistics on transitions
* Write 'watchers' for particular conditions
* Notify observers on events

Like it? Try it.

Build

CMake only

If you don't want to install nix, sudo apt install cmake ninja-build graphviz doxygen or brew install cmake ninja graphviz doxygen. Read the CMakeLists.txt to get a good feel for it. The cmake/superbuild folder contains the cmake files that download and install the external dependencies. With Nix, none of these cmake files are needed. The dependencies and environment will be set up when you invoke nix-shell or nix-build.

git clone https://github.com/tinverse/tsm.git
cd tsm; mkdir build; cd build
# cmake .. && make #if you want to use make
cmake -GNinja -DCMAKE_INSTALL_PREFIX=${HOME}/usr .. && ninja install
# run tests
./bin/tsm_test

Integrating with your CMake project

How do I use it from my project? Look at the example project CMakeLists.txt. Use this as a template for your project's CMakeLists.txt. The tsm_DIR variable should be set to point to the location of tsmConfig.cmake (for the case above, ${HOME}/usr/lib/cmake/tsm).

Nix - Recommended

Install nix by running curl https://nixos.org/nix/install | sh. The default.nix file is responsible for setting up your environment and installing all required dependencies.

nix-shell --run ./build.sh

And you are done! If you open build.sh you'll notice that it uses ninja instead of make. build.sh creates a build folder, invokes cmake to generate the build scripts and then calls ninja or make via the cmake --build command to create the build outputs. You can find the test executable tsm_test under build/test, documentation under build/docs/html and coverage data under test/tsm_test-coverage. For your normal workflow,

nix-shell
cd build
ninja tsm_test

You can also run nix-build from the command-line. This will build all targets and deploy them to /nix/store. A sym-link named result will be created in the cloned folder (the folder that contains default.nix) to the /nix/store. So to invoke tsm_test, run ./result/test/tsm_test.

Test Coverage

If making changes to tsm source, you can generate coverage reports as well. The cmake option -DBUILD_COVERAGE=ON turns it on. Feel free to steal this mechanism for your own projects. ninja coverage will invoke lcov and the report will be available under test/tsm_test-coverage/index.html in the build folder.

Documentation

To generate doxygen docs, use the cmake option -DBUILD_DOCUMENTATION=ON. This can be invoked as needed - ninja tsm_doc or just plain ninja from the build folder.

External Dependencies

None, if you just want to use the library. Just #include <tsm> and start using it! Catch2 unit test framework if you wish to run the unit tests. Set BUILD_TESTING=OFF in the CMakeCache file to disable building unit tests.

Contribute

Please feel free to write up issues and submit pull requests. There is a .clang-format file that comes with the source. Coding conventions can be inferred by looking at other source files.

TODOs

* UML front end to define State Machine.
* Implement concurrent execution policy for OrthogonalHsmExecutor.

About

A Hierarchical State Machine Framework in C++

Resources

License

Code of conduct

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • C++ 79.4%
  • CMake 18.7%
  • Other 1.9%