UP | HOME

Playing with explicit Kripke structures

Table of Contents

Kripke structures, can be defined as ω-automata in which labels are on states, and where all runs are accepting (i.e., the acceptance condition is t). They are typically used by model checkers to represent the state space of the model to verify.

On another page we have described how to implement a Kripke structure that can be explored on the fly, by implementing a function that take the current state, and produce its successor states. While implementing an on-the-fly Kripke structure is good for large example, it requires some implementation effort we are not always willing to put for a toy example.

This document shows how to create a Kripke structure that is stored as an explicit graph. The class for those is spot::kripke_graph and works in a similar way as the class spot::twa_graph used for automata. The main difference between those two classes is that Kripke structures labels the states instead of the transitions. Using spot::kripke_graph instead of spot::twa_graph saves a bit of memory.

Our Kripke structure represent a model whose states consist of pairs of modulo-3 integers \((x,y)\). At any state the possible actions will be to increment any one of the two integers (nondeterministically). That increment is obviously done modulo 3. For instance state \((1,2)\) has two possible successors:

Initially both variables will be 0. The complete state space has 9 states, that we will store explicitly as a graph.

In addition, we would like to label each state by atomic propositions odd_x and odd_y that are true only when the corresponding variables are odd. Using such variables, we could try to verify whether if odd_x infinitely often holds, then odd_y infinitely often holds as well.

C++ implementation

Building the state space

Here is how we could create the Kripke structure:

#include <spot/kripke/kripkegraph.hh>

spot::kripke_graph_ptr create_kripke(spot::bdd_dict_ptr dict,
                                     bool named_states = false)
{
  spot::kripke_graph_ptr k = spot::make_kripke_graph(dict);

  bdd odd_x = bdd_ithvar(k->register_ap("odd_x"));
  bdd odd_y = bdd_ithvar(k->register_ap("odd_y"));

  unsigned x0y0 = k->new_state((!odd_x) & !odd_y);
  unsigned x0y1 = k->new_state((!odd_x) & odd_y);
  unsigned x0y2 = k->new_state((!odd_x) & !odd_y);
  unsigned x1y0 = k->new_state(odd_x & !odd_y);
  unsigned x1y1 = k->new_state(odd_x & odd_y);
  unsigned x1y2 = k->new_state(odd_x & !odd_y);
  unsigned x2y0 = k->new_state((!odd_x) & !odd_y);
  unsigned x2y1 = k->new_state((!odd_x) & odd_y);
  unsigned x2y2 = k->new_state((!odd_x) & !odd_y);

  k->set_init_state(x0y0);

  k->new_edge(x0y0, x0y1); k->new_edge(x0y0, x1y0);
  k->new_edge(x0y1, x0y2); k->new_edge(x0y1, x1y1);
  k->new_edge(x0y2, x0y0); k->new_edge(x0y2, x1y2);
  k->new_edge(x1y0, x1y1); k->new_edge(x1y0, x2y0);
  k->new_edge(x1y1, x1y2); k->new_edge(x1y1, x2y1);
  k->new_edge(x1y2, x1y0); k->new_edge(x1y2, x2y2);
  k->new_edge(x2y0, x2y1); k->new_edge(x2y0, x0y0);
  k->new_edge(x2y1, x2y2); k->new_edge(x2y1, x0y1);
  k->new_edge(x2y2, x2y0); k->new_edge(x2y2, x0y2);

  if (named_states)
    {
      auto names = new std::vector<std::string>
        { "x0y0", "x0y1", "x0y2",
          "x1y0", "x1y1", "x1y2",
          "x2y0", "x2y1", "x2y2" };
      k->set_named_prop("state-names", names);
    }
  return k;
}

To display this Kripke structure, we can call the print_dot() function:

#include <spot/twaalgos/dot.hh>

int main()
{
  spot::bdd_dict_ptr dict = spot::make_bdd_dict();
  spot::print_dot(std::cout, create_kripke(dict));
}

print_dot() will output the graph in GraphViz's syntax, and you just have to pass it to GraphViz's dot command to render it:

Sorry, your browser does not support SVG.

By default, states are just numbered. If you want to name them, for instance for debugging, you should define the "state-names" properties to a vector of names. Our create_kripke() function does that when passed a true argument.

#include <spot/twaalgos/dot.hh>

int main()
{
  spot::bdd_dict_ptr dict = spot::make_bdd_dict();
  spot::print_dot(std::cout, create_kripke(dict, true));
}

Sorry, your browser does not support SVG.

Checking a property on this state space

Let us pretend that we want to verify the following property: if odd_x infinitely often holds, then odd_y infinitely often holds.

In LTL, that would be GF(odd_x) -> GF(odd_y).

To check this formula, we translate its negation into an automaton, build the product of this automaton with our Kripke structure, and check whether the output is empty. If it is not, that means we have found a counterexample. Here is some code that would show this counterexample:

#include <spot/tl/parse.hh>
#include <spot/twaalgos/translate.hh>
#include <spot/twaalgos/emptiness.hh>

int main()
{
   auto d = spot::make_bdd_dict();

   // Parse the input formula.
   spot::parsed_formula pf = spot::parse_infix_psl("GF(odd_x) -> GF(odd_y)");
   if (pf.format_errors(std::cerr))
     return 1;

   // Translate its negation.
   spot::formula f = spot::formula::Not(pf.f);
   spot::twa_graph_ptr af = spot::translator(d).run(f);

   // Find a run of our Kripke structure that intersects af.
   auto k = create_kripke(d, true);
   if (auto run = k->intersecting_run(af))
     std::cout << "formula is violated by the following run:\n" << *run;
   else
     std::cout << "formula is verified\n";
}
formula is violated by the following run:
Prefix:
  x0y0
  |  !odd_x & !odd_y
  x1y0
  |  odd_x & !odd_y
Cycle:
  x2y0
  |  !odd_x & !odd_y
  x0y0
  |  !odd_x & !odd_y
  x1y0
  |  odd_x & !odd_y

Note that this main function is similar to the main function we used for the on-the-fly version except for the line that creates the Kripke structure. You can modify it to display the counterexample in a similar way.

Python implementation

Building the state space

import spot
from buddy import bdd_ithvar

def create_kripke(bdddict, name_states=False):
  k = spot.make_kripke_graph(bdddict)
  odd_x = bdd_ithvar(k.register_ap("odd_x"))
  odd_y = bdd_ithvar(k.register_ap("odd_y"))

  x0y0 = k.new_state(-odd_x & -odd_y);
  x0y1 = k.new_state(-odd_x & odd_y);
  x0y2 = k.new_state(-odd_x & -odd_y);
  x1y0 = k.new_state(odd_x & -odd_y);
  x1y1 = k.new_state(odd_x & odd_y);
  x1y2 = k.new_state(odd_x & -odd_y);
  x2y0 = k.new_state(-odd_x & -odd_y);
  x2y1 = k.new_state(-odd_x & odd_y);
  x2y2 = k.new_state(-odd_x & -odd_y);

  k.set_init_state(x0y0);

  k.new_edge(x0y0, x0y1); k.new_edge(x0y0, x1y0);
  k.new_edge(x0y1, x0y2); k.new_edge(x0y1, x1y1);
  k.new_edge(x0y2, x0y0); k.new_edge(x0y2, x1y2);
  k.new_edge(x1y0, x1y1); k.new_edge(x1y0, x2y0);
  k.new_edge(x1y1, x1y2); k.new_edge(x1y1, x2y1);
  k.new_edge(x1y2, x1y0); k.new_edge(x1y2, x2y2);
  k.new_edge(x2y0, x2y1); k.new_edge(x2y0, x0y0);
  k.new_edge(x2y1, x2y2); k.new_edge(x2y1, x0y1);
  k.new_edge(x2y2, x2y0); k.new_edge(x2y2, x0y2);

  if name_states:
    k.set_state_names(["x0y0", "x0y1", "x0y2",
                       "x1y0", "x1y1", "x1y2",
                       "x2y0", "x2y1", "x2y2"])
  return k;

To display this structure, we would just call to_str('dot') (and convert the resulting textual output into graphical form using the dot command).

k = create_kripke(spot.make_bdd_dict())
print(k.to_str('dot'))

print_dot() will output the graph in GraphViz's syntax, and you just have to pass it to GraphViz's dot command to render it:

Sorry, your browser does not support SVG.

Again, states may be named if that can help. This is done by passing a vector of names (indexed by state numbers) to the set_name_states() method, as done conditionally in our create_kripke() function.

k = create_kripke(spot.make_bdd_dict(), True)
print(k.to_str('dot'))

Sorry, your browser does not support SVG.

Checking a property on this state space

Here is the Python code equivalent to our C++ check. Please read the C++ description for details.


d = spot.make_bdd_dict()

# Translate the negation of the formula
f = spot.formula.Not("GF(odd_x) -> GF(odd_y)")
af = spot.translate(f, dict=d)

# Find a run of our Kripke structure that intersects af.
k = create_kripke(d, True)
run = k.intersecting_run(af)
if run:
  print("formula is violated by the following run:\n", run)
else:
  print("formula is verified")
formula is violated by the following run:
 Prefix:
  x0y0
  |  !odd_x & !odd_y
  x1y0
  |  odd_x & !odd_y
Cycle:
  x2y0
  |  !odd_x & !odd_y
  x0y0
  |  !odd_x & !odd_y
  x1y0
  |  odd_x & !odd_y