* NEWS: Mention the change. * src/: Rename as ... * spot/: ... this, adjust all headers to include <spot/...> instead of "...", and adjust all Makefile.am to search headers from the top-level directory. * HACKING: Add conventions about #include. * spot/sanity/style.test: Add a few more grep to catch cases that do not follow these conventions. * .gitignore, Makefile.am, README, bench/stutter/Makefile.am, bench/stutter/stutter_invariance_formulas.cc, bench/stutter/stutter_invariance_randomgraph.cc, configure.ac, debian/rules, doc/Doxyfile.in, doc/Makefile.am, doc/org/.dir-locals.el.in, doc/org/g++wrap.in, doc/org/init.el.in, doc/org/tut01.org, doc/org/tut02.org, doc/org/tut03.org, doc/org/tut10.org, doc/org/tut20.org, doc/org/tut21.org, doc/org/tut22.org, doc/org/tut30.org, iface/ltsmin/Makefile.am, iface/ltsmin/kripke.test, iface/ltsmin/ltsmin.cc, iface/ltsmin/ltsmin.hh, iface/ltsmin/modelcheck.cc, wrap/python/Makefile.am, wrap/python/ajax/spotcgi.in, wrap/python/spot_impl.i, wrap/python/tests/ltl2tgba.py, wrap/python/tests/randgen.py, wrap/python/tests/run.in: 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 <spot/tl/exclusive.hh>
|
|
#include <spot/twaalgos/mask.hh>
|
|
#include <spot/misc/casts.hh>
|
|
#include <spot/misc/minato.hh>
|
|
#include <spot/tl/apcollect.hh>
|
|
|
|
namespace spot
|
|
{
|
|
namespace
|
|
{
|
|
static const std::vector<formula>
|
|
split_aps(const char* arg)
|
|
{
|
|
std::vector<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(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(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<formula> ap)
|
|
{
|
|
groups.push_back(ap);
|
|
}
|
|
|
|
namespace
|
|
{
|
|
formula
|
|
nand(formula lhs, formula rhs)
|
|
{
|
|
return formula::Not(formula::And({lhs, rhs}));
|
|
}
|
|
}
|
|
|
|
formula
|
|
exclusive_ap::constrain(formula f) const
|
|
{
|
|
auto* s = atomic_prop_collect(f);
|
|
|
|
std::vector<formula> group;
|
|
std::vector<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 formula::And({f, formula::G(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;
|
|
}
|
|
}
|