This helps working around missing C functions like strcasecmp that do not exist everywhere (e.g. on Cygwin), and for which lib/ supplies a replacement. Unfortunately we do not have such build in our current continuous integration suite, so we cannot easily detect files where such config.h inclusion would be useful. Therefore this patch simply makes it mandatory to include config.h in *.cc files. Including this in public *.hh file is currently forbidden. * spot/gen/automata.cc, spot/gen/formulas.cc, spot/kripke/fairkripke.cc, spot/kripke/kripke.cc, spot/ltsmin/ltsmin.cc, spot/misc/game.cc, spot/parseaut/fmterror.cc, spot/parsetl/fmterror.cc, spot/parsetl/parsetl.yy, spot/priv/bddalloc.cc, spot/priv/freelist.cc, spot/priv/satcommon.cc, spot/priv/trim.cc, spot/priv/weight.cc, spot/ta/ta.cc, spot/ta/taexplicit.cc, spot/ta/taproduct.cc, spot/ta/tgtaexplicit.cc, spot/ta/tgtaproduct.cc, spot/taalgos/dot.cc, spot/taalgos/emptinessta.cc, spot/taalgos/minimize.cc, spot/taalgos/reachiter.cc, spot/taalgos/statessetbuilder.cc, spot/taalgos/stats.cc, spot/taalgos/tgba2ta.cc, spot/tl/apcollect.cc, spot/tl/contain.cc, spot/tl/declenv.cc, spot/tl/defaultenv.cc, spot/tl/dot.cc, spot/tl/exclusive.cc, spot/tl/hierarchy.cc, spot/tl/length.cc, spot/tl/ltlf.cc, spot/tl/mark.cc, spot/tl/mutation.cc, spot/tl/nenoform.cc, spot/tl/print.cc, spot/tl/randomltl.cc, spot/tl/relabel.cc, spot/tl/remove_x.cc, spot/tl/simplify.cc, spot/tl/snf.cc, spot/tl/unabbrev.cc, spot/twa/acc.cc, spot/twa/bdddict.cc, spot/twa/bddprint.cc, spot/twa/formula2bdd.cc, spot/twa/taatgba.cc, spot/twa/twa.cc, spot/twa/twagraph.cc, spot/twa/twaproduct.cc, spot/twaalgos/aiger.cc, spot/twaalgos/alternation.cc, spot/twaalgos/are_isomorphic.cc, spot/twaalgos/bfssteps.cc, spot/twaalgos/canonicalize.cc, spot/twaalgos/cleanacc.cc, spot/twaalgos/cobuchi.cc, spot/twaalgos/complement.cc, spot/twaalgos/complete.cc, spot/twaalgos/compsusp.cc, spot/twaalgos/couvreurnew.cc, spot/twaalgos/cycles.cc, spot/twaalgos/degen.cc, spot/twaalgos/determinize.cc, spot/twaalgos/dot.cc, spot/twaalgos/dtbasat.cc, spot/twaalgos/dtwasat.cc, spot/twaalgos/dualize.cc, spot/twaalgos/emptiness.cc, spot/twaalgos/gtec/ce.cc, spot/twaalgos/gtec/gtec.cc, spot/twaalgos/gtec/sccstack.cc, spot/twaalgos/gtec/status.cc, spot/twaalgos/gv04.cc, spot/twaalgos/hoa.cc, spot/twaalgos/iscolored.cc, spot/twaalgos/isdet.cc, spot/twaalgos/isunamb.cc, spot/twaalgos/isweakscc.cc, spot/twaalgos/langmap.cc, spot/twaalgos/lbtt.cc, spot/twaalgos/ltl2taa.cc, spot/twaalgos/ltl2tgba_fm.cc, spot/twaalgos/magic.cc, spot/twaalgos/mask.cc, spot/twaalgos/minimize.cc, spot/twaalgos/neverclaim.cc, spot/twaalgos/parity.cc, spot/twaalgos/postproc.cc, spot/twaalgos/powerset.cc, spot/twaalgos/product.cc, spot/twaalgos/rabin2parity.cc, spot/twaalgos/randomgraph.cc, spot/twaalgos/randomize.cc, spot/twaalgos/reachiter.cc, spot/twaalgos/relabel.cc, spot/twaalgos/remfin.cc, spot/twaalgos/remprop.cc, spot/twaalgos/sbacc.cc, spot/twaalgos/sccfilter.cc, spot/twaalgos/sccinfo.cc, spot/twaalgos/se05.cc, spot/twaalgos/sepsets.cc, spot/twaalgos/simulation.cc, spot/twaalgos/split.cc, spot/twaalgos/stats.cc, spot/twaalgos/strength.cc, spot/twaalgos/stripacc.cc, spot/twaalgos/stutter.cc, spot/twaalgos/sum.cc, spot/twaalgos/tau03.cc, spot/twaalgos/tau03opt.cc, spot/twaalgos/totgba.cc, spot/twaalgos/toweak.cc, spot/twaalgos/translate.cc, spot/twaalgos/word.cc, tests/core/acc.cc, tests/core/bitvect.cc, tests/core/checkpsl.cc, tests/core/checkta.cc, tests/core/consterm.cc, tests/core/emptchk.cc, tests/core/equalsf.cc, tests/core/graph.cc, tests/core/ikwiad.cc, tests/core/intvcmp2.cc, tests/core/intvcomp.cc, tests/core/kind.cc, tests/core/kripkecat.cc, tests/core/length.cc, tests/core/ltlrel.cc, tests/core/ngraph.cc, tests/core/parity.cc, tests/core/randtgba.cc, tests/core/readltl.cc, tests/core/reduc.cc, tests/core/safra.cc, tests/core/sccif.cc, tests/core/syntimpl.cc, tests/core/taatgba.cc, tests/core/tostring.cc, tests/core/trival.cc, tests/core/twagraph.cc, tests/ltsmin/modelcheck.cc, spot/parseaut/scanaut.ll, spot/parsetl/scantl.ll: Include config.h. * spot/gen/Makefile.am, spot/graph/Makefile.am, spot/kripke/Makefile.am, spot/ltsmin/Makefile.am, spot/parseaut/Makefile.am, spot/parsetl/Makefile.am, spot/priv/Makefile.am, spot/ta/Makefile.am, spot/taalgos/Makefile.am, spot/tl/Makefile.am, spot/twa/Makefile.am, spot/twaalgos/Makefile.am, spot/twaalgos/gtec/Makefile.am, tests/Makefile.am: Add the -I lib/ flags. * tests/sanity/includes.test: Catch missing config.h in *.cc, and diagnose config.h in *.hh. * tests/sanity/style.test: Better diagnostics.
377 lines
10 KiB
C++
377 lines
10 KiB
C++
// -*- coding: utf-8 -*-
|
|
// Copyright (C) 2011-2018 Laboratoire de Recherche et Developpement
|
|
// 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/>.
|
|
|
|
#include "config.h"
|
|
#include <spot/ltsmin/ltsmin.hh>
|
|
#include <spot/twaalgos/dot.hh>
|
|
#include <spot/tl/defaultenv.hh>
|
|
#include <spot/tl/parse.hh>
|
|
#include <spot/twaalgos/translate.hh>
|
|
#include <spot/twaalgos/emptiness.hh>
|
|
#include <spot/twaalgos/postproc.hh>
|
|
#include <spot/twa/twaproduct.hh>
|
|
#include <spot/misc/timer.hh>
|
|
#include <spot/misc/memusage.hh>
|
|
#include <cstring>
|
|
#include <spot/kripke/kripkegraph.hh>
|
|
#include <spot/twaalgos/hoa.hh>
|
|
|
|
static void
|
|
syntax(char* prog)
|
|
{
|
|
// Display the supplied name unless it appears to be a libtool wrapper.
|
|
char* slash = strrchr(prog, '/');
|
|
if (slash && (strncmp(slash + 1, "lt-", 3) == 0))
|
|
prog = slash + 4;
|
|
|
|
std::cerr << "usage: " << prog << " [options] model formula\n\
|
|
\n\
|
|
Options:\n\
|
|
-dDEAD use DEAD as property for marking DEAD states\n\
|
|
(by default DEAD = true)\n\
|
|
-e[ALGO] run emptiness check, expect an accepting run\n\
|
|
-E[ALGO] run emptiness check, expect no accepting run\n\
|
|
-C compute an accepting run (Counterexample) if it exists\n\
|
|
-D favor a deterministic translation over a small transition\n\
|
|
-gf output the automaton of the formula in dot format\n\
|
|
-gm output the model state-space in dot format\n\
|
|
-gK output the model state-space in Kripke format\n\
|
|
-gp output the product state-space in dot format\n\
|
|
-T time the different phases of the execution\n\
|
|
-z compress states to handle larger models\n\
|
|
-Z compress states (faster) assuming all values in [0 .. 2^28-1]\n\
|
|
";
|
|
exit(1);
|
|
}
|
|
|
|
static int
|
|
checked_main(int argc, char **argv)
|
|
{
|
|
spot::timer_map tm;
|
|
|
|
bool use_timer = false;
|
|
|
|
enum { DotFormula, DotModel, DotProduct, EmptinessCheck, Kripke }
|
|
output = EmptinessCheck;
|
|
bool accepting_run = false;
|
|
bool expect_counter_example = false;
|
|
bool deterministic = false;
|
|
char *dead = nullptr;
|
|
int compress_states = 0;
|
|
|
|
const char* echeck_algo = "Cou99";
|
|
|
|
int dest = 1;
|
|
int n = argc;
|
|
for (int i = 1; i < n; ++i)
|
|
{
|
|
char* opt = argv[i];
|
|
if (*opt == '-')
|
|
{
|
|
switch (*++opt)
|
|
{
|
|
case 'C':
|
|
accepting_run = true;
|
|
break;
|
|
case 'd':
|
|
dead = opt + 1;
|
|
break;
|
|
case 'D':
|
|
deterministic = true;
|
|
break;
|
|
case 'e':
|
|
case 'E':
|
|
{
|
|
echeck_algo = opt + 1;
|
|
if (!*echeck_algo)
|
|
echeck_algo = "Cou99";
|
|
|
|
expect_counter_example = (*opt == 'e');
|
|
output = EmptinessCheck;
|
|
break;
|
|
}
|
|
case 'g':
|
|
switch (opt[1])
|
|
{
|
|
case 'm':
|
|
output = DotModel;
|
|
break;
|
|
case 'p':
|
|
output = DotProduct;
|
|
break;
|
|
case 'f':
|
|
output = DotFormula;
|
|
break;
|
|
case 'K':
|
|
output = Kripke;
|
|
break;
|
|
default:
|
|
goto error;
|
|
}
|
|
break;
|
|
case 'T':
|
|
use_timer = true;
|
|
break;
|
|
case 'z':
|
|
compress_states = 1;
|
|
break;
|
|
case 'Z':
|
|
compress_states = 2;
|
|
break;
|
|
default:
|
|
error:
|
|
std::cerr << "Unknown option `" << argv[i] << "'." << std::endl;
|
|
exit(1);
|
|
}
|
|
--argc;
|
|
}
|
|
else
|
|
{
|
|
argv[dest++] = argv[i];
|
|
}
|
|
}
|
|
|
|
if (argc != 3)
|
|
syntax(argv[0]);
|
|
|
|
spot::default_environment& env =
|
|
spot::default_environment::instance();
|
|
|
|
|
|
spot::atomic_prop_set ap;
|
|
auto dict = spot::make_bdd_dict();
|
|
spot::const_kripke_ptr model = nullptr;
|
|
spot::const_twa_ptr prop = nullptr;
|
|
spot::const_twa_ptr product = nullptr;
|
|
spot::emptiness_check_instantiator_ptr echeck_inst = nullptr;
|
|
int exit_code = 0;
|
|
spot::postprocessor post;
|
|
spot::formula deadf = nullptr;
|
|
spot::formula f = nullptr;
|
|
|
|
if (!dead || !strcasecmp(dead, "true"))
|
|
{
|
|
deadf = spot::formula::tt();
|
|
}
|
|
else if (!strcasecmp(dead, "false"))
|
|
{
|
|
deadf = spot::formula::ff();
|
|
}
|
|
else
|
|
{
|
|
deadf = env.require(dead);
|
|
}
|
|
|
|
if (output == EmptinessCheck)
|
|
{
|
|
const char* err;
|
|
echeck_inst = spot::make_emptiness_check_instantiator(echeck_algo, &err);
|
|
if (!echeck_inst)
|
|
{
|
|
std::cerr << "Failed to parse argument of -e/-E near `"
|
|
<< err << "'\n";
|
|
exit_code = 1;
|
|
goto safe_exit;
|
|
}
|
|
}
|
|
|
|
tm.start("parsing formula");
|
|
{
|
|
auto pf = spot::parse_infix_psl(argv[2], env, false);
|
|
exit_code = pf.format_errors(std::cerr);
|
|
f = pf.f;
|
|
}
|
|
tm.stop("parsing formula");
|
|
|
|
if (exit_code)
|
|
goto safe_exit;
|
|
|
|
tm.start("translating formula");
|
|
{
|
|
spot::translator trans(dict);
|
|
if (deterministic)
|
|
trans.set_pref(spot::postprocessor::Deterministic);
|
|
|
|
prop = trans.run(&f);
|
|
}
|
|
tm.stop("translating formula");
|
|
|
|
atomic_prop_collect(f, &ap);
|
|
|
|
if (output != DotFormula)
|
|
{
|
|
tm.start("loading ltsmin model");
|
|
try
|
|
{
|
|
model = spot::ltsmin_model::load(argv[1]).kripke(&ap, dict, deadf,
|
|
compress_states);
|
|
}
|
|
catch (std::runtime_error& e)
|
|
{
|
|
std::cerr << e.what() << '\n';
|
|
}
|
|
tm.stop("loading ltsmin model");
|
|
|
|
if (!model)
|
|
{
|
|
exit_code = 1;
|
|
goto safe_exit;
|
|
}
|
|
|
|
if (output == DotModel)
|
|
{
|
|
tm.start("dot output");
|
|
spot::print_dot(std::cout, model);
|
|
tm.stop("dot output");
|
|
goto safe_exit;
|
|
}
|
|
if (output == Kripke)
|
|
{
|
|
tm.start("kripke output");
|
|
spot::print_hoa(std::cout, model);
|
|
tm.stop("kripke output");
|
|
goto safe_exit;
|
|
}
|
|
}
|
|
|
|
if (output == DotFormula)
|
|
{
|
|
tm.start("dot output");
|
|
spot::print_dot(std::cout, prop);
|
|
tm.stop("dot output");
|
|
goto safe_exit;
|
|
}
|
|
|
|
product = spot::otf_product(model, prop);
|
|
|
|
if (output == DotProduct)
|
|
{
|
|
tm.start("dot output");
|
|
spot::print_dot(std::cout, product);
|
|
tm.stop("dot output");
|
|
goto safe_exit;
|
|
}
|
|
|
|
assert(echeck_inst);
|
|
|
|
{
|
|
auto ec = echeck_inst->instantiate(product);
|
|
bool search_many = echeck_inst->options().get("repeated");
|
|
assert(ec);
|
|
do
|
|
{
|
|
int memused = spot::memusage();
|
|
tm.start("running emptiness check");
|
|
spot::emptiness_check_result_ptr res;
|
|
try
|
|
{
|
|
res = ec->check();
|
|
}
|
|
catch (const std::bad_alloc&)
|
|
{
|
|
std::cerr << "Out of memory during emptiness check."
|
|
<< std::endl;
|
|
if (!compress_states)
|
|
std::cerr << "Try option -z for state compression." << std::endl;
|
|
exit_code = 2;
|
|
exit(exit_code);
|
|
}
|
|
tm.stop("running emptiness check");
|
|
memused = spot::memusage() - memused;
|
|
|
|
ec->print_stats(std::cout);
|
|
std::cout << memused << " pages allocated for emptiness check"
|
|
<< std::endl;
|
|
|
|
if (expect_counter_example == !res &&
|
|
(!expect_counter_example || ec->safe()))
|
|
exit_code = 1;
|
|
|
|
if (!res)
|
|
{
|
|
std::cout << "no accepting run found";
|
|
if (!ec->safe() && expect_counter_example)
|
|
{
|
|
std::cout << " even if expected" << std::endl;
|
|
std::cout << "this may be due to the use of the bit"
|
|
<< " state hashing technique" << std::endl;
|
|
std::cout << "you can try to increase the heap size "
|
|
<< "or use an explicit storage"
|
|
<< std::endl;
|
|
}
|
|
std::cout << std::endl;
|
|
break;
|
|
}
|
|
else if (accepting_run)
|
|
{
|
|
|
|
spot::twa_run_ptr run;
|
|
tm.start("computing accepting run");
|
|
try
|
|
{
|
|
run = res->accepting_run();
|
|
}
|
|
catch (const std::bad_alloc&)
|
|
{
|
|
std::cerr << "Out of memory while looking for counterexample."
|
|
<< std::endl;
|
|
exit_code = 2;
|
|
exit(exit_code);
|
|
}
|
|
tm.stop("computing accepting run");
|
|
|
|
if (!run)
|
|
{
|
|
std::cout << "an accepting run exists" << std::endl;
|
|
}
|
|
else
|
|
{
|
|
tm.start("reducing accepting run");
|
|
run = run->reduce();
|
|
tm.stop("reducing accepting run");
|
|
tm.start("printing accepting run");
|
|
std::cout << *run;
|
|
tm.stop("printing accepting run");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
std::cout << "an accepting run exists "
|
|
<< "(use -C to print it)" << std::endl;
|
|
}
|
|
}
|
|
while (search_many);
|
|
}
|
|
|
|
safe_exit:
|
|
if (use_timer)
|
|
tm.print(std::cout);
|
|
tm.reset_all(); // This helps valgrind.
|
|
return exit_code;
|
|
}
|
|
|
|
int
|
|
main(int argc, char **argv)
|
|
{
|
|
auto exit_code = checked_main(argc, argv);
|
|
|
|
// Additional checks to debug reference counts in formulas.
|
|
assert(spot::fnode::instances_check());
|
|
exit(exit_code);
|
|
}
|