// -*- coding: utf-8 -*- // Copyright (C) 2014 Laboratoire de Recherche et Développement de // l'Epita. // // 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 . #ifndef SPOT_TGBA_TGBAGRAPH_HH # define SPOT_TGBA_TGBAGRAPH_HH #include "graph/graph.hh" #include "graph/ngraph.hh" #include "tgba/bdddict.hh" #include "tgba/tgba.hh" #include "misc/bddop.hh" #include namespace spot { struct SPOT_API tgba_graph_state: public spot::state { public: virtual ~tgba_graph_state() noexcept { } virtual int compare(const spot::state* other) const { auto o = down_cast(other); assert(o); // Do not simply return "other - this", it might not fit in an int. if (o < this) return -1; if (o > this) return 1; return 0; } virtual size_t hash() const { return reinterpret_cast(this) - static_cast(nullptr); } virtual tgba_graph_state* clone() const { return const_cast(this); } virtual void destroy() const { } }; struct SPOT_API tgba_graph_trans_data { bdd cond; bdd acc; explicit tgba_graph_trans_data() : cond(bddfalse), acc(bddfalse) { } tgba_graph_trans_data(bdd cond, bdd acc = bddfalse) : cond(cond), acc(acc) { } }; template class tgba_digraph_succ_iterator: public tgba_succ_iterator { private: typedef typename Graph::transition transition; typedef typename Graph::state_data_t state; const Graph* g_; transition t_; transition p_; public: tgba_digraph_succ_iterator(const Graph* g, transition t) : g_(g), t_(t) { } virtual void recycle(transition t) { t_ = t; } virtual bool first() { p_ = t_; return p_; } virtual bool next() { p_ = g_->trans_storage(p_).next_succ; return p_; } virtual bool done() const { return !p_; } virtual tgba_graph_state* current_state() const { assert(!done()); return const_cast (&g_->state_data(g_->trans_storage(p_).dst)); } virtual bdd current_condition() const { assert(!done()); return g_->trans_data(p_).cond; } virtual bdd current_acceptance_conditions() const { assert(!done()); return g_->trans_data(p_).acc; } transition pos() const { return p_; } }; class tgba_digraph: public tgba { public: typedef digraph graph_t; protected: graph_t g_; bdd_dict* dict_; bdd all_acceptance_conditions_; bdd neg_acceptance_conditions_; mutable const tgba_graph_state* init_; public: tgba_digraph(bdd_dict* dict) : dict_(dict), all_acceptance_conditions_(bddfalse), neg_acceptance_conditions_(bddtrue), init_(nullptr) { } virtual ~tgba_digraph() { dict_->unregister_all_my_variables(this); // Prevent this state from being destroyed by ~tgba(), // as the state will be destroyed when g_ is destroyed. last_support_conditions_input_ = 0; } // FIXME: Once we ditch GCC 4.6, we can using a template aliases // (supported from GCC 4.7 onward) instead of this. template , typename Name_Equal = std::equal_to> struct namer { typedef named_graph type; }; template , typename Name_Equal = std::equal_to> typename namer::type* create_namer() { return new named_graph(g_); } graph_t& get_graph() { return g_; } const graph_t& get_graph() const { return g_; } virtual bdd_dict* get_dict() const { return this->dict_; } unsigned num_states() const { return g_.num_states(); } unsigned num_transitions() const { return g_.num_transitions(); } void set_init_state(graph_t::state s) { init_ = &g_.state_data(s); } void set_init_state(const state* s) { init_ = down_cast(s); assert(init_); } virtual tgba_graph_state* get_init_state() const { if (num_states() == 0) const_cast(g_).new_state(); if (!init_) init_ = &g_.state_data(0); return const_cast(init_); } virtual graph_t::state get_init_state_number() const { if (num_states() == 0) const_cast(g_).new_state(); if (!init_) return 0; return state_number(init_); } virtual tgba_succ_iterator* succ_iter(const state* st) const { auto s = down_cast(st); assert(s); assert(!s->succ || g_.valid_trans(s->succ)); if (this->iter_cache_) { auto it = down_cast*>(this->iter_cache_); it->recycle(s->succ); this->iter_cache_ = nullptr; return it; } return new tgba_digraph_succ_iterator(&g_, s->succ); } graph_t::state state_number(const state* st) const { auto s = down_cast(st); assert(s); return s - &g_.state_storage(0); } const state* state_from_number(graph_t::state n) const { return &g_.state_data(n); } virtual std::string format_state(const state* st) const { std::stringstream ss; ss << state_number(st); return ss.str(); } tgba_graph_trans_data& trans_data(const tgba_succ_iterator* it) { auto* i = down_cast*>(it); return g_.trans_data(i->pos()); } tgba_graph_trans_data& trans_data(unsigned t) { return g_.trans_data(t); } void set_acceptance_conditions(bdd all) { bdd sup = bdd_support(all); this->dict_->register_acceptance_variables(sup, this); neg_acceptance_conditions_ = bddtrue; while (sup != bddtrue) { neg_acceptance_conditions_ &= bdd_nithvar(bdd_var(sup)); sup = bdd_high(sup); } all_acceptance_conditions_ = compute_all_acceptance_conditions(neg_acceptance_conditions_); } unsigned new_state() { return g_.new_state(); } unsigned new_states(unsigned n) { return g_.new_states(n); } unsigned new_transition(unsigned src, unsigned dst, bdd cond, bdd acc = bddfalse) { return g_.new_transition(src, dst, cond, acc); } internal::state_out out(unsigned src) const { return g_.out(src); } internal::state_out out(unsigned src) { return g_.out(src); } /// \brief Copy the acceptance conditions of another tgba. void copy_acceptance_conditions_of(const tgba *a) { set_acceptance_conditions(a->neg_acceptance_conditions()); } virtual bdd all_acceptance_conditions() const { return all_acceptance_conditions_; } virtual bdd neg_acceptance_conditions() const { return neg_acceptance_conditions_; } virtual bdd compute_support_conditions(const state*) const { return bddtrue; } /// Iterate over all transitions, and merge those with compatible /// extremities and acceptance. void merge_transitions() { for (auto& s: g_.states()) { // Map a pair (dest state, acc) to the first transition seen // with such characteristic. typedef std::pair key_t; std::unordered_map trmap; auto t = g_.out_iteraser(s); while (t) { // Simply skip false transitions. if (t->cond == bddfalse) { t.erase(); continue; } key_t k(t->dst, t->acc.id()); auto p = trmap.emplace(k, t.trans()); if (!p.second) { // A previous transitions exist for k, merge the // condition, and schedule the transition for // removal. g_.trans_data(p.first->second).cond |= t->cond; t.erase(); } else { ++t; } } } g_.defrag(); } }; } #endif // SPOT_TGBA_TGBAGRAPH_HH