spot/src/ltltest/randltl.cc
Alexandre Duret-Lutz 67f4e8b5ce Deprecate reduce() in favor of ltl_simplifier.
* src/ltlvisit/reduce.hh: Mark the file as obsolete.
(reduce): Declare this function as obsolete.
* src/ltlvisit/reduce.cc: Define SKIP_DEPRECATED_WARNING
so we can include reduce.hh.
* src/sanity/includes.test: Also use SKIP_DEPRECATED_WARNING
when compiling headers.
* iface/dve2/dve2check.cc,
src/ltltest/equals.cc, src/ltltest/randltl.cc,
src/ltltest/reduc.cc, src/tgbaalgos/ltl2tgba_fm.hh,
src/tgbaalgos/ltl2tgba_fm.cc, src/tgbatest/randtgba.cc,
wrap/python/ajax/spot.in, wrap/python/spot.i: Adjust
to use ltl_simplifier.
* src/tgbatest/ltl2tgba.cc: Adjust to use ltl_simplifier,
and replace -fr1...-fr7 options by a single -fr option.
* src/tgbatest/spotlbtt.test: Adjust -fr flags accordingly.
* src/tgbatest/reductgba.cc: Do not include reduce.hh.
2012-04-28 09:34:42 +02:00

364 lines
9.1 KiB
C++

// Copyright (C) 2008, 2009, 2011 Laboratoire de Recherche et Développement
// de l'Epita (LRDE).
// Copyright (C) 2003, 2005 Laboratoire d'Informatique de
// Paris 6 (LIP6), département Systèmes Répartis Coopératifs (SRC),
// Université Pierre et Marie Curie.
//
// 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 2 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 Spot; see the file COPYING. If not, write to the Free
// Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
// 02111-1307, USA.
#include <cassert>
#include <iostream>
#include <set>
#include <string>
#include <cstdlib>
#include <cstring>
#include "ltlast/atomic_prop.hh"
#include "ltlvisit/randomltl.hh"
#include "ltlvisit/tostring.hh"
#include "ltlvisit/length.hh"
#include "ltlvisit/simplify.hh"
#include "ltlenv/defaultenv.hh"
#include "misc/random.hh"
void
syntax(char* prog)
{
std::cerr << "Usage: "<< prog << " [OPTIONS...] PROPS..." << std::endl
<< std::endl
<< "Output selection:" << std::endl
<< " -B generate Boolean formulae" << std::endl
<< " -L generate LTL formulae [default]" << std::endl
<< " -S generate SERE" << std::endl
<< " -P generate PSL formulae" << std::endl
<< std::endl
<< "Options:" << std::endl
<< " -d dump priorities, do not generate any formula"
<< std::endl
<< " -f N the size of the formula [15]" << std::endl
<< " -F N number of formulae to generate [1]" << std::endl
<< " -pL S priorities to use for LTL formula" << std::endl
<< " -pS S priorities to use for SERE" << std::endl
<< " -pB S priorities to use for Boolean formulae" << std::endl
<< " -r N simplify formulae using all available reductions"
<< " and reject those" << std::endl
<< " strictly smaller than N" << std::endl
<< " -u generate unique formulae"
<< std::endl
<< " -s N seed for the random number generator" << std::endl
<< std::endl
<< "Where:" << std::endl
<< " F are floating values" << std::endl
<< " S are `KEY=F, KEY=F, ...' strings" << std::endl
<< " N are positive integers" << std::endl
<< " PROPS are the atomic properties to use on transitions"
<< std::endl
<< "Use -d to see the list of KEYs." << std::endl;
exit(2);
}
int
to_int(const char* s)
{
char* endptr;
int res = strtol(s, &endptr, 10);
if (*endptr)
{
std::cerr << "Failed to parse `" << s << "' as an integer." << std::endl;
exit(1);
}
return res;
}
int
main(int argc, char** argv)
{
enum { OutputBool, OutputLTL, OutputSERE, OutputPSL } output = OutputLTL;
bool opt_d = false;
int opt_s = 0;
int opt_f = 15;
int opt_F = 1;
char* opt_pL = 0;
char* opt_pS = 0;
char* opt_pB = 0;
int opt_r = 0;
bool opt_u = false;
spot::ltl::ltl_simplifier_options simpopt(true, true, true, true, true);
spot::ltl::ltl_simplifier simp(simpopt);
spot::ltl::environment& env(spot::ltl::default_environment::instance());
spot::ltl::atomic_prop_set* ap = new spot::ltl::atomic_prop_set;
int argn = 0;
if (argc <= 1)
syntax(argv[0]);
while (++argn < argc)
{
if (!strcmp(argv[argn], "-B"))
{
output = OutputBool;
}
else if (!strcmp(argv[argn], "-d"))
{
opt_d = true;
}
else if (!strcmp(argv[argn], "-f"))
{
if (argc < argn + 2)
syntax(argv[0]);
opt_f = to_int(argv[++argn]);
}
else if (!strcmp(argv[argn], "-F"))
{
if (argc < argn + 2)
syntax(argv[0]);
opt_F = to_int(argv[++argn]);
}
else if (!strcmp(argv[argn], "-L"))
{
output = OutputLTL;
}
else if ((!strcmp(argv[argn], "-p")) || (!strcmp(argv[argn], "-pL")))
{
if (argc < argn + 2)
syntax(argv[0]);
opt_pL = argv[++argn];
}
else if (!strcmp(argv[argn], "-pS"))
{
if (argc < argn + 2)
syntax(argv[0]);
opt_pS = argv[++argn];
}
else if (!strcmp(argv[argn], "-pB"))
{
if (argc < argn + 2)
syntax(argv[0]);
opt_pB = argv[++argn];
}
else if (!strcmp(argv[argn], "-P"))
{
output = OutputPSL;
}
else if (!strcmp(argv[argn], "-r"))
{
if (argc < argn + 2)
syntax(argv[0]);
opt_r = to_int(argv[++argn]);
}
else if (!strcmp(argv[argn], "-s"))
{
if (argc < argn + 2)
syntax(argv[0]);
opt_s = to_int(argv[++argn]);
}
else if (!strcmp(argv[argn], "-S"))
{
output = OutputSERE;
}
else if (!strcmp(argv[argn], "-u"))
{
opt_u = true;
}
else
{
ap->insert(static_cast<spot::ltl::atomic_prop*>
(env.require(argv[argn])));
}
}
spot::ltl::random_formula* rf = 0;
spot::ltl::random_psl* rp = 0;
spot::ltl::random_sere* rs = 0;
const char* tok_pL = 0;
const char* tok_pS = 0;
const char* tok_pB = 0;
switch (output)
{
case OutputLTL:
rf = new spot::ltl::random_ltl(ap);
tok_pL = rf->parse_options(opt_pL);
if (opt_pS)
{
std::cerr << "option -pS unsupported for LTL output" << std::endl;
exit(2);
}
if (opt_pB)
{
std::cerr << "option -pB unsupported for LTL output" << std::endl;
exit(2);
}
break;
case OutputBool:
rf = new spot::ltl::random_boolean(ap);
tok_pB = rf->parse_options(opt_pB);
if (opt_pL)
{
std::cerr << "option -pL unsupported for Boolean output" << std::endl;
exit(2);
}
if (opt_pS)
{
std::cerr << "option -pS unsupported for Boolean output" << std::endl;
exit(2);
}
break;
case OutputSERE:
rf = rs = new spot::ltl::random_sere(ap);
tok_pS = rf->parse_options(opt_pS);
if (opt_pL)
{
std::cerr << "option -pL unsupported for SERE output" << std::endl;
exit(2);
}
break;
case OutputPSL:
rf = rp = new spot::ltl::random_psl(ap);
rs = &rp->rs;
tok_pL = rp->parse_options(opt_pL);
tok_pS = rs->parse_options(opt_pS);
tok_pB = rs->rb.parse_options(opt_pB);
break;
}
if (tok_pL)
{
std::cerr << "failed to parse priorities (option -pL) near `"
<< tok_pL << "'" << std::endl;
exit(2);
}
if (tok_pS)
{
std::cerr << "failed to parse priorities (option -pS) near `"
<< tok_pS << "'" << std::endl;
exit(2);
}
if (tok_pB)
{
std::cerr << "failed to parse priorities (option -pB) near `"
<< tok_pB << "'" << std::endl;
exit(2);
}
if (opt_r > opt_f)
{
std::cerr << "-r's argument (" << opt_r << ") should not be larger than "
<< "-f's (" << opt_f << ")" << std::endl;
exit(2);
}
if (opt_d)
{
switch (output)
{
case OutputLTL:
std::cout << "Use option -pL to set the following LTL priorities:"
<< std::endl;
rf->dump_priorities(std::cout);
break;
case OutputBool:
std::cout << "Use option -pB to set the following Boolean "
<< "formula priorities:" << std::endl;
rf->dump_priorities(std::cout);
break;
case OutputPSL:
std::cout << "Use option -pL to set the following LTL priorities:"
<< std::endl;
rp->dump_priorities(std::cout);
// Fall through.
case OutputSERE:
std::cout << "Use option -pS to set the following SERE priorities:"
<< std::endl;
rs->dump_priorities(std::cout);
std::cout << "Use option -pB to set the following Boolean "
<< "formula priorities:" << std::endl;
rs->rb.dump_priorities(std::cout);
break;
}
}
else
{
std::set<std::string> unique;
while (opt_F--)
{
int max_tries_u = 1000;
while (max_tries_u--)
{
spot::srand(opt_s++);
spot::ltl::formula* f = 0;
int max_tries_r = 1000;
while (max_tries_r--)
{
f = rf->generate(opt_f);
if (opt_r)
{
spot::ltl::formula* g = simp.simplify(f);
f->destroy();
if (spot::ltl::length(g) < opt_r)
{
g->destroy();
continue;
}
f = g;
}
else
{
assert(spot::ltl::length(f) <= opt_f);
}
break;
}
if (max_tries_r < 0)
{
assert(opt_r);
std::cerr << "Failed to generate non-reducible formula "
<< "of size " << opt_r << " or more." << std::endl;
exit(2);
}
std::string txt = spot::ltl::to_string(f, false,
output == OutputSERE);
f->destroy();
if (!opt_u || unique.insert(txt).second)
{
std::cout << txt << std::endl;
break;
}
}
if (max_tries_u < 0)
{
assert(opt_u);
std::cerr << "Failed to generate another unique formula."
<< std::endl;
exit(2);
}
}
}
delete rf;
spot::ltl::atomic_prop_set::const_iterator i = ap->begin();
while (i != ap->end())
{
spot::ltl::atomic_prop_set::const_iterator j = i;
++i;
(*j)->destroy();
}
delete ap;
}