The ltl prefix does not make a lot of sens anymore (since we support psl as well). ltlast/ and ltlenv/ were almost empty. And ltlvisit/ did not contain any visitor anymore. * src/ltlvisit/, src/ltlast/, src/ltlenv/: Merge into... * src/tl/: ...this. * NEWS: Mention the change. * README, bench/stutter/stutter_invariance_formulas.cc, bench/stutter/stutter_invariance_randomgraph.cc, configure.ac, doc/org/tut01.org, doc/org/tut02.org, doc/org/tut10.org, doc/tl/tl.tex, iface/ltsmin/ltsmin.hh, iface/ltsmin/modelcheck.cc, src/Makefile.am, src/bin/autfilt.cc, src/bin/common_output.cc, src/bin/common_output.hh, src/bin/common_r.hh, src/bin/common_trans.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/randltl.cc, src/kripke/kripkeexplicit.hh, src/kripkeparse/public.hh, src/parseaut/public.hh, src/priv/accmap.hh, src/ta/taexplicit.hh, src/ta/tgtaexplicit.hh, src/tests/equalsf.cc, src/tests/ikwiad.cc, src/tests/length.cc, src/tests/ltlrel.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/twagraph.cc, src/twa/acc.hh, src/twa/bdddict.cc, src/twa/bdddict.hh, src/twa/bddprint.cc, src/twa/taatgba.cc, src/twa/taatgba.hh, src/twa/twa.hh, src/twa/twagraph.cc, src/twa/twagraph.hh, src/twa/twasafracomplement.cc, src/twaalgos/compsusp.cc, src/twaalgos/compsusp.hh, src/twaalgos/dtgbasat.cc, src/twaalgos/hoa.cc, src/twaalgos/isweakscc.cc, src/twaalgos/lbtt.cc, src/twaalgos/ltl2taa.cc, src/twaalgos/ltl2taa.hh, src/twaalgos/ltl2tgba_fm.cc, src/twaalgos/ltl2tgba_fm.hh, src/twaalgos/minimize.hh, src/twaalgos/neverclaim.cc, src/twaalgos/randomgraph.hh, src/twaalgos/relabel.hh, src/twaalgos/remprop.hh, src/twaalgos/stats.cc, src/twaalgos/stutter.cc, src/twaalgos/translate.hh, wrap/python/spot_impl.i, src/ltlparse/ltlparse.yy, src/ltlparse/public.hh: Adjust.
207 lines
4.7 KiB
C++
207 lines
4.7 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 "exclusive.hh"
|
|
#include "twaalgos/mask.hh"
|
|
#include "misc/casts.hh"
|
|
#include "misc/minato.hh"
|
|
#include "apcollect.hh"
|
|
|
|
namespace spot
|
|
{
|
|
namespace
|
|
{
|
|
static const std::vector<ltl::formula>
|
|
split_aps(const char* arg)
|
|
{
|
|
std::vector<ltl::formula> group;
|
|
auto start = arg;
|
|
while (*start)
|
|
{
|
|
while (*start == ' ' || *start == '\t')
|
|
++start;
|
|
if (!*start)
|
|
break;
|
|
if (*start == ',')
|
|
{
|
|
std::string s = "unexpected ',' in ";
|
|
s += arg;
|
|
throw std::invalid_argument(s);
|
|
}
|
|
if (*start == '"')
|
|
{
|
|
++start;
|
|
auto end = start;
|
|
while (*end && *end != '"')
|
|
{
|
|
if (*end == '\\')
|
|
++end;
|
|
++end;
|
|
}
|
|
if (!*end)
|
|
{
|
|
std::string s = "missing closing '\"' in ";
|
|
s += arg;
|
|
throw std::invalid_argument(s);
|
|
}
|
|
std::string ap(start, end - start);
|
|
group.emplace_back(ltl::formula::ap(ap));
|
|
do
|
|
++end;
|
|
while (*end == ' ' || *end == '\t');
|
|
if (*end && *end != ',')
|
|
{
|
|
std::string s = "unexpected character '";
|
|
s += *end;
|
|
s += "' in ";
|
|
s += arg;
|
|
throw std::invalid_argument(s);
|
|
}
|
|
if (*end == ',')
|
|
++end;
|
|
start = end;
|
|
}
|
|
else
|
|
{
|
|
auto end = start;
|
|
while (*end && *end != ',')
|
|
++end;
|
|
auto rend = end;
|
|
while (rend > start && (rend[-1] == ' ' || rend[-1] == '\t'))
|
|
--rend;
|
|
std::string ap(start, rend - start);
|
|
group.emplace_back(ltl::formula::ap(ap));
|
|
if (*end == ',')
|
|
start = end + 1;
|
|
else
|
|
break;
|
|
}
|
|
}
|
|
return group;
|
|
}
|
|
}
|
|
|
|
void exclusive_ap::add_group(const char* ap_csv)
|
|
{
|
|
add_group(split_aps(ap_csv));
|
|
}
|
|
|
|
void exclusive_ap::add_group(std::vector<ltl::formula> ap)
|
|
{
|
|
groups.push_back(ap);
|
|
}
|
|
|
|
namespace
|
|
{
|
|
ltl::formula
|
|
nand(ltl::formula lhs, ltl::formula rhs)
|
|
{
|
|
return ltl::formula::Not(ltl::formula::And({lhs, rhs}));
|
|
}
|
|
}
|
|
|
|
ltl::formula
|
|
exclusive_ap::constrain(ltl::formula f) const
|
|
{
|
|
auto* s = atomic_prop_collect(f);
|
|
|
|
std::vector<ltl::formula> group;
|
|
std::vector<ltl::formula> v;
|
|
|
|
for (auto& g: groups)
|
|
{
|
|
group.clear();
|
|
|
|
for (auto ap: g)
|
|
if (s->find(ap) != s->end())
|
|
group.push_back(ap);
|
|
|
|
unsigned s = group.size();
|
|
for (unsigned j = 0; j < s; ++j)
|
|
for (unsigned k = j + 1; k < s; ++k)
|
|
v.push_back(nand(group[j], group[k]));
|
|
};
|
|
|
|
delete s;
|
|
return ltl::formula::And({f, ltl::formula::G(ltl::formula::And(v))});
|
|
}
|
|
|
|
twa_graph_ptr exclusive_ap::constrain(const_twa_graph_ptr aut,
|
|
bool simplify_guards) const
|
|
{
|
|
// Compute the support of the automaton.
|
|
bdd support = bddtrue;
|
|
{
|
|
std::set<int> bdd_seen;
|
|
for (auto& t: aut->edges())
|
|
if (bdd_seen.insert(t.cond.id()).second)
|
|
support &= bdd_support(t.cond);
|
|
}
|
|
|
|
bdd restrict = bddtrue;
|
|
auto d = aut->get_dict();
|
|
|
|
std::vector<bdd> group;
|
|
for (auto& g: groups)
|
|
{
|
|
group.clear();
|
|
|
|
for (auto ap: g)
|
|
{
|
|
int v = d->has_registered_proposition(ap, aut);
|
|
if (v >= 0)
|
|
group.push_back(bdd_nithvar(v));
|
|
}
|
|
|
|
unsigned s = group.size();
|
|
for (unsigned j = 0; j < s; ++j)
|
|
for (unsigned k = j + 1; k < s; ++k)
|
|
restrict &= group[j] | group[k];
|
|
}
|
|
|
|
twa_graph_ptr res = make_twa_graph(aut->get_dict());
|
|
res->copy_ap_of(aut);
|
|
res->prop_copy(aut, { true, true, true, true });
|
|
res->copy_acceptance_of(aut);
|
|
if (simplify_guards)
|
|
{
|
|
transform_accessible(aut, res, [&](unsigned, bdd& cond,
|
|
acc_cond::mark_t&, unsigned)
|
|
{
|
|
minato_isop isop(cond & restrict,
|
|
cond | !restrict,
|
|
true);
|
|
bdd res = bddfalse;
|
|
bdd cube = bddfalse;
|
|
while ((cube = isop.next()) != bddfalse)
|
|
res |= cube;
|
|
cond = res;
|
|
});
|
|
}
|
|
else
|
|
{
|
|
transform_accessible(aut, res, [&](unsigned, bdd& cond,
|
|
acc_cond::mark_t&, unsigned)
|
|
{
|
|
cond &= restrict;
|
|
});
|
|
}
|
|
return res;
|
|
}
|
|
}
|