// -*- coding: utf-8 -*- // Copyright (C) 2012-2017 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 . #include #include #include namespace spot { namespace { [[noreturn]] static void invalid_scc_number(const char* fn) { throw std::invalid_argument(std::string(fn) + "(): invalid SCC number"); } } bool scc_has_rejecting_cycle(scc_info& map, unsigned scc) { if (SPOT_UNLIKELY(scc >= map.scc_count())) invalid_scc_number("scc_has_rejecting_cycle"); auto aut = map.get_aut(); // We check that by cloning the SCC and complementing its // acceptance condition. std::vector keep(aut->num_states(), false); auto& states = map.states_of(scc); for (auto s: states) keep[s] = true; auto sccaut = mask_keep_accessible_states(aut, keep, states.front(), /* drop_univ_branch = */ true); sccaut->set_acceptance(sccaut->acc().num_sets(), sccaut->get_acceptance().complement()); return !sccaut->is_empty(); } bool is_inherently_weak_scc(scc_info& map, unsigned scc) { if (SPOT_UNLIKELY(scc >= map.scc_count())) invalid_scc_number("is_inherently_weak_scc"); // Weak SCCs are inherently weak. if (is_weak_scc(map, scc)) return true; // If we reach this place, we now the SCC has an accepting cycle. // The question is now to find whether is also contains a // rejecting cycle. return !scc_has_rejecting_cycle(map, scc); } bool is_weak_scc(scc_info& map, unsigned scc) { if (SPOT_UNLIKELY(scc >= map.scc_count())) invalid_scc_number("is_weak_scc"); // Rejecting SCCs are weak. if (map.is_rejecting_scc(scc)) return true; // If all transitions use the same acceptance set, the SCC is weak. return map.used_acc_of(scc).size() == 1; } bool is_complete_scc(scc_info& map, unsigned scc) { if (SPOT_UNLIKELY(scc >= map.scc_count())) invalid_scc_number("is_complete_scc"); auto a = map.get_aut(); for (auto s: map.states_of(scc)) { bool has_succ = false; bdd sumall = bddfalse; for (auto& t: a->out(s)) { has_succ = true; bool in = true; for (auto d: a->univ_dests(t.dst)) if (map.scc_of(d) != scc) { in = false; break; } if (!in) continue; sumall |= t.cond; if (sumall == bddtrue) break; } if (!has_succ || sumall != bddtrue) return false; } return true; } bool is_terminal_scc(scc_info& map, unsigned scc) { if (SPOT_UNLIKELY(scc >= map.scc_count())) invalid_scc_number("is_terminal_scc"); // If all transitions use all acceptance conditions, the SCC is weak. return (map.is_accepting_scc(scc) && map.used_acc_of(scc).size() == 1 && is_complete_scc(map, scc)); } }