* spot/twaalgos/game.cc, spot/twaalgos/game.hh (game_info): Move... * spot/twaalgos/synthesis.cc, spot/twaalgos/synthesis.hh (synthesis_info): ... here, because this structure contains only synthesis stuff. Also rename "solver" to "algo" to match the ltlsynt option. (solve_game): Move the two argument version of this function here, since that's the only game-related generic function that use synthesis_info. * bin/ltlsynt.cc, tests/core/ltlsynt.test, tests/python/games.ipynb, tests/python/synthesis.py, NEWS: Adjust all uses and mentions.
1885 lines
61 KiB
C++
1885 lines
61 KiB
C++
// -*- coding: utf-8 -*-
|
||
// Copyright (C) 2020, 2021 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 "config.h"
|
||
|
||
#include <spot/misc/bddlt.hh>
|
||
#include <spot/misc/hash.hh>
|
||
#include <spot/misc/timer.hh>
|
||
#include <spot/twa/formula2bdd.hh>
|
||
#include <spot/twaalgos/degen.hh>
|
||
#include <spot/twaalgos/determinize.hh>
|
||
#include <spot/twaalgos/isdet.hh>
|
||
#include <spot/twaalgos/mealy_machine.hh>
|
||
#include <spot/twaalgos/sbacc.hh>
|
||
#include <spot/twaalgos/synthesis.hh>
|
||
#include <spot/twaalgos/translate.hh>
|
||
#include <spot/misc/minato.hh>
|
||
#include <spot/twaalgos/totgba.hh>
|
||
#include <spot/twaalgos/toparity.hh>
|
||
#include <spot/tl/parse.hh>
|
||
|
||
|
||
#include <algorithm>
|
||
|
||
// Helper function/structures for split_2step
|
||
namespace{
|
||
// Computes and stores the restriction
|
||
// of each cond to the input domain and the support
|
||
// This is useful as it avoids recomputation
|
||
// and often many conditions are mapped to the same
|
||
// restriction
|
||
struct small_cacher_t
|
||
{
|
||
//e to e_in and support
|
||
std::unordered_map<bdd, std::pair<bdd, bdd>, spot::bdd_hash> cond_hash_;
|
||
|
||
void fill(const spot::const_twa_graph_ptr& aut, bdd output_bdd)
|
||
{
|
||
cond_hash_.reserve(aut->num_edges()/5+1);
|
||
// 20% is about lowest number of different edge conditions
|
||
// for benchmarks taken from syntcomp
|
||
|
||
for (const auto& e : aut->edges())
|
||
{
|
||
// Check if stored
|
||
if (cond_hash_.find(e.cond) != cond_hash_.end())
|
||
continue;
|
||
|
||
cond_hash_[e.cond] =
|
||
std::pair<bdd, bdd>(
|
||
bdd_exist(e.cond, output_bdd),
|
||
bdd_exist(bdd_support(e.cond), output_bdd));
|
||
}
|
||
}
|
||
|
||
// Get the condition restricted to input and support of a condition
|
||
const std::pair<bdd, bdd>& operator[](const bdd& econd) const
|
||
{
|
||
return cond_hash_.at(econd);
|
||
}
|
||
};
|
||
|
||
// Struct to locally store the information of all outgoing edges
|
||
// of the state.
|
||
struct e_info_t
|
||
{
|
||
e_info_t(const spot::twa_graph::edge_storage_t& e,
|
||
const small_cacher_t& sm)
|
||
: dst(e.dst),
|
||
econd(e.cond),
|
||
einsup(sm[e.cond]),
|
||
acc(e.acc)
|
||
{
|
||
pre_hash = (spot::wang32_hash(dst)
|
||
^ std::hash<spot::acc_cond::mark_t>()(acc))
|
||
* spot::fnv<size_t>::prime;
|
||
}
|
||
|
||
inline size_t hash() const
|
||
{
|
||
return spot::wang32_hash(spot::bdd_hash()(econdout)) ^ pre_hash;
|
||
}
|
||
|
||
unsigned dst;
|
||
bdd econd;
|
||
mutable bdd econdout;
|
||
std::pair<bdd, bdd> einsup;
|
||
spot::acc_cond::mark_t acc;
|
||
size_t pre_hash;
|
||
};
|
||
// We define an order between the edges to avoid creating multiple
|
||
// states that in fact correspond to permutations of the order of the
|
||
// outgoing edges
|
||
struct less_info_t
|
||
{
|
||
// Note: orders via !econd!
|
||
inline bool operator()(const e_info_t &lhs, const e_info_t &rhs) const
|
||
{
|
||
const int l_id = lhs.econd.id();
|
||
const int r_id = rhs.econd.id();
|
||
return std::tie(lhs.dst, lhs.acc, l_id)
|
||
< std::tie(rhs.dst, rhs.acc, r_id);
|
||
}
|
||
}less_info;
|
||
|
||
struct less_info_ptr_t
|
||
{
|
||
// Note: orders via !econdout!
|
||
inline bool operator()(const e_info_t* lhs, const e_info_t* rhs)const
|
||
{
|
||
const int l_id = lhs->econdout.id();
|
||
const int r_id = rhs->econdout.id();
|
||
return std::tie(lhs->dst, lhs->acc, l_id)
|
||
< std::tie(rhs->dst, rhs->acc, r_id);
|
||
}
|
||
}less_info_ptr;
|
||
}
|
||
|
||
namespace spot
|
||
{
|
||
twa_graph_ptr
|
||
split_2step(const const_twa_graph_ptr& aut,
|
||
const bdd& output_bdd, bool complete_env)
|
||
{
|
||
auto split = make_twa_graph(aut->get_dict());
|
||
split->copy_ap_of(aut);
|
||
split->copy_acceptance_of(aut);
|
||
split->new_states(aut->num_states());
|
||
split->set_init_state(aut->get_init_state_number());
|
||
split->set_named_prop<bdd>("synthesis-output", new bdd(output_bdd));
|
||
|
||
auto [is_unsat, unsat_mark] = aut->acc().unsat_mark();
|
||
if (!is_unsat && complete_env)
|
||
throw std::runtime_error("split_2step(): Can not complete arena for "
|
||
"env as there is no unsat mark.");
|
||
|
||
bdd input_bdd = bddtrue;
|
||
{
|
||
bdd allbdd = aut->ap_vars();
|
||
while (allbdd != bddtrue)
|
||
{
|
||
bdd l = bdd_ithvar(bdd_var(allbdd));
|
||
if (not bdd_implies(output_bdd, l))
|
||
// Input
|
||
input_bdd &= l;
|
||
allbdd = bdd_high(allbdd);
|
||
assert(allbdd != bddfalse);
|
||
}
|
||
}
|
||
|
||
// The environment has all states
|
||
// with num <= aut->num_states();
|
||
// So we can first loop over the aut
|
||
// and then deduce the owner
|
||
|
||
// a sort of hash-map for all new intermediate states
|
||
std::unordered_multimap<size_t, unsigned> env_hash;
|
||
env_hash.reserve((int) (1.5 * aut->num_states()));
|
||
// a local map for edges leaving the current src
|
||
// this avoids creating and then combining edges for each minterm
|
||
// Note there are usually "few" edges leaving a state
|
||
// and map has shown to be faster than unordered_map for
|
||
// syntcomp examples
|
||
std::map<unsigned, std::pair<unsigned, bdd>> env_edge_hash;
|
||
typedef std::map<unsigned, std::pair<unsigned, bdd>>::mapped_type eeh_t;
|
||
|
||
small_cacher_t small_cacher;
|
||
small_cacher.fill(aut, output_bdd);
|
||
|
||
// Cache vector for all outgoing edges of this states
|
||
std::vector<e_info_t> e_cache;
|
||
|
||
// Vector of destinations actually reachable for a given
|
||
// minterm in ins
|
||
// Automatically "almost" sorted due to the sorting of e_cache
|
||
std::vector<const e_info_t*> dests;
|
||
|
||
// If a completion is demanded we might have to create sinks
|
||
// Sink controlled by player
|
||
unsigned sink_con = -1u;
|
||
unsigned sink_env = -1u;
|
||
auto get_sink_con_state = [&split, &sink_con, &sink_env,
|
||
um = unsat_mark]
|
||
(bool create = true)
|
||
{
|
||
if (SPOT_UNLIKELY((sink_con == -1u) && create))
|
||
{
|
||
sink_con = split->new_state();
|
||
sink_env = split->new_state();
|
||
split->new_edge(sink_con, sink_env, bddtrue, um);
|
||
split->new_edge(sink_env, sink_con, bddtrue, um);
|
||
}
|
||
return sink_con;
|
||
};
|
||
|
||
// Loop over all states
|
||
for (unsigned src = 0; src < aut->num_states(); ++src)
|
||
{
|
||
env_edge_hash.clear();
|
||
e_cache.clear();
|
||
|
||
auto out_cont = aut->out(src);
|
||
|
||
// Avoid looping over all minterms
|
||
// we only loop over the minterms that actually exist
|
||
bdd all_letters = bddfalse;
|
||
bdd support = bddtrue;
|
||
|
||
for (const auto& e : out_cont)
|
||
{
|
||
e_cache.emplace_back(e, small_cacher);
|
||
all_letters |= e_cache.back().einsup.first;
|
||
support &= e_cache.back().einsup.second;
|
||
}
|
||
// Complete for env
|
||
if (complete_env && (all_letters != bddtrue))
|
||
split->new_edge(src, get_sink_con_state(), bddtrue - all_letters);
|
||
|
||
// Sort to avoid that permutations of the same edges
|
||
// get different intermediate states
|
||
std::sort(e_cache.begin(), e_cache.end(), less_info);
|
||
|
||
for (auto one_letter : minterms_of(all_letters, input_bdd))
|
||
{
|
||
|
||
dests.clear();
|
||
for (const auto& e_info : e_cache)
|
||
// implies is faster than and
|
||
if (bdd_implies(one_letter, e_info.einsup.first))
|
||
{
|
||
e_info.econdout =
|
||
bdd_appex(e_info.econd, one_letter,
|
||
bddop_and, input_bdd);
|
||
dests.push_back(&e_info);
|
||
assert(e_info.econdout != bddfalse);
|
||
}
|
||
// By construction this should not be empty
|
||
assert(!dests.empty());
|
||
// # dests is almost sorted -> insertion sort
|
||
if (dests.size()>1)
|
||
for (auto it = dests.begin(); it != dests.end(); ++it)
|
||
std::rotate(std::upper_bound(dests.begin(), it, *it,
|
||
less_info_ptr),
|
||
it, it + 1);
|
||
|
||
bool to_add = true;
|
||
size_t h = fnv<size_t>::init;
|
||
for (const auto& t: dests)
|
||
h ^= t->hash();
|
||
|
||
auto range_h = env_hash.equal_range(h);
|
||
for (auto it_h = range_h.first; it_h != range_h.second; ++it_h)
|
||
{
|
||
unsigned i = it_h->second;
|
||
auto out = split->out(i);
|
||
if (std::equal(out.begin(), out.end(),
|
||
dests.begin(), dests.end(),
|
||
[](const twa_graph::edge_storage_t& x,
|
||
const e_info_t* y)
|
||
{
|
||
return x.dst == y->dst
|
||
&& x.acc == y->acc
|
||
&& x.cond.id() == y->econdout.id();
|
||
}))
|
||
{
|
||
to_add = false;
|
||
auto it = env_edge_hash.find(i);
|
||
if (it != env_edge_hash.end())
|
||
it->second.second |= one_letter;
|
||
else
|
||
env_edge_hash.emplace(i,
|
||
eeh_t(split->new_edge(src, i, bddtrue), one_letter));
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (to_add)
|
||
{
|
||
unsigned d = split->new_state();
|
||
unsigned n_e = split->new_edge(src, d, bddtrue);
|
||
env_hash.emplace(h, d);
|
||
env_edge_hash.emplace(d, eeh_t(n_e, one_letter));
|
||
for (const auto &t: dests)
|
||
split->new_edge(d, t->dst, t->econdout, t->acc);
|
||
}
|
||
} // letters
|
||
// save locally stored condition
|
||
for (const auto& elem : env_edge_hash)
|
||
split->edge_data(elem.second.first).cond = elem.second.second;
|
||
} // v-src
|
||
|
||
split->merge_edges();
|
||
split->prop_universal(trival::maybe());
|
||
|
||
// The named property
|
||
// compute the owners
|
||
// env is equal to false
|
||
std::vector<bool>* owner =
|
||
new std::vector<bool>(split->num_states(), false);
|
||
// All "new" states belong to the player
|
||
std::fill(owner->begin()+aut->num_states(), owner->end(), true);
|
||
// Check if sinks have been created
|
||
if (sink_env != -1u)
|
||
owner->at(sink_env) = false;
|
||
split->set_named_prop("state-player", owner);
|
||
|
||
// Done
|
||
return split;
|
||
}
|
||
|
||
void
|
||
split_2step_fast_here(const twa_graph_ptr& aut, const bdd& output_bdd)
|
||
{
|
||
struct dst_cond_color_t
|
||
{
|
||
std::pair<unsigned, int> dst_cond;
|
||
acc_cond::mark_t color;
|
||
};
|
||
|
||
auto hasher = [](const dst_cond_color_t& dcc) noexcept
|
||
{
|
||
return dcc.color.hash() ^ pair_hash()(dcc.dst_cond);
|
||
};
|
||
auto equal = [](const dst_cond_color_t& dcc1,
|
||
const dst_cond_color_t& dcc2) noexcept
|
||
{
|
||
return (dcc1.dst_cond == dcc2.dst_cond)
|
||
&& (dcc1.color == dcc2.color);
|
||
};
|
||
|
||
std::unordered_map<dst_cond_color_t, unsigned,
|
||
decltype(hasher),
|
||
decltype(equal)> player_map(aut->num_edges(),
|
||
hasher, equal);
|
||
|
||
auto get_ps = [&](unsigned dst, const bdd& ocond,
|
||
acc_cond::mark_t color)
|
||
{
|
||
dst_cond_color_t key{std::make_pair(dst, ocond.id()),
|
||
color};
|
||
auto [it, inserted] =
|
||
player_map.try_emplace(key, aut->num_states() + 1);
|
||
if (!inserted)
|
||
return it->second;
|
||
unsigned ns = aut->new_state();
|
||
assert(ns == it->second);
|
||
aut->new_edge(ns, dst, ocond, color);
|
||
return ns;
|
||
};
|
||
|
||
std::vector<unsigned> to_treat(aut->num_edges());
|
||
std::transform(aut->edges().begin(), aut->edges().end(),
|
||
to_treat.begin(), [&](const auto& e)
|
||
{ return aut->edge_number(e); });
|
||
|
||
std::for_each(to_treat.begin(), to_treat.end(),
|
||
[&](unsigned eidx)
|
||
{
|
||
const auto& e = aut->edge_storage(eidx);
|
||
bdd incond = bdd_exist(e.cond, output_bdd);
|
||
bdd outcond = bdd_existcomp(e.cond, output_bdd);
|
||
assert(((incond&outcond) == e.cond)
|
||
&& "Precondition violated");
|
||
// Modify
|
||
// Create new state and trans
|
||
unsigned new_dst = get_ps(e.dst, outcond, e.acc);
|
||
// redirect
|
||
aut->edge_storage(eidx).dst = new_dst;
|
||
aut->edge_storage(eidx).cond = incond;
|
||
});
|
||
|
||
auto* sp_ptr =
|
||
aut->get_or_set_named_prop<std::vector<bool>>("state-player");
|
||
|
||
sp_ptr->resize(aut->num_states());
|
||
std::fill(sp_ptr->begin(), sp_ptr->end(), false);
|
||
for (auto& eit : player_map)
|
||
(*sp_ptr)[eit.second] = true;
|
||
//Done
|
||
}
|
||
|
||
twa_graph_ptr
|
||
split_2step_fast(const const_twa_graph_ptr& aut, const bdd& output_bdd)
|
||
{
|
||
auto aut2 = make_twa_graph(aut, twa::prop_set::all());
|
||
aut2->copy_acceptance_of(aut);
|
||
aut2->set_named_prop<bdd>("synthesis-output", new bdd(output_bdd));
|
||
split_2step_fast_here(aut2, output_bdd);
|
||
return aut2;
|
||
}
|
||
|
||
twa_graph_ptr
|
||
unsplit_2step(const const_twa_graph_ptr& aut)
|
||
{
|
||
constexpr unsigned unseen_mark = std::numeric_limits<unsigned>::max();
|
||
|
||
twa_graph_ptr out = make_twa_graph(aut->get_dict());
|
||
out->copy_acceptance_of(aut);
|
||
out->copy_ap_of(aut);
|
||
|
||
// split_2step is guaranteed to produce an alternating arena
|
||
auto owner = get_state_players(aut);
|
||
#ifndef NDEGUB
|
||
(void) owner;
|
||
#endif
|
||
|
||
std::vector<unsigned> state_map(aut->num_states(), unseen_mark);
|
||
auto seen = [&](unsigned s){return state_map[s] != unseen_mark; };
|
||
auto map_s = [&](unsigned s)
|
||
{
|
||
if (!seen(s))
|
||
state_map[s] = out->new_state();
|
||
return state_map[s];
|
||
};
|
||
|
||
std::deque<unsigned> todo;
|
||
todo.push_back(aut->get_init_state_number());
|
||
map_s(aut->get_init_state_number());
|
||
while (!todo.empty())
|
||
{
|
||
unsigned cur = todo.front();
|
||
unsigned cur_m = map_s(cur);
|
||
todo.pop_front();
|
||
|
||
for (const auto& i : aut->out(cur))
|
||
for (const auto& o : aut->out(i.dst))
|
||
{
|
||
assert((owner.at(cur) != owner.at(o.src))
|
||
&& (owner.at(cur) == owner.at(o.dst)
|
||
&& "Arena is not alternating!"));
|
||
if (!seen(o.dst))
|
||
todo.push_back(o.dst);
|
||
out->new_edge(cur_m, map_s(o.dst),
|
||
i.cond & o.cond, i.acc | o.acc);
|
||
}
|
||
}
|
||
out->set_init_state(map_s(aut->get_init_state_number()));
|
||
// Try to set outputs
|
||
if (bdd* outptr = aut->get_named_prop<bdd>("synthesis-outputs"))
|
||
set_synthesis_outputs(out, *outptr);
|
||
return out;
|
||
}
|
||
|
||
// Improved apply strat, that reduces the number of edges/states
|
||
// while keeping the needed edge-properties
|
||
// Note, this only deals with deterministic strategies
|
||
// Note, assumes that env starts playing
|
||
spot::twa_graph_ptr
|
||
apply_strategy(const spot::twa_graph_ptr& arena,
|
||
bool unsplit, bool keep_acc)
|
||
{
|
||
const auto& win = get_state_winners(arena);
|
||
const auto& strat = get_strategy(arena);
|
||
const auto& sp = get_state_players(arena);
|
||
auto outs = get_synthesis_outputs(arena);
|
||
|
||
if (!win[arena->get_init_state_number()])
|
||
throw std::runtime_error("Player does not win initial state, strategy "
|
||
"is not applicable");
|
||
|
||
assert((sp[arena->get_init_state_number()] == false)
|
||
&& "Env needs to have first turn!");
|
||
|
||
assert(std::none_of(arena->edges().begin(), arena->edges().end(),
|
||
[&sp](const auto& e)
|
||
{ bool same_player = sp.at(e.src) == sp.at(e.dst);
|
||
if (same_player)
|
||
std::cerr << e.src << " and " << e.dst << " belong to same "
|
||
<< "player, arena not alternating!\n";
|
||
return same_player; }));
|
||
|
||
auto strat_split = spot::make_twa_graph(arena->get_dict());
|
||
strat_split->copy_ap_of(arena);
|
||
if (keep_acc)
|
||
strat_split->copy_acceptance_of(arena);
|
||
|
||
std::stack<unsigned> todo;
|
||
todo.push(arena->get_init_state_number());
|
||
|
||
struct dca{
|
||
unsigned dst;
|
||
unsigned condvar;
|
||
acc_cond::mark_t acc;
|
||
};
|
||
struct dca_hash
|
||
{
|
||
size_t operator()(const dca& d) const noexcept
|
||
{
|
||
return pair_hash()(std::make_pair(d.dst, d.condvar))
|
||
^ wang32_hash(d.acc.hash());
|
||
}
|
||
};
|
||
struct dca_equal
|
||
{
|
||
bool operator()(const dca& d1, const dca& d2) const noexcept
|
||
{
|
||
return std::tie(d1.dst, d1.condvar, d1.acc)
|
||
== std::tie(d2.dst, d2.condvar, d2.acc);
|
||
}
|
||
};
|
||
|
||
//env dst + player cond + acc -> p stat
|
||
std::unordered_map<dca,
|
||
unsigned,
|
||
dca_hash,
|
||
dca_equal> p_map;
|
||
|
||
constexpr unsigned unseen_mark = std::numeric_limits<unsigned>::max();
|
||
std::vector<unsigned> env_map(arena->num_states(), unseen_mark);
|
||
strat_split->set_init_state(strat_split->new_state());
|
||
env_map[arena->get_init_state_number()] =
|
||
strat_split->get_init_state_number();
|
||
|
||
// The states in the new graph are qualified local
|
||
// Get a local environment state
|
||
auto get_sel = [&](unsigned s)
|
||
{
|
||
if (SPOT_UNLIKELY(env_map[s] == unseen_mark))
|
||
env_map[s] = strat_split->new_state();
|
||
return env_map[s];
|
||
};
|
||
|
||
// local dst
|
||
// Check if there exists already a local player state
|
||
// that has the same dst, cond and (if keep_acc is true) acc
|
||
auto get_spl = [&](unsigned dst, const bdd& cond, acc_cond::mark_t acc)
|
||
{
|
||
dca d{dst, (unsigned) cond.id(), acc};
|
||
auto it = p_map.find(d);
|
||
if (it != p_map.end())
|
||
return it->second;
|
||
unsigned ns = strat_split->new_state();
|
||
p_map[d] = ns;
|
||
strat_split->new_edge(ns, dst, cond, acc);
|
||
return ns;
|
||
};
|
||
|
||
while (!todo.empty())
|
||
{
|
||
unsigned src_env = todo.top();
|
||
unsigned src_envl = get_sel(src_env);
|
||
todo.pop();
|
||
// All env edges
|
||
for (const auto& e_env : arena->out(src_env))
|
||
{
|
||
// Get the corresponding strat
|
||
const auto& e_strat = arena->edge_storage(strat[e_env.dst]);
|
||
// Check if already explored
|
||
if (env_map[e_strat.dst] == unseen_mark)
|
||
todo.push(e_strat.dst);
|
||
unsigned dst_envl = get_sel(e_strat.dst);
|
||
// The new env edge, player is constructed automatically
|
||
auto used_acc = keep_acc ? e_strat.acc : acc_cond::mark_t({});
|
||
strat_split->new_edge(src_envl,
|
||
get_spl(dst_envl, e_strat.cond, used_acc),
|
||
e_env.cond, used_acc);
|
||
}
|
||
}
|
||
// All states exists, we can try to further merge them
|
||
// Specialized merge
|
||
std::vector<bool> spnew(strat_split->num_states(), false);
|
||
for (const auto& p : p_map)
|
||
spnew[p.second] = true;
|
||
|
||
// Sorting edges in place
|
||
auto comp_edge = [](const auto& e1, const auto& e2)
|
||
{
|
||
return std::tuple(e1.dst, e1.acc, e1.cond.id())
|
||
< std::tuple(e2.dst, e2.acc, e2.cond.id());
|
||
};
|
||
auto sort_edges_of =
|
||
[&, &split_graph = strat_split->get_graph()](unsigned s)
|
||
{
|
||
static std::vector<unsigned> edge_nums;
|
||
edge_nums.clear();
|
||
|
||
auto eit = strat_split->out(s);
|
||
const auto eit_e = eit.end();
|
||
// 0 Check if it is already sorted
|
||
if (std::is_sorted(eit.begin(), eit_e, comp_edge))
|
||
return false; // No change
|
||
// 1 Get all edges numbers and sort them
|
||
std::transform(eit.begin(), eit_e, std::back_inserter(edge_nums),
|
||
[&](const auto& e)
|
||
{ return strat_split->edge_number(e); });
|
||
std::sort(edge_nums.begin(), edge_nums.end(),
|
||
[&](unsigned ne1, unsigned ne2)
|
||
{ return comp_edge(strat_split->edge_storage(ne1),
|
||
strat_split->edge_storage(ne2)); });
|
||
// 2 Correct the order
|
||
auto& sd = split_graph.state_storage(s);
|
||
sd.succ = edge_nums.front();
|
||
sd.succ_tail = edge_nums.back();
|
||
const unsigned n_edges_p = edge_nums.size()-1;
|
||
for (unsigned i = 0; i < n_edges_p; ++i)
|
||
split_graph.edge_storage(edge_nums[i]).next_succ = edge_nums[i+1];
|
||
split_graph.edge_storage(edge_nums.back()).next_succ = 0; //terminal
|
||
// All nicely chained
|
||
return true;
|
||
};
|
||
auto merge_edges_of = [&](unsigned s)
|
||
{
|
||
// Call this after sort edges of
|
||
// Mergeable edges are nicely chained
|
||
bool changed = false;
|
||
auto eit = strat_split->out_iteraser(s);
|
||
unsigned idx_last = strat_split->edge_number(*eit);
|
||
++eit;
|
||
while (eit)
|
||
{
|
||
auto& e_last = strat_split->edge_storage(idx_last);
|
||
if (std::tie(e_last.dst, e_last.acc)
|
||
== std::tie(eit->dst, eit->acc))
|
||
{
|
||
//Same dest and acc -> or condition
|
||
e_last.cond |= eit->cond;
|
||
eit.erase();
|
||
changed = true;
|
||
}
|
||
else
|
||
{
|
||
idx_last = strat_split->edge_number(*eit);
|
||
++eit;
|
||
}
|
||
}
|
||
return changed;
|
||
};
|
||
auto merge_able = [&](unsigned s1, unsigned s2)
|
||
{
|
||
auto eit1 = strat_split->out(s1);
|
||
auto eit2 = strat_split->out(s2);
|
||
// Note: No self-loops here
|
||
return std::equal(eit1.begin(), eit1.end(),
|
||
eit2.begin(), eit2.end(),
|
||
[](const auto& e1, const auto& e2)
|
||
{
|
||
return std::tuple(e1.dst, e1.acc, e1.cond.id())
|
||
== std::tuple(e2.dst, e2.acc, e2.cond.id());
|
||
});
|
||
};
|
||
|
||
const unsigned n_sstrat = strat_split->num_states();
|
||
std::vector<unsigned> remap(n_sstrat, -1u);
|
||
bool changed_any;
|
||
std::vector<unsigned> to_check;
|
||
to_check.reserve(n_sstrat);
|
||
// First iteration -> all env states are candidates
|
||
for (unsigned i = 0; i < n_sstrat; ++i)
|
||
if (!spnew[i])
|
||
to_check.push_back(i);
|
||
|
||
while (true)
|
||
{
|
||
changed_any = false;
|
||
std::for_each(to_check.begin(), to_check.end(),
|
||
[&](unsigned s){ sort_edges_of(s); merge_edges_of(s); });
|
||
// Check if we can merge env states
|
||
for (unsigned s1 : to_check)
|
||
for (unsigned s2 = 0; s2 < n_sstrat; ++s2)
|
||
{
|
||
if (spnew[s2] || (s1 == s2))
|
||
continue; // Player state or s1 == s2
|
||
if ((remap[s2] < s2))
|
||
continue; //s2 is already remapped
|
||
if (merge_able(s1, s2))
|
||
{
|
||
changed_any = true;
|
||
if (s1 < s2)
|
||
remap[s2] = s1;
|
||
else
|
||
remap[s1] = s2;
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (!changed_any)
|
||
break;
|
||
// Redirect changed targets and set possibly mergeable states
|
||
to_check.clear();
|
||
for (auto& e : strat_split->edges())
|
||
if (remap[e.dst] != -1u)
|
||
{
|
||
e.dst = remap[e.dst];
|
||
to_check.push_back(e.src);
|
||
assert(spnew[e.src]);
|
||
}
|
||
|
||
// Now check the player states
|
||
// We can skip sorting -> only one edge
|
||
// todo change when multistrat
|
||
changed_any = false;
|
||
for (unsigned s1 : to_check)
|
||
for (unsigned s2 = 0; s2 < n_sstrat; ++s2)
|
||
{
|
||
if (!spnew[s2] || (s1 == s2))
|
||
continue; // Env state or s1 == s2
|
||
if ((remap[s2] < s2))
|
||
continue; //s2 is already remapped
|
||
if (merge_able(s1, s2))
|
||
{
|
||
changed_any = true;
|
||
if (s1 < s2)
|
||
remap[s2] = s1;
|
||
else
|
||
remap[s1] = s2;
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (!changed_any)
|
||
break;
|
||
// Redirect changed targets and set possibly mergeable states
|
||
to_check.clear();
|
||
for (auto& e : strat_split->edges())
|
||
if (remap[e.dst] != -1u)
|
||
{
|
||
e.dst = remap[e.dst];
|
||
to_check.push_back(e.src);
|
||
assert(!spnew[e.src]);
|
||
}
|
||
}
|
||
|
||
// Defrag and alternate
|
||
if (remap[strat_split->get_init_state_number()] != -1u)
|
||
strat_split->set_init_state(remap[strat_split->get_init_state_number()]);
|
||
unsigned st = 0;
|
||
for (auto& s: remap)
|
||
if (s == -1U)
|
||
s = st++;
|
||
else
|
||
s = -1U;
|
||
|
||
strat_split->defrag_states(remap, st);
|
||
|
||
alternate_players(strat_split, false, false);
|
||
// What we do now depends on whether we unsplit or not
|
||
if (unsplit)
|
||
{
|
||
auto final = unsplit_2step(strat_split);
|
||
set_synthesis_outputs(final, outs);
|
||
return final;
|
||
}
|
||
// Keep the splitted version
|
||
set_state_winners(strat_split, region_t(strat_split->num_states(), true));
|
||
|
||
std::vector<unsigned> new_strat(strat_split->num_states(), -1u);
|
||
|
||
for (unsigned i = 0; i < strat_split->num_states(); ++i)
|
||
{
|
||
if (!sp[i])
|
||
continue;
|
||
new_strat[i] = strat_split->edge_number(*strat_split->out(i).begin());
|
||
}
|
||
set_strategy(strat_split, std::move(new_strat));
|
||
set_synthesis_outputs(strat_split, outs);
|
||
return strat_split;
|
||
}
|
||
|
||
std::ostream& operator<<(std::ostream& os, synthesis_info::algo s)
|
||
{
|
||
using algo = synthesis_info::algo;
|
||
const char* name = nullptr;
|
||
switch (s)
|
||
{
|
||
case (algo::DET_SPLIT):
|
||
name = "ds";
|
||
break;
|
||
case (algo::SPLIT_DET):
|
||
name = "sd";
|
||
break;
|
||
case (algo::DPA_SPLIT):
|
||
name = "ps";
|
||
break;
|
||
case (algo::LAR):
|
||
name = "lar";
|
||
break;
|
||
case (algo::LAR_OLD):
|
||
name = "lar.old";
|
||
break;
|
||
}
|
||
return os << name;
|
||
}
|
||
|
||
std::ostream&
|
||
operator<<(std::ostream& os, const synthesis_info& gi)
|
||
{
|
||
os << "force sbacc: " << gi.force_sbacc << '\n'
|
||
<< "solver: " << gi.s << '\n'
|
||
<< "minimization-lvl: " << gi.minimize_lvl << '\n'
|
||
<< (gi.verbose_stream ? "Is verbose\n" : "Is not verbose\n")
|
||
<< "the bdd_dict used is " << gi.dict.get();
|
||
return os;
|
||
}
|
||
|
||
|
||
namespace // Anonymous create_game
|
||
{
|
||
static translator
|
||
create_translator(synthesis_info& gi)
|
||
{
|
||
using algo = synthesis_info::algo;
|
||
|
||
option_map& extra_options = gi.opt;
|
||
auto sol = gi.s;
|
||
const bdd_dict_ptr& dict = gi.dict;
|
||
|
||
for (auto&& p : std::vector<std::pair<const char*, int>>
|
||
{{"simul", 0},
|
||
{"ba-simul", 0},
|
||
{"det-simul", 0},
|
||
{"tls-impl", 1},
|
||
{"wdba-minimize", 2}})
|
||
extra_options.set(p.first, extra_options.get(p.first, p.second));
|
||
|
||
translator trans(dict, &extra_options);
|
||
switch (sol)
|
||
{
|
||
case algo::LAR:
|
||
SPOT_FALLTHROUGH;
|
||
case algo::LAR_OLD:
|
||
trans.set_type(postprocessor::Generic);
|
||
trans.set_pref(postprocessor::Deterministic);
|
||
break;
|
||
case algo::DPA_SPLIT:
|
||
trans.set_type(postprocessor::ParityMaxOdd);
|
||
trans.set_pref(postprocessor::Deterministic | postprocessor::Colored);
|
||
break;
|
||
case algo::DET_SPLIT:
|
||
SPOT_FALLTHROUGH;
|
||
case algo::SPLIT_DET:
|
||
break;
|
||
}
|
||
return trans;
|
||
}
|
||
|
||
twa_graph_ptr
|
||
ntgba2dpa(const twa_graph_ptr& aut, bool force_sbacc)
|
||
{
|
||
// if the input automaton is deterministic, degeneralize it to be sure to
|
||
// end up with a parity automaton
|
||
auto dpa = tgba_determinize(degeneralize_tba(aut),
|
||
false, true, true, false);
|
||
dpa->merge_edges();
|
||
if (force_sbacc)
|
||
dpa = sbacc(dpa);
|
||
reduce_parity_here(dpa, true);
|
||
change_parity_here(dpa, parity_kind_max,
|
||
parity_style_odd);
|
||
assert((
|
||
[&dpa]() -> bool {
|
||
bool max, odd;
|
||
dpa->acc().is_parity(max, odd);
|
||
return max && odd;
|
||
}()));
|
||
assert(is_deterministic(dpa));
|
||
return dpa;
|
||
}
|
||
} // anonymous
|
||
|
||
twa_graph_ptr
|
||
create_game(const formula& f,
|
||
const std::vector<std::string>& all_outs,
|
||
synthesis_info& gi)
|
||
{
|
||
using algo = synthesis_info::algo;
|
||
|
||
[](std::vector<std::string> sv, std::string msg)
|
||
{
|
||
if (sv.size() < 2)
|
||
return;
|
||
std::sort(sv.begin(), sv.end());
|
||
const unsigned svs = sv.size() - 1;
|
||
for (unsigned i = 0; i < svs; ++i)
|
||
if (sv[i] == sv[i+1])
|
||
throw std::runtime_error(msg + sv[i]);
|
||
}(all_outs, "Output propositions are expected to appear once "
|
||
"in all_outs. Violating: ");
|
||
|
||
auto trans = create_translator(gi);
|
||
// Shortcuts
|
||
auto& bv = gi.bv;
|
||
auto& vs = gi.verbose_stream;
|
||
|
||
stopwatch sw;
|
||
|
||
if (bv)
|
||
sw.start();
|
||
auto aut = trans.run(f);
|
||
if (bv)
|
||
bv->trans_time += sw.stop();
|
||
|
||
if (vs)
|
||
{
|
||
assert(bv);
|
||
*vs << "translating formula done in "
|
||
<< bv->trans_time << " seconds\n";
|
||
*vs << "automaton has " << aut->num_states()
|
||
<< " states and " << aut->num_sets() << " colors\n";
|
||
}
|
||
auto tobdd = [&aut](const std::string& ap_name)
|
||
{
|
||
return bdd_ithvar(aut->register_ap(ap_name));
|
||
};
|
||
auto is_out = [&all_outs](const std::string& ao)->bool
|
||
{
|
||
return std::find(all_outs.begin(), all_outs.end(),
|
||
ao) != all_outs.end();
|
||
};
|
||
|
||
// Check for each ap that actually appears in the graph
|
||
// whether its an in or out
|
||
bdd ins = bddtrue;
|
||
bdd outs = bddtrue;
|
||
for (auto&& aap : aut->ap())
|
||
{
|
||
if (is_out(aap.ap_name()))
|
||
outs &= tobdd(aap.ap_name());
|
||
else
|
||
ins &= tobdd(aap.ap_name());
|
||
}
|
||
|
||
twa_graph_ptr dpa = nullptr;
|
||
|
||
switch (gi.s)
|
||
{
|
||
case algo::DET_SPLIT:
|
||
{
|
||
if (bv)
|
||
sw.start();
|
||
auto tmp = ntgba2dpa(aut, gi.force_sbacc);
|
||
if (vs)
|
||
*vs << "determinization done\nDPA has "
|
||
<< tmp->num_states() << " states, "
|
||
<< tmp->num_sets() << " colors\n";
|
||
tmp->merge_states();
|
||
if (bv)
|
||
bv->paritize_time += sw.stop();
|
||
if (vs)
|
||
*vs << "simplification done\nDPA has "
|
||
<< tmp->num_states() << " states\n"
|
||
<< "determinization and simplification took "
|
||
<< bv->paritize_time << " seconds\n";
|
||
if (bv)
|
||
sw.start();
|
||
dpa = split_2step(tmp, outs, true);
|
||
colorize_parity_here(dpa, true);
|
||
if (bv)
|
||
bv->split_time += sw.stop();
|
||
if (vs)
|
||
*vs << "split inputs and outputs done in " << bv->split_time
|
||
<< " seconds\nautomaton has "
|
||
<< tmp->num_states() << " states\n";
|
||
break;
|
||
}
|
||
case algo::DPA_SPLIT:
|
||
{
|
||
if (bv)
|
||
sw.start();
|
||
aut->merge_states();
|
||
if (bv)
|
||
bv->paritize_time += sw.stop();
|
||
if (vs)
|
||
*vs << "simplification done in " << bv->paritize_time
|
||
<< " seconds\nDPA has " << aut->num_states()
|
||
<< " states\n";
|
||
if (bv)
|
||
sw.start();
|
||
dpa = split_2step(aut, outs, true);
|
||
colorize_parity_here(dpa, true);
|
||
if (bv)
|
||
bv->split_time += sw.stop();
|
||
if (vs)
|
||
*vs << "split inputs and outputs done in " << bv->split_time
|
||
<< " seconds\nautomaton has "
|
||
<< dpa->num_states() << " states\n";
|
||
break;
|
||
}
|
||
case algo::SPLIT_DET:
|
||
{
|
||
sw.start();
|
||
auto split = split_2step(aut, outs, true);
|
||
if (bv)
|
||
bv->split_time += sw.stop();
|
||
if (vs)
|
||
*vs << "split inputs and outputs done in " << bv->split_time
|
||
<< " seconds\nautomaton has "
|
||
<< split->num_states() << " states\n";
|
||
if (bv)
|
||
sw.start();
|
||
dpa = ntgba2dpa(split, gi.force_sbacc);
|
||
if (vs)
|
||
*vs << "determinization done\nDPA has "
|
||
<< dpa->num_states() << " states, "
|
||
<< dpa->num_sets() << " colors\n";
|
||
dpa->merge_states();
|
||
if (bv)
|
||
bv->paritize_time += sw.stop();
|
||
if (vs)
|
||
*vs << "simplification done\nDPA has "
|
||
<< dpa->num_states() << " states\n"
|
||
<< "determinization and simplification took "
|
||
<< bv->paritize_time << " seconds\n";
|
||
// The named property "state-player" is set in split_2step
|
||
// but not propagated by ntgba2dpa
|
||
alternate_players(dpa);
|
||
break;
|
||
}
|
||
case algo::LAR:
|
||
SPOT_FALLTHROUGH;
|
||
case algo::LAR_OLD:
|
||
{
|
||
if (bv)
|
||
sw.start();
|
||
if (gi.s == algo::LAR)
|
||
{
|
||
dpa = to_parity(aut);
|
||
// reduce_parity is called by to_parity(),
|
||
// but with colorization turned off.
|
||
colorize_parity_here(dpa, true);
|
||
}
|
||
else
|
||
{
|
||
dpa = to_parity_old(aut);
|
||
dpa = reduce_parity_here(dpa, true);
|
||
}
|
||
change_parity_here(dpa, parity_kind_max, parity_style_odd);
|
||
if (bv)
|
||
bv->paritize_time += sw.stop();
|
||
if (vs)
|
||
*vs << "LAR construction done in " << bv->paritize_time
|
||
<< " seconds\nDPA has "
|
||
<< dpa->num_states() << " states, "
|
||
<< dpa->num_sets() << " colors\n";
|
||
|
||
if (bv)
|
||
sw.start();
|
||
dpa = split_2step(dpa, outs, true);
|
||
colorize_parity_here(dpa, true);
|
||
if (bv)
|
||
bv->split_time += sw.stop();
|
||
if (vs)
|
||
*vs << "split inputs and outputs done in " << bv->split_time
|
||
<< " seconds\nautomaton has "
|
||
<< dpa->num_states() << " states\n";
|
||
break;
|
||
}
|
||
} //end switch solver
|
||
// Set the named properties
|
||
assert(dpa->get_named_prop<std::vector<bool>>("state-player")
|
||
&& "DPA has no state-players");
|
||
set_synthesis_outputs(dpa, outs);
|
||
return dpa;
|
||
}
|
||
|
||
twa_graph_ptr
|
||
create_game(const formula& f,
|
||
const std::vector<std::string>& all_outs)
|
||
{
|
||
synthesis_info dummy;
|
||
return create_game(f, all_outs, dummy);
|
||
}
|
||
|
||
twa_graph_ptr
|
||
create_game(const std::string& f,
|
||
const std::vector<std::string>& all_outs)
|
||
{
|
||
return create_game(parse_formula(f), all_outs);
|
||
}
|
||
|
||
twa_graph_ptr
|
||
create_game(const std::string& f,
|
||
const std::vector<std::string>& all_outs,
|
||
synthesis_info& gi)
|
||
{
|
||
return create_game(parse_formula(f), all_outs, gi);
|
||
}
|
||
|
||
void
|
||
minimize_strategy_here(twa_graph_ptr& strat, int min_lvl)
|
||
{
|
||
if (!strat->acc().is_t())
|
||
throw std::runtime_error("minimize_strategy_here(): strategy is expected "
|
||
"to accept all runs! Otherwise it does not "
|
||
"correspond to a mealy machine and can not be "
|
||
"optimized this way.");
|
||
// 0 -> No minimization
|
||
// 1 -> Regular monitor/DFA minimization
|
||
// 2 -> Mealy minimization based on language inclusion
|
||
// with output assignement
|
||
// 3 -> Exact Mealy minimization using SAT
|
||
// 4 -> Monitor min then exact minimization
|
||
// 5 -> Mealy minimization based on language inclusion then exact
|
||
// minimization
|
||
bdd outs = get_synthesis_outputs(strat);
|
||
|
||
switch (min_lvl)
|
||
{
|
||
case 0:
|
||
return;
|
||
case 1:
|
||
{
|
||
minimize_mealy_fast_here(strat, false);
|
||
break;
|
||
}
|
||
case 2:
|
||
{
|
||
minimize_mealy_fast_here(strat, true);
|
||
break;
|
||
}
|
||
case 3:
|
||
{
|
||
strat = minimize_mealy(strat, -1);
|
||
break;
|
||
}
|
||
case 4:
|
||
{
|
||
strat = minimize_mealy(strat, 0);
|
||
break;
|
||
}
|
||
case 5:
|
||
{
|
||
strat = minimize_mealy(strat, 1);
|
||
break;
|
||
}
|
||
default:
|
||
throw std::runtime_error("Unknown minimization option");
|
||
}
|
||
set_synthesis_outputs(strat, outs);
|
||
}
|
||
|
||
twa_graph_ptr
|
||
minimize_strategy(const_twa_graph_ptr& strat, int min_lvl)
|
||
{
|
||
auto strat_dup = spot::make_twa_graph(strat, spot::twa::prop_set::all());
|
||
set_synthesis_outputs(strat_dup,
|
||
get_synthesis_outputs(strat));
|
||
if (auto sp_ptr = strat->get_named_prop<region_t>("state-player"))
|
||
set_state_players(strat_dup, *sp_ptr);
|
||
minimize_strategy_here(strat_dup, min_lvl);
|
||
return strat_dup;
|
||
}
|
||
|
||
twa_graph_ptr
|
||
create_strategy(twa_graph_ptr arena, synthesis_info& gi)
|
||
{
|
||
if (!arena)
|
||
throw std::runtime_error("Arena can not be null");
|
||
|
||
spot::stopwatch sw;
|
||
|
||
if (gi.bv)
|
||
sw.start();
|
||
|
||
if (!get_state_winner(arena, arena->get_init_state_number()))
|
||
return nullptr;
|
||
|
||
// If we use minimizations 0,1 or 2 -> unsplit
|
||
const bool do_unsplit = gi.minimize_lvl < 3;
|
||
twa_graph_ptr strat_aut = apply_strategy(arena, do_unsplit, false);
|
||
|
||
strat_aut->prop_universal(true);
|
||
|
||
minimize_strategy_here(strat_aut, gi.minimize_lvl);
|
||
|
||
assert(do_unsplit
|
||
|| strat_aut->get_named_prop<std::vector<bool>>("state-player"));
|
||
if (!do_unsplit)
|
||
strat_aut = unsplit_2step(strat_aut);
|
||
|
||
if (gi.bv)
|
||
{
|
||
gi.bv->strat2aut_time += sw.stop();
|
||
gi.bv->nb_strat_states += strat_aut->num_states();
|
||
gi.bv->nb_strat_edges += strat_aut->num_edges();
|
||
}
|
||
|
||
return strat_aut;
|
||
}
|
||
|
||
twa_graph_ptr
|
||
create_strategy(twa_graph_ptr arena)
|
||
{
|
||
synthesis_info dummy;
|
||
return create_strategy(arena, dummy);
|
||
}
|
||
|
||
}
|
||
|
||
namespace spot
|
||
{
|
||
//Anonymous for try_create_strat
|
||
namespace
|
||
{
|
||
class formula_2_inout_props
|
||
{
|
||
private:
|
||
std::map<formula, std::pair<std::set<formula>, std::set<formula>>> data_;
|
||
std::vector<std::string> outs_;
|
||
|
||
public:
|
||
|
||
formula_2_inout_props(std::vector<std::string> outs) : outs_(outs)
|
||
{}
|
||
|
||
std::pair<std::set<formula>, std::set<formula>>
|
||
aps_of(formula f)
|
||
{
|
||
auto cache_value = data_.find(f);
|
||
if (cache_value != data_.end())
|
||
return cache_value->second;
|
||
std::set<formula> ins_f, outs_f;
|
||
f.traverse([&](const formula& f)
|
||
{
|
||
if (f.is(op::ap))
|
||
{
|
||
auto cache_value =
|
||
std::find(outs_.begin(), outs_.end(), f.ap_name());
|
||
if (cache_value != outs_.end())
|
||
outs_f.insert(f);
|
||
else
|
||
ins_f.insert(f);
|
||
}
|
||
return false;
|
||
});
|
||
std::pair<std::set<formula>, std::set<formula>> res({ins_f, outs_f});
|
||
data_.insert({f, res});
|
||
return res;
|
||
}
|
||
};
|
||
}
|
||
|
||
strategy_like_t
|
||
try_create_direct_strategy(formula f,
|
||
const std::vector<std::string>& output_aps,
|
||
synthesis_info &gi)
|
||
{
|
||
formula_2_inout_props form2props(output_aps);
|
||
auto vs = gi.verbose_stream;
|
||
|
||
if (vs)
|
||
*vs << "trying to create strategy directly\n";
|
||
|
||
auto ret_sol_maybe = [&vs]()
|
||
{
|
||
if (vs)
|
||
*vs << "direct strategy might exist but was not found.\n";
|
||
return strategy_like_t{0, nullptr, bddfalse};
|
||
};
|
||
auto ret_sol_none = [&vs]()
|
||
{
|
||
if (vs)
|
||
*vs << "no direct strategy exists.\n";
|
||
return strategy_like_t{-1, nullptr, bddfalse};
|
||
};
|
||
auto ret_sol_exists = [&vs](auto strat)
|
||
{
|
||
if (vs)
|
||
{
|
||
*vs << "direct strategy was found.\n"
|
||
<< "direct strat has " << strat->num_states()
|
||
<< " states and " << strat->num_sets() << " colors\n";
|
||
}
|
||
return strategy_like_t{1, strat, bddfalse};
|
||
};
|
||
|
||
// We need a lot of look-ups
|
||
auto output_aps_set = std::set<std::string>(output_aps.begin(),
|
||
output_aps.end());
|
||
|
||
bool is_and = f.is(op::And);
|
||
formula f_g, f_equiv;
|
||
// Rewrite a conjunction as G(…) ∧ …
|
||
if (is_and)
|
||
{
|
||
if (f.size() != 2)
|
||
return ret_sol_maybe();
|
||
if (f[1].is(op::G))
|
||
f = formula::And({f[1], f[0]});
|
||
f_equiv = f[1];
|
||
f_g = f[0];
|
||
}
|
||
else
|
||
{
|
||
f_equiv = f;
|
||
f_g = spot::formula::tt();
|
||
}
|
||
|
||
if (!f_equiv.is(op::Equiv) || (!f_g.is_tt() && (!f_g.is(op::G)
|
||
|| !f_g[0].is_boolean())))
|
||
return ret_sol_maybe();
|
||
// stopwatch sw;
|
||
twa_graph_ptr res;
|
||
|
||
formula left = f_equiv[0],
|
||
right = f_equiv[1];
|
||
|
||
auto [left_ins, left_outs] = form2props.aps_of(left);
|
||
auto [right_ins, right_outs] = form2props.aps_of(right);
|
||
|
||
bool has_left_outs = !left_outs.empty();
|
||
bool has_left_ins = !left_ins.empty();
|
||
bool has_right_outs = !right_outs.empty();
|
||
bool has_right_ins = !right_ins.empty();
|
||
|
||
// Try to rewrite the equivalence as f(b1) <-> f(b2) where b2 has not any
|
||
// input
|
||
if (has_right_ins)
|
||
{
|
||
std::swap(left, right);
|
||
std::swap(has_left_ins, has_right_ins);
|
||
std::swap(has_left_outs, has_right_outs);
|
||
std::swap(left_ins, right_ins);
|
||
std::swap(left_outs, right_outs);
|
||
}
|
||
// We need to have f(inputs) <-> f(outputs)
|
||
if (has_right_ins || has_left_outs || !has_right_outs)
|
||
return ret_sol_maybe();
|
||
|
||
bool is_gf_bool_right = right.is({op::G, op::F});
|
||
bool is_fg_bool_right = right.is({op::F, op::G});
|
||
|
||
// Now we have to detect if we have persistence(ins) <-> FG(outs)
|
||
// or Büchi(ins) <-> GF(outs).
|
||
|
||
bool is_ok = ((is_gf_bool_right && left.is_syntactic_recurrence())
|
||
|| (is_fg_bool_right && left.is_syntactic_guarantee()));
|
||
|
||
// TODO: synthesis_info not updated
|
||
// TODO: Verbose
|
||
auto& bv = gi.bv;
|
||
stopwatch sw;
|
||
if (is_ok)
|
||
{
|
||
bool right_bool = right[0][0].is_boolean();
|
||
if (!right_bool)
|
||
return ret_sol_maybe();
|
||
auto trans = create_translator(gi);
|
||
trans.set_type(postprocessor::Buchi);
|
||
trans.set_pref(postprocessor::Deterministic | postprocessor::Complete);
|
||
|
||
|
||
auto right_sub = right[0][0];
|
||
|
||
if (bv)
|
||
sw.start();
|
||
res = trans.run(left);
|
||
if (bv)
|
||
{
|
||
auto delta = sw.stop();
|
||
bv->trans_time += delta;
|
||
if (vs)
|
||
*vs << "tanslating formula done in " << delta << " seconds\n";
|
||
}
|
||
|
||
for (auto& out : right_outs)
|
||
res->register_ap(out.ap_name());
|
||
if (!is_deterministic(res))
|
||
return ret_sol_maybe();
|
||
bdd form_bdd = bddtrue;
|
||
if (is_and)
|
||
{
|
||
bdd output_bdd = bddtrue;
|
||
for (auto &out : output_aps_set)
|
||
output_bdd &= bdd_ithvar(res->register_ap(out));
|
||
form_bdd = f_g.is_tt() ? (bdd) bddtrue :
|
||
formula_to_bdd(f_g[0],
|
||
res->get_dict(), res);
|
||
if (bdd_exist(form_bdd, output_bdd) != bddtrue)
|
||
return ret_sol_maybe();
|
||
}
|
||
bdd right_bdd = formula_to_bdd(right_sub, res->get_dict(), res);
|
||
bdd neg_right_bdd = bdd_not(right_bdd);
|
||
assert(right_ins.empty());
|
||
const bool is_true = res->acc().is_t();
|
||
scc_info si(res, scc_info_options::NONE);
|
||
for (auto& e : res->edges())
|
||
{
|
||
// Here the part describing the outputs is based on the fact that
|
||
// they must be seen infinitely often. As these edges are seen
|
||
// finitely often, we can let the minimization choose the value.
|
||
if (si.scc_of(e.src) == si.scc_of(e.dst))
|
||
{
|
||
if (e.acc || is_true)
|
||
e.cond &= (right_bdd);
|
||
else
|
||
e.cond &= (neg_right_bdd);
|
||
}
|
||
// form_bdd has to be true all the time. So we cannot only do it
|
||
// between SCCs.
|
||
if (!bdd_have_common_assignment(e.cond, form_bdd))
|
||
return ret_sol_none();
|
||
e.acc = {};
|
||
}
|
||
|
||
bdd output_bdd = bddtrue;
|
||
for (auto &out : output_aps_set)
|
||
output_bdd &= bdd_ithvar(res->register_ap(out));
|
||
|
||
set_synthesis_outputs(res, output_bdd);
|
||
res->set_acceptance(acc_cond::acc_code::t());
|
||
|
||
res->prop_complete(trival::maybe());
|
||
return ret_sol_exists(res);
|
||
}
|
||
return ret_sol_maybe();
|
||
}
|
||
|
||
} // spot
|
||
|
||
namespace // anonymous for subsformula
|
||
{
|
||
using namespace spot;
|
||
// Checks that 2 sets have a common element. Use it instead
|
||
// of set_intersection when we just want to check if they have a common
|
||
// element because it avoids going through the rest of the sets after an
|
||
// element is found.
|
||
static bool
|
||
are_intersecting(const std::set<formula> &v1,
|
||
const std::set<formula> &v2)
|
||
{
|
||
auto v1_pos = v1.begin(), v2_pos = v2.begin(), v1_end = v1.end(),
|
||
v2_end = v2.end();
|
||
while (v1_pos != v1_end && v2_pos != v2_end)
|
||
{
|
||
if (*v1_pos < *v2_pos)
|
||
++v1_pos;
|
||
else if (*v2_pos < *v1_pos)
|
||
++v2_pos;
|
||
else
|
||
return true;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
static std::pair<std::set<formula>, std::set<formula>>
|
||
algo4(const std::vector<formula> &assumptions,
|
||
const std::set<std::string> &outs,
|
||
formula_2_inout_props& form2props)
|
||
{
|
||
// Two variables are "connected" if they share an assumption.
|
||
// It creates a dependency graph and our goal is to find the propositions
|
||
// that are in the same connected components as outputs.
|
||
const auto ass_size = assumptions.size();
|
||
std::vector<bool> done(ass_size, false);
|
||
std::pair<std::set<formula>, std::set<formula>> result;
|
||
// // An output is in the result.
|
||
for (auto &o : outs)
|
||
result.second.insert(formula::ap(o));
|
||
std::stack<unsigned> todo;
|
||
unsigned first_free = 0;
|
||
|
||
auto next_free = [&]()
|
||
{
|
||
while (first_free < ass_size)
|
||
{
|
||
if (done[first_free])
|
||
{
|
||
++first_free;
|
||
continue;
|
||
}
|
||
auto f = assumptions[first_free];
|
||
auto o = form2props.aps_of(f).second;
|
||
if (!o.empty())
|
||
return true;
|
||
++first_free;
|
||
}
|
||
return false;
|
||
};
|
||
while (next_free())
|
||
{
|
||
todo.push(first_free);
|
||
while (!todo.empty())
|
||
{
|
||
unsigned current_index = todo.top();
|
||
todo.pop();
|
||
formula current_form = assumptions[current_index];
|
||
done[current_index] = true;
|
||
auto [ins_current, outs_current] = form2props.aps_of(current_form);
|
||
result.first.insert(ins_current.begin(), ins_current.end());
|
||
result.second.insert(outs_current.begin(), outs_current.end());
|
||
for (unsigned i = 0; i < ass_size; ++i)
|
||
{
|
||
if (done[i])
|
||
continue;
|
||
auto other_form = assumptions[i];
|
||
auto [ins_other, outs_other] = form2props.aps_of(other_form);
|
||
if (are_intersecting(ins_current, ins_other) ||
|
||
are_intersecting(outs_other, outs_other))
|
||
todo.push(i);
|
||
}
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
|
||
static formula
|
||
split_implication(const formula &f, const std::set<std::string> &outs,
|
||
formula_2_inout_props& form2props)
|
||
{
|
||
assert(f.is(op::Implies));
|
||
assert(f[0].is(op::And));
|
||
if (!(f[1].is(op::And)))
|
||
return f;
|
||
std::vector<formula> assumptions, guarantees;
|
||
for (auto a : f[0])
|
||
assumptions.push_back(a);
|
||
for (auto g : f[1])
|
||
guarantees.push_back(g);
|
||
// Set of input/output props that cannot be shared between
|
||
// subspecifications
|
||
auto decRelProps = algo4(assumptions, outs, form2props);
|
||
auto &decRelProps_ins = decRelProps.first;
|
||
auto &decRelProps_outs = decRelProps.second;
|
||
// Assumptions that don't contain an atomic proposition in decRelProps
|
||
auto free_assumptions = formula::tt();
|
||
// The set of subspecifications described as [(assum, guar), (assum, guar)]
|
||
std::vector<std::pair<formula, formula>> specs;
|
||
// We merge two assumpt or guar. that share a proposition from decRelProps
|
||
std::vector<formula> assumptions_split, guarantees_split;
|
||
|
||
auto fus = [&](std::vector<formula> &forms, std::vector<formula> &res)
|
||
{
|
||
std::stack<unsigned> todo;
|
||
todo.emplace(0);
|
||
unsigned first_free = 1;
|
||
const unsigned forms_size = forms.size();
|
||
std::vector<bool> done(forms_size, false);
|
||
while (!todo.empty())
|
||
{
|
||
auto current_res = formula::tt();
|
||
while (!todo.empty())
|
||
{
|
||
auto current_index = todo.top();
|
||
todo.pop();
|
||
done[current_index] = true;
|
||
auto current_form = forms[current_index];
|
||
current_res = formula::And({current_res, current_form});
|
||
auto [ins_f, outs_f] = form2props.aps_of(current_form);
|
||
std::set<formula> ins_f_dec, outs_f_dec;
|
||
std::set_intersection(ins_f.begin(), ins_f.end(),
|
||
decRelProps_ins.begin(),
|
||
decRelProps_ins.end(),
|
||
std::inserter(ins_f_dec,
|
||
ins_f_dec.begin()));
|
||
std::set_intersection(outs_f.begin(), outs_f.end(),
|
||
decRelProps_outs.begin(),
|
||
decRelProps_outs.end(),
|
||
std::inserter(ins_f_dec,
|
||
ins_f_dec.begin()));
|
||
for (unsigned i = 0; i < forms_size; ++i)
|
||
{
|
||
if (done[i])
|
||
continue;
|
||
auto [ins_i, outs_i] = form2props.aps_of(forms[i]);
|
||
if (are_intersecting(ins_i, ins_f_dec)
|
||
|| are_intersecting(outs_i, outs_f_dec))
|
||
todo.emplace(i);
|
||
}
|
||
}
|
||
res.push_back(current_res);
|
||
while (first_free < forms_size && done[first_free])
|
||
++first_free;
|
||
if (first_free < forms_size)
|
||
{
|
||
todo.emplace(first_free);
|
||
++first_free;
|
||
}
|
||
}
|
||
};
|
||
|
||
fus(assumptions, assumptions_split);
|
||
fus(guarantees, guarantees_split);
|
||
|
||
// Now we just have to find connected components in a bipartite graph
|
||
std::function<void(formula f, std::vector<formula> &,
|
||
std::vector<formula> &,
|
||
std::set<formula> &,
|
||
std::set<formula> &,
|
||
formula &, formula &,
|
||
std::vector<bool> &,
|
||
std::vector<bool> &)> bip;
|
||
bip = [&bip, &form2props](formula f, std::vector<formula> &src_vect,
|
||
std::vector<formula> &dst_vect,
|
||
std::set<formula> &ins_dec,
|
||
std::set<formula> &outs_dec,
|
||
formula &left_res, formula &right_res,
|
||
std::vector<bool> &done_left,
|
||
std::vector<bool> &done_right)
|
||
{
|
||
left_res = formula::And({left_res, f});
|
||
auto [ins_f, outs_f] = form2props.aps_of(f);
|
||
std::set<formula> f_ins_dec, f_outs_dec;
|
||
std::set_intersection(ins_f.begin(), ins_f.end(), ins_dec.begin(),
|
||
ins_dec.end(), std::inserter(f_ins_dec,
|
||
f_ins_dec.begin()));
|
||
std::set_intersection(outs_f.begin(), outs_f.end(), outs_dec.begin(),
|
||
outs_dec.end(),
|
||
std::inserter(f_outs_dec, f_outs_dec.begin()));
|
||
std::stack<unsigned> todo;
|
||
for (unsigned i = 0; i < dst_vect.size(); ++i)
|
||
{
|
||
if (done_right[i])
|
||
continue;
|
||
auto f2 = dst_vect[i];
|
||
auto [f2_ins, f2_outs] = form2props.aps_of(f2);
|
||
if (are_intersecting(f2_ins, f_ins_dec)
|
||
|| are_intersecting(f2_outs, f_outs_dec))
|
||
{
|
||
todo.push(i);
|
||
right_res = formula::And({right_res, f2});
|
||
done_right[i] = true;
|
||
}
|
||
}
|
||
while (!todo.empty())
|
||
{
|
||
auto right_index = todo.top();
|
||
todo.pop();
|
||
bip(dst_vect[right_index], dst_vect, src_vect, ins_dec, outs_dec,
|
||
right_res, left_res, done_right, done_left);
|
||
}
|
||
};
|
||
|
||
std::vector<bool> done_ass(assumptions_split.size(), false),
|
||
done_gua(guarantees_split.size(), false);
|
||
for (unsigned i = 0; i < assumptions_split.size(); ++i)
|
||
{
|
||
if (done_ass[i])
|
||
continue;
|
||
done_ass[i] = true;
|
||
auto &ass = assumptions_split[i];
|
||
auto [left_aps, right_aps] = form2props.aps_of(ass);
|
||
// If an assumption hasn't any decRelProp, it is considered as
|
||
// a free assumption.
|
||
if (!are_intersecting(left_aps, decRelProps_ins))
|
||
free_assumptions = formula::And({free_assumptions, ass});
|
||
else
|
||
{
|
||
auto left = formula::tt(), right = formula::tt();
|
||
bip(ass, assumptions_split, guarantees_split, decRelProps_ins,
|
||
decRelProps_outs, left, right, done_ass, done_gua);
|
||
specs.push_back({left, right});
|
||
}
|
||
}
|
||
|
||
for (unsigned i = 0; i < guarantees_split.size(); ++i)
|
||
if (!done_gua[i])
|
||
specs.push_back({formula::tt(), guarantees_split[i]});
|
||
|
||
if (!free_assumptions.is_tt())
|
||
for (auto &sp : specs)
|
||
sp.first = formula::And({sp.first, free_assumptions});
|
||
std::vector<formula> elems;
|
||
for (auto &[ass, gua] : specs)
|
||
{
|
||
auto new_impl = formula::Implies(ass, gua);
|
||
elems.push_back(new_impl);
|
||
}
|
||
return formula::And(elems);
|
||
}
|
||
|
||
static formula
|
||
extract_and(const formula& f, const std::set<std::string>& outs,
|
||
bool can_extract_impl, formula_2_inout_props& form2props)
|
||
{
|
||
if (f.is(op::And))
|
||
{
|
||
std::vector<formula> children;
|
||
for (auto fi : f)
|
||
children.push_back(
|
||
extract_and(fi, outs, can_extract_impl, form2props));
|
||
return formula::And(children);
|
||
}
|
||
if (f.is(op::Not))
|
||
{
|
||
auto child = extract_and(f[0], outs, false, form2props);
|
||
// ¬(⋀¬xᵢ) ≡ ⋁xᵢ
|
||
if (child.is(op::And))
|
||
{
|
||
bool ok = true;
|
||
for (auto sub : child)
|
||
if (!(sub.is(op::Not)))
|
||
{
|
||
ok = false;
|
||
break;
|
||
}
|
||
if (ok)
|
||
{
|
||
std::vector<formula> children;
|
||
for (auto fi : child)
|
||
children.push_back(
|
||
extract_and(formula::Not(fi), outs, false, form2props));
|
||
return formula::Or(children);
|
||
}
|
||
}
|
||
// ¬Fφ ≡ G¬φ
|
||
if (child.is(op::F))
|
||
{
|
||
// The result can be G(And).
|
||
auto f2 =
|
||
formula::G(
|
||
extract_and(formula::Not(child[0]), outs, false, form2props));
|
||
// What ?
|
||
return f2;
|
||
}
|
||
// ¬(φ→ψ) ≡ φ ∧ ¬ψ
|
||
else if (child.is(op::Implies))
|
||
{
|
||
return formula::And({
|
||
extract_and(child[0], outs, false, form2props),
|
||
extract_and(formula::Not(child[1]), outs, false, form2props)
|
||
});
|
||
}
|
||
// ¬(φ ∨ ψ) ≡ ¬φ ∧ ¬ψ
|
||
else if (child.is(op::Or))
|
||
{
|
||
std::vector<formula> children;
|
||
for (auto fi : child)
|
||
children.push_back(
|
||
extract_and(formula::Not(fi), outs, false, form2props));
|
||
return formula::And(children);
|
||
}
|
||
}
|
||
// G(⋀φᵢ) = ⋀(G(φᵢ))
|
||
// X(⋀φᵢ) = ⋀(X(φᵢ))
|
||
if (f.is(op::G, op::X))
|
||
{
|
||
auto child_ex = extract_and(f[0], outs, false, form2props);
|
||
if (child_ex.is(op::And))
|
||
{
|
||
std::vector<formula> children;
|
||
const auto f_kind = f.kind();
|
||
for (auto fi : child_ex)
|
||
children.push_back(
|
||
extract_and(
|
||
formula::unop(f_kind, fi), outs, false, form2props));
|
||
return formula::And(children);
|
||
}
|
||
}
|
||
// ⋀φᵢ U ψ ≡ ⋀(φᵢ U ψ)
|
||
if (f.is(op::U))
|
||
{
|
||
auto left_child_ex = extract_and(f[0], outs, false, form2props);
|
||
if (left_child_ex.is(op::And))
|
||
{
|
||
std::vector<formula> children;
|
||
for (auto fi : left_child_ex)
|
||
children.push_back(formula::U(fi, f[1]));
|
||
return formula::And(children);
|
||
}
|
||
}
|
||
if (f.is(op::Implies))
|
||
{
|
||
auto right_extr = extract_and(f[1], outs, false, form2props);
|
||
auto left_extr = extract_and(f[0], outs, false, form2props);
|
||
// φ → (⋀ψᵢ) ≡ ⋀(φ → ψᵢ)
|
||
if (!(left_extr.is(op::And)))
|
||
{
|
||
if (right_extr.is(op::And))
|
||
{
|
||
std::vector<formula> children;
|
||
for (auto fi : right_extr)
|
||
children.push_back(formula::Implies(left_extr, fi));
|
||
return formula::And(children);
|
||
}
|
||
}
|
||
// ⋀φᵢ → ⋀ψᵢ
|
||
else if (right_extr.is(op::And) && can_extract_impl)
|
||
{
|
||
auto extr_f = formula::Implies(left_extr, right_extr);
|
||
return split_implication(extr_f, outs, form2props);
|
||
}
|
||
}
|
||
return f;
|
||
}
|
||
|
||
}
|
||
|
||
namespace spot
|
||
{
|
||
std::pair<std::vector<formula>, std::vector<std::set<formula>>>
|
||
split_independant_formulas(formula f, const std::vector<std::string>& outs)
|
||
{
|
||
formula_2_inout_props form2props(outs);
|
||
std::set<std::string> outs_set(outs.begin(), outs.end());
|
||
|
||
f = extract_and(f, outs_set, true, form2props);
|
||
if (!(f.is(op::And)))
|
||
return { {f}, { form2props.aps_of(f).second } };
|
||
// Atomics prop of children
|
||
std::vector<std::set<formula>> children_outs;
|
||
// Independent formulas
|
||
std::vector<formula> res;
|
||
// And the appearing propositions
|
||
std::vector<std::set<formula>> res_outs;
|
||
// For each conj, we calculate the set of output AP
|
||
for (auto child : f)
|
||
children_outs.push_back(form2props.aps_of(child).second);
|
||
unsigned nb_children = f.size();
|
||
// flag to test if a conj is in a "class"
|
||
std::vector<bool> children_class(nb_children, false);
|
||
// The first element not in a class
|
||
unsigned first_free = 0;
|
||
std::stack<unsigned> todo;
|
||
while (first_free < nb_children)
|
||
{
|
||
todo.emplace(first_free);
|
||
std::vector<formula> current_and;
|
||
std::set<formula> current_outs;
|
||
while (!todo.empty())
|
||
{
|
||
auto current = todo.top();
|
||
todo.pop();
|
||
children_class[current] = true;
|
||
current_and.push_back(f[current]);
|
||
current_outs.insert(children_outs[current].begin(),
|
||
children_outs[current].end());
|
||
for (unsigned i = 0; i < nb_children; ++i)
|
||
if (!children_class[i]
|
||
&& are_intersecting(children_outs[current],
|
||
children_outs[i]))
|
||
{
|
||
todo.emplace(i);
|
||
children_class[i] = true;
|
||
}
|
||
}
|
||
auto elem = formula::And(current_and);
|
||
res.push_back(elem);
|
||
res_outs.push_back(current_outs);
|
||
|
||
while (first_free < nb_children && children_class[first_free])
|
||
++first_free;
|
||
}
|
||
assert(res.size() == res_outs.size());
|
||
|
||
return {res, res_outs};
|
||
}
|
||
|
||
std::pair<std::vector<formula>, std::vector<std::set<formula>>>
|
||
split_independant_formulas(const std::string& f,
|
||
const std::vector<std::string>& outs)
|
||
{
|
||
return split_independant_formulas(parse_formula(f), outs);
|
||
}
|
||
|
||
bool
|
||
solve_game(twa_graph_ptr arena, synthesis_info& gi)
|
||
{
|
||
stopwatch sw;
|
||
if (gi.bv)
|
||
sw.start();
|
||
if (gi.verbose_stream)
|
||
{
|
||
*(gi.verbose_stream) << "solving game with acceptance: ";
|
||
std::string name = arena->acc().name();
|
||
if (!name.empty())
|
||
*(gi.verbose_stream) << name;
|
||
else
|
||
*(gi.verbose_stream) << arena->get_acceptance();
|
||
*(gi.verbose_stream) << '\n';
|
||
}
|
||
bool res = solve_game(arena);
|
||
if (gi.bv)
|
||
gi.bv->solve_time += sw.stop();
|
||
if (gi.verbose_stream)
|
||
*(gi.verbose_stream) << "game solved in "
|
||
<< gi.bv->solve_time << " seconds\n";
|
||
return res;
|
||
}
|
||
|
||
} // spot
|