* NEWS: Mention it. * bench/stutter/stutter_invariance_formulas.cc, bench/stutter/stutter_invariance_randomgraph.cc, doc/mainpage.dox, doc/org/tut01.org, doc/org/tut02.org, doc/org/tut10.org, doc/tl/tl.tex, iface/ltsmin/ltsmin.cc, iface/ltsmin/ltsmin.hh, iface/ltsmin/modelcheck.cc, src/bin/autfilt.cc, src/bin/common_aoutput.cc, src/bin/common_aoutput.hh, src/bin/common_finput.cc, src/bin/common_finput.hh, src/bin/common_output.cc, src/bin/common_output.hh, src/bin/common_r.hh, src/bin/common_trans.cc, src/bin/common_trans.hh, src/bin/dstar2tgba.cc, src/bin/genltl.cc, src/bin/ltl2tgba.cc, src/bin/ltl2tgta.cc, src/bin/ltlcross.cc, src/bin/ltldo.cc, src/bin/ltlfilt.cc, src/bin/ltlgrind.cc, src/bin/randaut.cc, src/bin/randltl.cc, src/kripke/kripkeexplicit.cc, src/kripke/kripkeexplicit.hh, src/kripkeparse/kripkeparse.yy, src/kripkeparse/public.hh, src/ltlparse/fmterror.cc, src/ltlparse/ltlparse.yy, src/ltlparse/ltlscan.ll, src/ltlparse/parsedecl.hh, src/ltlparse/public.hh, src/parseaut/parseaut.yy, src/parseaut/public.hh, src/tests/checkpsl.cc, src/tests/checkta.cc, src/tests/complementation.cc, src/tests/consterm.cc, src/tests/emptchk.cc, src/tests/equalsf.cc, src/tests/ikwiad.cc, src/tests/kind.cc, src/tests/length.cc, src/tests/ltlprod.cc, src/tests/ltlrel.cc, src/tests/parse.test, src/tests/parse_print_test.cc, src/tests/randtgba.cc, src/tests/readltl.cc, src/tests/reduc.cc, src/tests/syntimpl.cc, src/tests/taatgba.cc, src/tests/tostring.cc, src/tests/tostring.test, src/tl/apcollect.cc, src/tl/apcollect.hh, src/tl/contain.cc, src/tl/contain.hh, src/tl/declenv.cc, src/tl/declenv.hh, src/tl/defaultenv.cc, src/tl/defaultenv.hh, src/tl/dot.cc, src/tl/dot.hh, src/tl/environment.hh, src/tl/exclusive.cc, src/tl/exclusive.hh, src/tl/formula.cc, src/tl/formula.hh, src/tl/length.cc, src/tl/length.hh, src/tl/mark.cc, src/tl/mark.hh, src/tl/mutation.cc, src/tl/mutation.hh, src/tl/nenoform.cc, src/tl/nenoform.hh, src/tl/print.cc, src/tl/print.hh, src/tl/randomltl.cc, src/tl/randomltl.hh, src/tl/relabel.cc, src/tl/relabel.hh, src/tl/remove_x.cc, src/tl/remove_x.hh, src/tl/simpfg.cc, src/tl/simpfg.hh, src/tl/simplify.cc, src/tl/simplify.hh, src/tl/snf.cc, src/tl/snf.hh, src/tl/unabbrev.cc, src/tl/unabbrev.hh, src/twa/bdddict.cc, src/twa/bdddict.hh, src/twa/bddprint.cc, src/twa/formula2bdd.cc, src/twa/formula2bdd.hh, src/twa/taatgba.cc, src/twa/taatgba.hh, src/twa/twa.hh, src/twa/twagraph.cc, src/twa/twagraph.hh, src/twaalgos/compsusp.cc, src/twaalgos/compsusp.hh, src/twaalgos/ltl2taa.cc, src/twaalgos/ltl2taa.hh, src/twaalgos/ltl2tgba_fm.cc, src/twaalgos/ltl2tgba_fm.hh, src/twaalgos/minimize.cc, src/twaalgos/minimize.hh, src/twaalgos/neverclaim.cc, src/twaalgos/postproc.cc, src/twaalgos/postproc.hh, src/twaalgos/powerset.cc, src/twaalgos/powerset.hh, src/twaalgos/randomgraph.cc, src/twaalgos/randomgraph.hh, src/twaalgos/relabel.cc, src/twaalgos/relabel.hh, src/twaalgos/remprop.cc, src/twaalgos/remprop.hh, src/twaalgos/stats.cc, src/twaalgos/stats.hh, src/twaalgos/stutter.cc, src/twaalgos/stutter.hh, src/twaalgos/translate.cc, src/twaalgos/translate.hh, wrap/python/spot_impl.i: Remove the ltl namespace.
250 lines
5.3 KiB
C++
250 lines
5.3 KiB
C++
// -*- coding: utf-8 -*-
|
|
// Copyright (C) 2015 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/>.
|
|
|
|
|
|
#include "unabbrev.hh"
|
|
|
|
namespace spot
|
|
{
|
|
unabbreviator::unabbreviator(const char* opt)
|
|
{
|
|
while (*opt)
|
|
switch (char c = *opt++)
|
|
{
|
|
case 'e':
|
|
re_e_ = true;
|
|
re_some_bool_ = true;
|
|
break;
|
|
case 'F':
|
|
re_f_ = true;
|
|
re_some_f_g_ = true;
|
|
break;
|
|
case 'G':
|
|
re_g_ = true;
|
|
re_some_f_g_ = true;
|
|
break;
|
|
case 'i':
|
|
re_i_ = true;
|
|
re_some_bool_ = true;
|
|
break;
|
|
case 'M':
|
|
re_m_ = true;
|
|
re_some_other_ = true;
|
|
break;
|
|
case 'R':
|
|
re_r_ = true;
|
|
re_some_other_ = true;
|
|
break;
|
|
case 'W':
|
|
re_w_ = true;
|
|
re_some_other_ = true;
|
|
break;
|
|
case '^':
|
|
re_xor_ = true;
|
|
re_some_bool_ = true;
|
|
break;
|
|
default:
|
|
throw std::runtime_error
|
|
(std::string("unknown unabbreviation option: ")
|
|
+ c);
|
|
}
|
|
}
|
|
|
|
formula unabbreviator::run(formula in)
|
|
{
|
|
auto entry = cache_.emplace(in, nullptr);
|
|
if (!entry.second)
|
|
return entry.first->second;
|
|
|
|
// Skip recursion whenever possible
|
|
bool no_boolean_rewrite = !re_some_bool_ || in.is_sugar_free_boolean();
|
|
bool no_f_g_rewrite = !re_some_f_g_ || in.is_sugar_free_ltl();
|
|
if (no_boolean_rewrite
|
|
&& (in.is_boolean() || (no_f_g_rewrite && !re_some_other_)))
|
|
return entry.first->second = in;
|
|
|
|
auto rec = [this](formula f)
|
|
{
|
|
return this->run(f);
|
|
};
|
|
|
|
formula out = in;
|
|
if (in.size() > 0)
|
|
out = in.map(rec);
|
|
|
|
switch (out.kind())
|
|
{
|
|
case op::ff:
|
|
case op::tt:
|
|
case op::eword:
|
|
case op::ap:
|
|
case op::Not:
|
|
case op::X:
|
|
case op::Closure:
|
|
case op::NegClosure:
|
|
case op::NegClosureMarked:
|
|
case op::EConcat:
|
|
case op::EConcatMarked:
|
|
case op::UConcat:
|
|
case op::U:
|
|
case op::Or:
|
|
case op::OrRat:
|
|
case op::And:
|
|
case op::AndRat:
|
|
case op::AndNLM:
|
|
case op::Concat:
|
|
case op::Fusion:
|
|
case op::Star:
|
|
case op::FStar:
|
|
break;
|
|
case op::F:
|
|
// F f = true U f
|
|
if (!re_f_)
|
|
break;
|
|
out = formula::U(formula::tt(), out[0]);
|
|
break;
|
|
case op::G:
|
|
// G f = false R f
|
|
// G f = f W false
|
|
// G f = !F!f
|
|
// G f = !(true U !f)
|
|
if (!re_g_)
|
|
break;
|
|
if (!re_r_)
|
|
{
|
|
out = formula::R(formula::ff(), out[0]);
|
|
break;
|
|
}
|
|
if (!re_w_)
|
|
{
|
|
out = formula::W(out[0], formula::ff());
|
|
break;
|
|
}
|
|
{
|
|
auto nc = formula::Not(out[0]);
|
|
if (!re_f_)
|
|
{
|
|
out = formula::Not(formula::F(nc));
|
|
break;
|
|
}
|
|
out = formula::Not(formula::U(formula::tt(), nc));
|
|
break;
|
|
}
|
|
case op::Xor:
|
|
// f1 ^ f2 == !(f1 <-> f2)
|
|
// f1 ^ f2 == (f1 & !f2) | (f2 & !f1)
|
|
if (!re_xor_)
|
|
break;
|
|
{
|
|
auto f1 = out[0];
|
|
auto f2 = out[1];
|
|
if (!re_e_)
|
|
{
|
|
out = formula::Not(formula::Equiv(f1, f2));
|
|
}
|
|
else
|
|
{
|
|
auto a = formula::And({f1, formula::Not(f2)});
|
|
auto b = formula::And({f2, formula::Not(f1)});
|
|
out = formula::Or({a, b});
|
|
}
|
|
}
|
|
break;
|
|
case op::Implies:
|
|
// f1 => f2 == !f1 | f2
|
|
if (!re_i_)
|
|
break;
|
|
out = formula::Or({formula::Not(out[0]), out[1]});
|
|
break;
|
|
case op::Equiv:
|
|
// f1 <=> f2 == (f1 & f2) | (!f1 & !f2)
|
|
if (!re_e_)
|
|
break;
|
|
{
|
|
auto f1 = out[0];
|
|
auto f2 = out[1];
|
|
auto nf1 = formula::Not(f1);
|
|
auto nf2 = formula::Not(f2);
|
|
auto term1 = formula::And({f1, f2});
|
|
auto term2 = formula::And({nf1, nf2});
|
|
out = formula::Or({term1, term2});
|
|
break;
|
|
}
|
|
case op::R:
|
|
// f1 R f2 = f2 W (f1 & f2)
|
|
// f1 R f2 = f2 U ((f1 & f2) | Gf2)
|
|
// f1 R f2 = f2 U ((f1 & f2) | !F!f2)
|
|
// f1 R f2 = f2 U ((f1 & f2) | !(1 U !f2))
|
|
if (!re_r_)
|
|
break;
|
|
{
|
|
auto f1 = out[0];
|
|
auto f2 = out[1];
|
|
auto f12 = formula::And({f1, f2});
|
|
if (!re_w_)
|
|
{
|
|
out = formula::W(f2, f12);
|
|
break;
|
|
}
|
|
auto gf2 = formula::G(f2);
|
|
if (re_g_)
|
|
gf2 = run(gf2);
|
|
out = formula::U(f2, formula::Or({f12, out}));
|
|
break;
|
|
}
|
|
case op::W:
|
|
// f1 W f2 = f2 R (f2 | f1)
|
|
// f1 W f2 = f1 U (f2 | G f1)
|
|
// f1 W f2 = f1 U (f2 | !F !f1)
|
|
// f1 W f2 = f1 U (f2 | !(1 U !f1))
|
|
if (!re_w_)
|
|
break;
|
|
{
|
|
auto f1 = out[0];
|
|
auto f2 = out[1];
|
|
if (!re_r_)
|
|
{
|
|
out = formula::R(f2, formula::Or({f2, f1}));
|
|
break;
|
|
}
|
|
auto gf1 = formula::G(f1);
|
|
if (re_g_)
|
|
gf1 = rec(gf1);
|
|
out = formula::U(f1, formula::Or({f2, out}));
|
|
break;
|
|
}
|
|
case op::M:
|
|
// f1 M f2 = f2 U (g2 & f1)
|
|
if (!re_m_)
|
|
break;
|
|
{
|
|
auto f2 = out[1];
|
|
out = formula::U(f2, formula::And({f2, out[0]}));
|
|
break;
|
|
}
|
|
}
|
|
return entry.first->second = out;
|
|
}
|
|
|
|
formula unabbreviate(formula in, const char* opt)
|
|
{
|
|
unabbreviator un(opt);
|
|
return un.run(in);
|
|
}
|
|
}
|