* 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.
149 lines
4.8 KiB
C++
149 lines
4.8 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/>.
|
|
|
|
#pragma once
|
|
|
|
#include <spot/twa/twagraph.hh>
|
|
|
|
namespace spot
|
|
{
|
|
/// \brief Clone and mask an automaton.
|
|
///
|
|
/// Copy the edges of automaton \a old, into automaton
|
|
/// \a cpy, creating new states at the same time. The argument \a
|
|
/// trans should behave as a function with the following prototype:
|
|
/// <code>
|
|
/// void (*trans) (unsigned srcbdd& cond, acc_cond::mark_t& acc,
|
|
/// unsigned dst)
|
|
/// </code>
|
|
/// It can modify either the condition or the acceptance sets of
|
|
/// the edges. Set the condition to bddfalse to remove the edge
|
|
/// (this will also remove the destination state and its descendants
|
|
/// if they are not reachable by another edge).
|
|
/// \param init The optional new initial state.
|
|
|
|
template<typename Trans>
|
|
void transform_accessible(const const_twa_graph_ptr& old,
|
|
twa_graph_ptr& cpy,
|
|
Trans trans, unsigned int init)
|
|
{
|
|
std::vector<unsigned> todo;
|
|
std::vector<unsigned> seen(old->num_states(), -1U);
|
|
|
|
auto new_state =
|
|
[&](unsigned old_state) -> unsigned
|
|
{
|
|
unsigned tmp = seen[old_state];
|
|
if (tmp == -1U)
|
|
{
|
|
tmp = cpy->new_state();
|
|
seen[old_state] = tmp;
|
|
todo.push_back(old_state);
|
|
}
|
|
return tmp;
|
|
};
|
|
|
|
cpy->set_init_state(new_state(init));
|
|
while (!todo.empty())
|
|
{
|
|
unsigned old_src = todo.back();
|
|
todo.pop_back();
|
|
|
|
unsigned new_src = seen[old_src];
|
|
assert(new_src != -1U);
|
|
|
|
for (auto& t: old->out(old_src))
|
|
{
|
|
bdd cond = t.cond;
|
|
acc_cond::mark_t acc = t.acc;
|
|
trans(t.src, cond, acc, t.dst);
|
|
|
|
if (cond != bddfalse)
|
|
cpy->new_edge(new_src,
|
|
new_state(t.dst),
|
|
cond, acc);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// \brief Copy an automaton and update each edge.
|
|
///
|
|
/// Copy the states of automaton \a old, into automaton
|
|
/// \a cpy. Each state in \a cpy will have the same id as the ones in \a old.
|
|
/// The argument \a trans
|
|
/// should behave as a function with the following prototype:
|
|
/// <code>
|
|
/// void (*trans) (unsigned srcbdd& cond, acc_cond::mark_t& acc,
|
|
/// unsigned dst)
|
|
/// </code>
|
|
/// It can modify either the condition or the acceptance sets of
|
|
/// the edges. Set the condition to bddfalse to remove it. Note that
|
|
/// all transtions will be processed.
|
|
/// \param init The optional new initial state.
|
|
template<typename Trans>
|
|
void transform_copy(const const_twa_graph_ptr& old,
|
|
twa_graph_ptr& cpy,
|
|
Trans trans, unsigned int init)
|
|
{
|
|
// Each state in cpy corresponds to a unique state in old.
|
|
cpy->new_states(old->num_states());
|
|
cpy->set_init_state(init);
|
|
|
|
for (auto& t: old->edges())
|
|
{
|
|
bdd cond = t.cond;
|
|
acc_cond::mark_t acc = t.acc;
|
|
trans(t.src, cond, acc, t.dst);
|
|
// Having the same number of states should assure that state ids are
|
|
// equivilent in old and cpy.
|
|
assert(t.src < cpy->num_states() && t.dst < cpy->num_states());
|
|
if (cond != bddfalse)
|
|
cpy->new_edge(t.src, t.dst, cond, acc);
|
|
}
|
|
}
|
|
|
|
template<typename Trans>
|
|
void transform_accessible(const const_twa_graph_ptr& old,
|
|
twa_graph_ptr& cpy,
|
|
Trans trans)
|
|
{
|
|
transform_accessible(old, cpy, trans, old->get_init_state_number());
|
|
}
|
|
template<typename Trans>
|
|
void transform_copy(const const_twa_graph_ptr& old,
|
|
twa_graph_ptr& cpy,
|
|
Trans trans)
|
|
{
|
|
transform_copy(old, cpy, trans, old->get_init_state_number());
|
|
}
|
|
|
|
/// \brief Remove all edges that belong to some given acceptance sets.
|
|
SPOT_API
|
|
twa_graph_ptr mask_acc_sets(const const_twa_graph_ptr& in,
|
|
acc_cond::mark_t to_remove);
|
|
|
|
/// \brief Keep only the states as specified by \a to_keep.
|
|
///
|
|
/// Each index in the vector \a to_keep specifies wether or not to keep that
|
|
/// state. The initial state will be set to \a init.
|
|
SPOT_API
|
|
twa_graph_ptr mask_keep_states(const const_twa_graph_ptr& in,
|
|
std::vector<bool>& to_keep,
|
|
unsigned int init);
|
|
}
|