Add support to load GAL models.
* spot/ltsmin/ltsmin.cc: Handle GAL models. * tests/Makefile.am: Test the new feature. * tests/ltsmin/check.test: Also check GAL. * tests/ltsmin/beem-peterson.4.gal: A new GAL model for tests. * tests/ltsmin/finite.gal: A new GAL model for tests. * tests/ltsmin/finite3.test: A new test for GAL.
This commit is contained in:
parent
ec83e60bb9
commit
c9aabcddab
6 changed files with 268 additions and 9 deletions
|
|
@ -948,7 +948,12 @@ namespace spot
|
||||||
std::string command;
|
std::string command;
|
||||||
std::string compiled_ext;
|
std::string compiled_ext;
|
||||||
|
|
||||||
if (ext == ".prom" || ext == ".pm" || ext == ".pml")
|
if (ext == ".gal")
|
||||||
|
{
|
||||||
|
command = "gal2c " + filename;
|
||||||
|
compiled_ext = "2C";
|
||||||
|
}
|
||||||
|
else if (ext == ".prom" || ext == ".pm" || ext == ".pml")
|
||||||
{
|
{
|
||||||
command = "spins " + filename;
|
command = "spins " + filename;
|
||||||
compiled_ext = ".spins";
|
compiled_ext = ".spins";
|
||||||
|
|
@ -962,7 +967,8 @@ namespace spot
|
||||||
{
|
{
|
||||||
throw std::runtime_error(std::string("Unknown extension '")
|
throw std::runtime_error(std::string("Unknown extension '")
|
||||||
+ ext + "'. Use '.prom', '.pm', '.pml', "
|
+ ext + "'. Use '.prom', '.pm', '.pml', "
|
||||||
"'.dve', '.dve2C', or '.prom.spins'.");
|
"'.dve', '.dve2C', '.gal', '.gal2C' or "
|
||||||
|
"'.prom.spins'.");
|
||||||
}
|
}
|
||||||
|
|
||||||
struct stat s;
|
struct stat s;
|
||||||
|
|
@ -980,7 +986,7 @@ namespace spot
|
||||||
struct stat d;
|
struct stat d;
|
||||||
if (stat(filename.c_str(), &d) == 0)
|
if (stat(filename.c_str(), &d) == 0)
|
||||||
if (s.st_mtime < d.st_mtime)
|
if (s.st_mtime < d.st_mtime)
|
||||||
// The .spins or .dve2C is up-to-date, no need to recompile it.
|
// The .spins or .dve2C or .gal2C is up-to-date, no need to compile.
|
||||||
return;
|
return;
|
||||||
|
|
||||||
int res = system(command.c_str());
|
int res = system(command.c_str());
|
||||||
|
|
@ -1002,7 +1008,7 @@ namespace spot
|
||||||
file = "./" + file_arg;
|
file = "./" + file_arg;
|
||||||
|
|
||||||
std::string ext = file.substr(file.find_last_of("."));
|
std::string ext = file.substr(file.find_last_of("."));
|
||||||
if (ext != ".spins" && ext != ".dve2C")
|
if (ext != ".spins" && ext != ".dve2C" && ext != ".gal2C")
|
||||||
compile_model(file, ext);
|
compile_model(file, ext);
|
||||||
|
|
||||||
if (lt_dlinit())
|
if (lt_dlinit())
|
||||||
|
|
@ -1020,9 +1026,10 @@ namespace spot
|
||||||
d->handle = h;
|
d->handle = h;
|
||||||
|
|
||||||
// SpinS interface.
|
// SpinS interface.
|
||||||
if ((d->get_initial_state = (void (*)(void*))
|
if (ext == ".spins")
|
||||||
lt_dlsym(h, "spins_get_initial_state")))
|
|
||||||
{
|
{
|
||||||
|
d->get_initial_state = (void (*)(void*))
|
||||||
|
lt_dlsym(h, "spins_get_initial_state");
|
||||||
d->have_property = nullptr;
|
d->have_property = nullptr;
|
||||||
d->get_successors = (int (*)(void*, int*, TransitionCB, void*))
|
d->get_successors = (int (*)(void*, int*, TransitionCB, void*))
|
||||||
lt_dlsym(h, "spins_get_successor_all");
|
lt_dlsym(h, "spins_get_successor_all");
|
||||||
|
|
@ -1042,6 +1049,30 @@ namespace spot
|
||||||
lt_dlsym(h, "spins_get_type_value_name");
|
lt_dlsym(h, "spins_get_type_value_name");
|
||||||
}
|
}
|
||||||
// dve2 interface.
|
// dve2 interface.
|
||||||
|
else if (ext == ".dve2C")
|
||||||
|
{
|
||||||
|
d->get_initial_state = (void (*)(void*))
|
||||||
|
lt_dlsym(h, "get_initial_state");
|
||||||
|
d->have_property = (int (*)())
|
||||||
|
lt_dlsym(h, "have_property");
|
||||||
|
d->get_successors = (int (*)(void*, int*, TransitionCB, void*))
|
||||||
|
lt_dlsym(h, "get_successors");
|
||||||
|
d->get_state_size = (int (*)())
|
||||||
|
lt_dlsym(h, "get_state_variable_count");
|
||||||
|
d->get_state_variable_name = (const char* (*)(int))
|
||||||
|
lt_dlsym(h, "get_state_variable_name");
|
||||||
|
d->get_state_variable_type = (int (*)(int))
|
||||||
|
lt_dlsym(h, "get_state_variable_type");
|
||||||
|
d->get_type_count = (int (*)())
|
||||||
|
lt_dlsym(h, "get_state_variable_type_count");
|
||||||
|
d->get_type_name = (const char* (*)(int))
|
||||||
|
lt_dlsym(h, "get_state_variable_type_name");
|
||||||
|
d->get_type_value_count = (int (*)(int))
|
||||||
|
lt_dlsym(h, "get_state_variable_type_value_count");
|
||||||
|
d->get_type_value_name = (const char* (*)(int, int))
|
||||||
|
lt_dlsym(h, "get_state_variable_type_value");
|
||||||
|
}
|
||||||
|
// .gal2C interface.
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
d->get_initial_state = (void (*)(void*))
|
d->get_initial_state = (void (*)(void*))
|
||||||
|
|
|
||||||
|
|
@ -375,20 +375,22 @@ check_SCRIPTS += ltsmin/defs
|
||||||
|
|
||||||
ltsmin/defs: $(top_builddir)/config.status $(srcdir)/core/defs.in
|
ltsmin/defs: $(top_builddir)/config.status $(srcdir)/core/defs.in
|
||||||
$(top_builddir)/config.status --file ltsmin/defs:core/defs.in
|
$(top_builddir)/config.status --file ltsmin/defs:core/defs.in
|
||||||
|
endif
|
||||||
|
|
||||||
|
if USE_LTSMIN
|
||||||
TESTS_ltsmin = \
|
TESTS_ltsmin = \
|
||||||
ltsmin/check.test \
|
ltsmin/check.test \
|
||||||
ltsmin/finite.test \
|
ltsmin/finite.test \
|
||||||
ltsmin/finite2.test \
|
ltsmin/finite2.test \
|
||||||
|
ltsmin/finite3.test \
|
||||||
ltsmin/kripke.test
|
ltsmin/kripke.test
|
||||||
|
|
||||||
EXTRA_DIST += ltsmin/beem-peterson.4.dve ltsmin/elevator2.1.pm \
|
EXTRA_DIST += ltsmin/beem-peterson.4.dve ltsmin/beem-peterson.4.gal \
|
||||||
ltsmin/finite.dve ltsmin/finite.pm
|
ltsmin/elevator2.1.pm ltsmin/finite.dve ltsmin/finite.pm ltsmin/finite.gal
|
||||||
|
|
||||||
ltlsmin/kripke.log: core/kripkecat$(EXEEXT)
|
ltlsmin/kripke.log: core/kripkecat$(EXEEXT)
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
|
||||||
############################## SANITY ##############################
|
############################## SANITY ##############################
|
||||||
|
|
||||||
TESTS_sanity = \
|
TESTS_sanity = \
|
||||||
|
|
|
||||||
135
tests/ltsmin/beem-peterson.4.gal
Normal file
135
tests/ltsmin/beem-peterson.4.gal
Normal file
|
|
@ -0,0 +1,135 @@
|
||||||
|
GAL peterson_4_dve
|
||||||
|
{
|
||||||
|
//arrays
|
||||||
|
array[4] pos= (0, 0, 0, 0) ;
|
||||||
|
array[4] step= (0, 0, 0, 0) ;
|
||||||
|
//variables
|
||||||
|
int P_0.state=0;
|
||||||
|
int P_0.j=0;
|
||||||
|
int P_0.k=0;
|
||||||
|
int P_1.state=0;
|
||||||
|
int P_1.j=0;
|
||||||
|
int P_1.k=0;
|
||||||
|
int P_2.state=0;
|
||||||
|
int P_2.j=0;
|
||||||
|
int P_2.k=0;
|
||||||
|
int P_3.state=0;
|
||||||
|
int P_3.j=0;
|
||||||
|
int P_3.k=0;
|
||||||
|
//transitions
|
||||||
|
transition t0 [ ( P_0.state == 0 ) ]
|
||||||
|
{ P_0.state = 2;
|
||||||
|
P_0.j = 1; }
|
||||||
|
|
||||||
|
transition t1 [ ( P_0.state == 1 ) ]
|
||||||
|
{ P_0.state = 0;
|
||||||
|
pos[0] = 0; }
|
||||||
|
|
||||||
|
transition t2 [ ( ( P_0.j < 4 ) && ( P_0.state == 2 ) ) ]
|
||||||
|
{ P_0.state = 3;
|
||||||
|
pos[0] = P_0.j; }
|
||||||
|
|
||||||
|
transition t3 [ ( ( P_0.j == 4 ) && ( P_0.state == 2 ) ) ]
|
||||||
|
{ P_0.state = 1; }
|
||||||
|
|
||||||
|
transition t4 [ ( P_0.state == 3 ) ]
|
||||||
|
{ P_0.state = 4;
|
||||||
|
step[( P_0.j - 1 )] = 0;
|
||||||
|
P_0.k = 0; }
|
||||||
|
|
||||||
|
transition t5 [ ( ( P_0.state == 4 ) && ( P_0.k < 4 ) && ( ( pos[P_0.k] < P_0.j ) || ( P_0.k == 0 ) ) ) ]
|
||||||
|
{ P_0.state = 4;
|
||||||
|
P_0.k = ( 1 + P_0.k ); }
|
||||||
|
|
||||||
|
transition t6 [ ( ( P_0.state == 4 ) && ( ( P_0.k == 4 ) || ( step[( P_0.j - 1 )] != 0 ) ) ) ]
|
||||||
|
{ P_0.state = 2;
|
||||||
|
P_0.j = ( P_0.j + 1 ); }
|
||||||
|
|
||||||
|
transition t7 [ ( P_1.state == 0 ) ]
|
||||||
|
{ P_1.state = 2;
|
||||||
|
P_1.j = 1; }
|
||||||
|
|
||||||
|
transition t8 [ ( P_1.state == 1 ) ]
|
||||||
|
{ P_1.state = 0;
|
||||||
|
pos[1] = 0; }
|
||||||
|
|
||||||
|
transition t9 [ ( ( P_1.j < 4 ) && ( P_1.state == 2 ) ) ]
|
||||||
|
{ P_1.state = 3;
|
||||||
|
pos[1] = P_1.j; }
|
||||||
|
|
||||||
|
transition t10 [ ( ( P_1.j == 4 ) && ( P_1.state == 2 ) ) ]
|
||||||
|
{ P_1.state = 1; }
|
||||||
|
|
||||||
|
transition t11 [ ( P_1.state == 3 ) ]
|
||||||
|
{ P_1.state = 4;
|
||||||
|
step[( P_1.j - 1 )] = 1;
|
||||||
|
P_1.k = 0; }
|
||||||
|
|
||||||
|
transition t12 [ ( ( P_1.state == 4 ) && ( ( pos[P_1.k] < P_1.j ) || ( P_1.k == 1 ) ) && ( P_1.k < 4 ) ) ]
|
||||||
|
{ P_1.state = 4;
|
||||||
|
P_1.k = ( 1 + P_1.k ); }
|
||||||
|
|
||||||
|
transition t13 [ ( ( P_1.state == 4 ) && ( ( step[( P_1.j - 1 )] != 1 ) || ( P_1.k == 4 ) ) ) ]
|
||||||
|
{ P_1.state = 2;
|
||||||
|
P_1.j = ( 1 + P_1.j ); }
|
||||||
|
|
||||||
|
transition t14 [ ( P_2.state == 0 ) ]
|
||||||
|
{ P_2.state = 2;
|
||||||
|
P_2.j = 1; }
|
||||||
|
|
||||||
|
transition t15 [ ( P_2.state == 1 ) ]
|
||||||
|
{ P_2.state = 0;
|
||||||
|
pos[2] = 0; }
|
||||||
|
|
||||||
|
transition t16 [ ( ( P_2.j < 4 ) && ( P_2.state == 2 ) ) ]
|
||||||
|
{ P_2.state = 3;
|
||||||
|
pos[2] = P_2.j; }
|
||||||
|
|
||||||
|
transition t17 [ ( ( P_2.j == 4 ) && ( P_2.state == 2 ) ) ]
|
||||||
|
{ P_2.state = 1; }
|
||||||
|
|
||||||
|
transition t18 [ ( P_2.state == 3 ) ]
|
||||||
|
{ P_2.state = 4;
|
||||||
|
step[( P_2.j - 1 )] = 2;
|
||||||
|
P_2.k = 0; }
|
||||||
|
|
||||||
|
transition t19 [ ( ( P_2.state == 4 ) && ( P_2.k < 4 ) && ( ( pos[P_2.k] < P_2.j ) || ( P_2.k == 2 ) ) ) ]
|
||||||
|
{ P_2.state = 4;
|
||||||
|
P_2.k = ( 1 + P_2.k ); }
|
||||||
|
|
||||||
|
transition t20 [ ( ( P_2.state == 4 ) && ( ( step[( P_2.j - 1 )] != 2 ) || ( P_2.k == 4 ) ) ) ]
|
||||||
|
{ P_2.state = 2;
|
||||||
|
P_2.j = ( 1 + P_2.j ); }
|
||||||
|
|
||||||
|
transition t21 [ ( P_3.state == 0 ) ]
|
||||||
|
{ P_3.state = 2;
|
||||||
|
P_3.j = 1; }
|
||||||
|
|
||||||
|
transition t22 [ ( P_3.state == 1 ) ]
|
||||||
|
{ P_3.state = 0;
|
||||||
|
pos[3] = 0; }
|
||||||
|
|
||||||
|
transition t23 [ ( ( P_3.state == 2 ) && ( P_3.j < 4 ) ) ]
|
||||||
|
{ P_3.state = 3;
|
||||||
|
pos[3] = P_3.j; }
|
||||||
|
|
||||||
|
transition t24 [ ( ( P_3.state == 2 ) && ( P_3.j == 4 ) ) ]
|
||||||
|
{ P_3.state = 1; }
|
||||||
|
|
||||||
|
transition t25 [ ( P_3.state == 3 ) ]
|
||||||
|
{ P_3.state = 4;
|
||||||
|
step[( P_3.j - 1 )] = 3;
|
||||||
|
P_3.k = 0; }
|
||||||
|
|
||||||
|
transition t26 [ ( ( P_3.state == 4 ) && ( P_3.k < 4 ) && ( ( P_3.k == 3 ) || ( pos[P_3.k] < P_3.j ) ) ) ]
|
||||||
|
{ P_3.state = 4;
|
||||||
|
P_3.k = ( 1 + P_3.k ); }
|
||||||
|
|
||||||
|
transition t27 [ ( ( P_3.state == 4 ) && ( ( step[( P_3.j - 1 )] != 3 ) || ( P_3.k == 4 ) ) ) ]
|
||||||
|
{ P_3.state = 2;
|
||||||
|
P_3.j = ( 1 + P_3.j ); }
|
||||||
|
|
||||||
|
// transient predicate
|
||||||
|
TRANSIENT = false;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -35,6 +35,11 @@ if ! test -x "$(which spins)"; then
|
||||||
exit 77
|
exit 77
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
if ! test -x "$(which gal2c)"; then
|
||||||
|
echo "gal2c not installed."
|
||||||
|
exit 77
|
||||||
|
fi
|
||||||
|
|
||||||
set -e
|
set -e
|
||||||
|
|
||||||
# Promela
|
# Promela
|
||||||
|
|
@ -65,6 +70,18 @@ for opt in '' '-z'; do
|
||||||
run 0 ../modelcheck $opt -e $srcdir/beem-peterson.4.dve '!G("pos[1] < 3")'
|
run 0 ../modelcheck $opt -e $srcdir/beem-peterson.4.dve '!G("pos[1] < 3")'
|
||||||
done
|
done
|
||||||
|
|
||||||
|
# gal
|
||||||
|
# TODO also compare gal and dve results
|
||||||
|
for opt in '' '-z'; do
|
||||||
|
# The three examples from the README.
|
||||||
|
# (Don't run the first one using "run 0" because it would take too much
|
||||||
|
# time with valgrind.).
|
||||||
|
|
||||||
|
run 0 ../modelcheck $opt -e $srcdir/beem-peterson.4.gal \
|
||||||
|
'!G("P_0.state==2" -> F "P_0.state==1")'
|
||||||
|
run 0 ../modelcheck $opt -e $srcdir/beem-peterson.4.gal '!G("pos[1] < 3")'a
|
||||||
|
done
|
||||||
|
|
||||||
# Now check some error messages.
|
# Now check some error messages.
|
||||||
run 1 ../modelcheck foo.dve "F(P_0.CS)" 2>stderr
|
run 1 ../modelcheck foo.dve "F(P_0.CS)" 2>stderr
|
||||||
cat stderr
|
cat stderr
|
||||||
|
|
|
||||||
19
tests/ltsmin/finite.gal
Executable file
19
tests/ltsmin/finite.gal
Executable file
|
|
@ -0,0 +1,19 @@
|
||||||
|
gal finite
|
||||||
|
{
|
||||||
|
//arrays
|
||||||
|
//variables
|
||||||
|
int P.a=0;
|
||||||
|
int P.b=0;
|
||||||
|
//transitions
|
||||||
|
transition t0 [ ( ( P.a < 3 ) && ( P.b < 3 ) ) ]
|
||||||
|
{ P.a = ( 1 + P.a );
|
||||||
|
}
|
||||||
|
|
||||||
|
transition t1 [ ( ( P.a < 3 ) && ( P.b < 3 ) ) ]
|
||||||
|
{ P.b = ( 1 + P.b );
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
55
tests/ltsmin/finite3.test
Executable file
55
tests/ltsmin/finite3.test
Executable file
|
|
@ -0,0 +1,55 @@
|
||||||
|
#!/bin/sh
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Copyright (C) 2011, 2013, 2014, 2016 Laboratoire de Recherche et
|
||||||
|
# Développement de l'Epita (LRDE).
|
||||||
|
#
|
||||||
|
# This file is part of Spot, a model checking library.
|
||||||
|
#
|
||||||
|
# Spot is free software; you can redistribute it and/or modify it
|
||||||
|
# under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation; either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# Spot is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
# or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public
|
||||||
|
# License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
|
||||||
|
. ./defs
|
||||||
|
|
||||||
|
# Compile the model beforehand to avoid compiler's output interference.
|
||||||
|
if ! gal2c $srcdir/finite.gal; then
|
||||||
|
echo "gal2c not functionnal."
|
||||||
|
exit 77
|
||||||
|
fi
|
||||||
|
|
||||||
|
set -e
|
||||||
|
run 0 ../modelcheck -gm $srcdir/finite.gal '"P.a < 10"' > stdout
|
||||||
|
test `grep ' -> ' stdout | wc -l` = 25
|
||||||
|
test `grep 'P.a=' stdout | wc -l` = 15
|
||||||
|
|
||||||
|
run 0 ../modelcheck -dtrue -gm $srcdir/finite.gal '"P.a < 10"' > stdout2
|
||||||
|
cmp stdout stdout2
|
||||||
|
|
||||||
|
run 0 ../modelcheck -dfalse -gm $srcdir/finite.gal '"P.a < 10"' > stdout
|
||||||
|
test `grep ' -> ' stdout | wc -l` = 19
|
||||||
|
test `grep 'P.a=' stdout | wc -l` = 15
|
||||||
|
|
||||||
|
# the same with compressed states
|
||||||
|
run 0 ../modelcheck -z -dfalse -gm $srcdir/finite.gal '"P.a < 10"' > stdout
|
||||||
|
test `grep ' -> ' stdout | wc -l` = 19
|
||||||
|
test `grep 'P.a=' stdout | wc -l` = 15
|
||||||
|
|
||||||
|
run 0 ../modelcheck -ddead -E $srcdir/finite.gal \
|
||||||
|
'!(G(dead -> ("P.a==3" | "P.b==3")))'
|
||||||
|
|
||||||
|
run 0 ../modelcheck -ddead -e $srcdir/finite.gal \
|
||||||
|
'!(G(dead -> ("P.a==2" | "P.b==3")))'
|
||||||
|
|
||||||
|
# This used to segfault because of a bug in a
|
||||||
|
# function that do not exist anymore.
|
||||||
|
run 0 ../modelcheck -gp $srcdir/finite.gal true
|
||||||
Loading…
Add table
Add a link
Reference in a new issue