Browse Source

allow external ltl2da tools

tempestpy_adaptions
hannah 4 years ago
committed by Stefan Pranger
parent
commit
3f030e23f0
  1. 68
      src/storm/automata/LTL2DeterministicAutomaton.cpp
  2. 8
      src/storm/automata/LTL2DeterministicAutomaton.h

68
src/storm/automata/LTL2DeterministicAutomaton.cpp

@ -5,6 +5,9 @@
#include "storm/utility/macros.h" #include "storm/utility/macros.h"
#include "storm/exceptions/ExpressionEvaluationException.h" #include "storm/exceptions/ExpressionEvaluationException.h"
#include "storm/exceptions/WrongFormatException.h" #include "storm/exceptions/WrongFormatException.h"
#include "storm/exceptions/FileIoException.h"
#include <sys/wait.h>
#ifdef STORM_HAVE_SPOT #ifdef STORM_HAVE_SPOT
#include "spot/tl/formula.hh" #include "spot/tl/formula.hh"
@ -19,10 +22,13 @@
namespace storm { namespace storm {
namespace automata { namespace automata {
std::shared_ptr<DeterministicAutomaton> LTL2DeterministicAutomaton::ltl2da(storm::logic::Formula const& f, bool dnf) {
bool LTL2DeterministicAutomaton::isExternalDaToolSet() {
return std::getenv("LTL2DA");
}
std::shared_ptr<DeterministicAutomaton> LTL2DeterministicAutomaton::ltl2daInternalTool(std::string const& prefixLtl, bool dnf) {
#ifdef STORM_HAVE_SPOT #ifdef STORM_HAVE_SPOT
std::string prefixLtl = f.toPrefixString();
spot::parsed_formula spotPrefixLtl = spot::parse_prefix_ltl(prefixLtl); spot::parsed_formula spotPrefixLtl = spot::parse_prefix_ltl(prefixLtl);
if(!spotPrefixLtl.errors.empty()){ if(!spotPrefixLtl.errors.empty()){
std::ostringstream errorMsg; std::ostringstream errorMsg;
@ -34,12 +40,11 @@ namespace storm {
// Request a deterministic, complete automaton with state-based acceptance // Request a deterministic, complete automaton with state-based acceptance
spot::translator trans = spot::translator(); spot::translator trans = spot::translator();
trans.set_type(spot::postprocessor::Generic); //Generic
trans.set_type(spot::postprocessor::Generic);
trans.set_pref(spot::postprocessor::Deterministic | spot::postprocessor::SBAcc | spot::postprocessor::Complete); trans.set_pref(spot::postprocessor::Deterministic | spot::postprocessor::SBAcc | spot::postprocessor::Complete);
STORM_LOG_INFO("Construct deterministic automaton for "<< spotFormula); STORM_LOG_INFO("Construct deterministic automaton for "<< spotFormula);
auto aut = trans.run(spotFormula); auto aut = trans.run(spotFormula);
// TODO necessary for MDP LTL-MC
if(!(aut->get_acceptance().is_dnf()) && dnf){ if(!(aut->get_acceptance().is_dnf()) && dnf){
STORM_LOG_INFO("Convert acceptance condition "<< aut->get_acceptance() << " into DNF..."); STORM_LOG_INFO("Convert acceptance condition "<< aut->get_acceptance() << " into DNF...");
// Transform the acceptance condition in disjunctive normal form and merge all the Fin-sets of each clause // Transform the acceptance condition in disjunctive normal form and merge all the Fin-sets of each clause
@ -61,5 +66,60 @@ namespace storm {
STORM_LOG_THROW(false, storm::exceptions::NotSupportedException, "Storm is compiled without Spot support."); STORM_LOG_THROW(false, storm::exceptions::NotSupportedException, "Storm is compiled without Spot support.");
#endif #endif
} }
// TODO: this is quite hacky, improve robustness
std::shared_ptr<DeterministicAutomaton> LTL2DeterministicAutomaton::ltl2daExternalTool(std::string const& prefixLtl) {
std::string ltl2da_tool = std::getenv("LTL2DA");
STORM_LOG_INFO("Calling external LTL->DA tool: " << ltl2da_tool << " '" << prefixLtl << "' da.hoa");
pid_t pid;
pid = fork();
STORM_LOG_THROW(pid >= 0, storm::exceptions::FileIoException, "Could not construct deterministic automaton, fork failed");
if (pid == 0) {
// we are in the child process
if (execlp(ltl2da_tool.c_str(), ltl2da_tool.c_str(), prefixLtl.c_str(), "da.hoa", NULL) < 0) {
std::cerr << "ERROR: exec failed: " << strerror(errno) << std::endl;
std::exit(1);
}
// never reached
return std::shared_ptr<DeterministicAutomaton>();
} else { // in the parent
int status;
// wait for completion
while (wait(&status) != pid);
int rv;
if (WIFEXITED(status)) {
rv = WEXITSTATUS(status);
} else {
STORM_LOG_THROW(false, storm::exceptions::FileIoException, "Could not construct deterministic automaton: process aborted");
}
STORM_LOG_THROW(rv == 0, storm::exceptions::FileIoException, "Could not construct deterministic automaton for " << prefixLtl << ", return code = " << rv);
// TODO for MDP-MC the acceptance condition must be in DNF, otherwise Exception is thrown during computation if accepting ECs
STORM_LOG_INFO("Reading automaton for " << prefixLtl << " from da.hoa");
return DeterministicAutomaton::parseFromFile("da.hoa");
}
}
std::shared_ptr<DeterministicAutomaton> LTL2DeterministicAutomaton::ltl2da(storm::logic::Formula const& f, bool dnf) {
std::string prefixLtl = f.toPrefixString();
storm::automata::DeterministicAutomaton::ptr da;
if(isExternalDaToolSet()){
da = ltl2daExternalTool(prefixLtl);
} else {
da = ltl2daInternalTool(prefixLtl, dnf);
}
return da;
}
} }
} }

8
src/storm/automata/LTL2DeterministicAutomaton.h

@ -10,14 +10,18 @@ namespace storm {
} }
namespace automata { namespace automata {
// fwd // fwd
class DeterministicAutomaton; class DeterministicAutomaton;
class LTL2DeterministicAutomaton { class LTL2DeterministicAutomaton {
public: public:
static std::shared_ptr<DeterministicAutomaton> ltl2da(storm::logic::Formula const&, bool dnf); static std::shared_ptr<DeterministicAutomaton> ltl2da(storm::logic::Formula const&, bool dnf);
private:
static bool isExternalDaToolSet();
static std::shared_ptr<DeterministicAutomaton> ltl2daInternalTool(std::string const& prefixLtl, bool dnf);
static std::shared_ptr<DeterministicAutomaton> ltl2daExternalTool(std::string const& prefixLtl);
}; };
} }

Loading…
Cancel
Save