#include "logic/Formula.h" #include "parser/DFTGalileoParser.h" #include "utility/initialize.h" #include "builder/ExplicitDFTModelBuilder.h" #include "modelchecker/results/CheckResult.h" #include "utility/storm.h" #include "storage/dft/DFTIsomorphism.h" #include /*! * Load DFT from filename, build corresponding Model and check against given property. * * @param filename Path to DFT file in Galileo format. * @param property PCTC formula capturing the property to check. */ template void analyzeDFT(std::string filename, std::string property, bool symred = false) { storm::settings::SettingsManager& manager = storm::settings::mutableManager(); manager.setFromString(""); // Building DFT std::chrono::high_resolution_clock::time_point buildingStart = std::chrono::high_resolution_clock::now(); storm::parser::DFTGalileoParser parser; storm::storage::DFT dft = parser.parseDFT(filename); std::vector> parsedFormulas = storm::parseFormulasForExplicit(property); std::vector> formulas(parsedFormulas.begin(), parsedFormulas.end()); assert(formulas.size() == 1); // Optimizing DFT dft = dft.optimize(); std::map>> emptySymmetry; storm::storage::DFTIndependentSymmetries symmetries(emptySymmetry); if(symred) { auto colouring = dft.colourDFT(); symmetries = dft.findSymmetries(colouring); std::cout << "Found " << symmetries.groups.size() << " symmetries." << std::endl; STORM_LOG_TRACE("Symmetries: " << std::endl << symmetries); } std::chrono::high_resolution_clock::time_point buildingEnd = std::chrono::high_resolution_clock::now(); // Building Markov Automaton std::cout << "Building Model..." << std::endl; storm::builder::ExplicitDFTModelBuilder builder(dft, symmetries); typename storm::builder::ExplicitDFTModelBuilder::LabelOptions labeloptions; // TODO initialize this with the formula std::shared_ptr> model = builder.buildModel(labeloptions); //model->printModelInformationToStream(std::cout); std::cout << "No. states (Explored): " << model->getNumberOfStates() << std::endl; std::cout << "No. transitions (Explored): " << model->getNumberOfTransitions() << std::endl; std::chrono::high_resolution_clock::time_point explorationEnd = std::chrono::high_resolution_clock::now(); // Bisimulation if (model->isOfType(storm::models::ModelType::Ctmc)) { std::cout << "Bisimulation..." << std::endl; model = storm::performDeterministicSparseBisimulationMinimization>(model->template as>(), formulas, storm::storage::BisimulationType::Weak)->template as>(); //model->printModelInformationToStream(std::cout); } std::cout << "No. states (Bisimulation): " << model->getNumberOfStates() << std::endl; std::cout << "No. transitions (Bisimulation): " << model->getNumberOfTransitions() << std::endl; std::chrono::high_resolution_clock::time_point bisimulationEnd = std::chrono::high_resolution_clock::now(); // Model checking std::cout << "Model checking..." << std::endl; std::unique_ptr result(storm::verifySparseModel(model, formulas[0])); assert(result); result->filter(storm::modelchecker::ExplicitQualitativeCheckResult(model->getInitialStates())); // Times std::chrono::high_resolution_clock::time_point modelCheckingEnd = std::chrono::high_resolution_clock::now(); std::chrono::duration buildingTime = buildingEnd - buildingStart; std::chrono::duration explorationTime = explorationEnd - buildingEnd; std::chrono::duration bisimulationTime = bisimulationEnd - explorationEnd; std::chrono::duration modelCheckingTime = modelCheckingEnd - bisimulationEnd; std::chrono::duration totalTime = modelCheckingEnd - buildingStart; std::cout << "Times:" << std::endl; std::cout << "Building:\t" << buildingTime.count() << std::endl; std::cout << "Exploration:\t" << explorationTime.count() << std::endl; std::cout << "Bisimulation:\t" << bisimulationTime.count() << std::endl; std::cout << "Modelchecking:\t" << modelCheckingTime.count() << std::endl; std::cout << "Total:\t\t" << totalTime.count() << std::endl; std::cout << "Result: "; std::cout << *result << std::endl; } /*! * Entry point for the DyFTeE backend. * * @param argc The argc argument of main(). * @param argv The argv argument of main(). * @return Return code, 0 if successfull, not 0 otherwise. */ int main(int argc, char** argv) { if(argc < 2) { std::cout << "Storm-DyFTeE should be called with a filename as argument." << std::endl; std::cout << "./storm-dft <--prop pctl-formula> <--parametric>" << std::endl; return 1; } // Parse cli arguments bool parametric = false; bool symred = false; bool minimal = true; std::string filename = argv[1]; std::string operatorType = ""; std::string targetFormula = ""; std::string pctlFormula = ""; for (int i = 2; i < argc; ++i) { std::string option = argv[i]; if (option == "--parametric") { parametric = true; } else if (option == "--expectedtime") { assert(targetFormula.empty()); operatorType = "ET"; targetFormula = "F \"failed\""; } else if (option == "--probability") { assert(targetFormula.empty()); operatorType = "P";; targetFormula = "F \"failed\""; } else if (option == "--timebound") { assert(targetFormula.empty()); ++i; assert(i < argc); double timeBound; try { timeBound = boost::lexical_cast(argv[i]); } catch (boost::bad_lexical_cast e) { std::cerr << "The time bound '" << argv[i] << "' is not valid." << std::endl; return 2; } std::stringstream stream; stream << "F<=" << timeBound << " \"failed\""; operatorType = "P"; targetFormula = stream.str(); } else if (option == "--trace") { STORM_GLOBAL_LOGLEVEL_TRACE(); } else if (option == "--debug") { STORM_GLOBAL_LOGLEVEL_DEBUG(); } else if (option == "--prop") { assert(pctlFormula.empty()); ++i; assert(i < argc); pctlFormula = argv[i]; } else if (option == "--symred") { symred = true; } else if (option == "--min") { minimal = true; } else if (option == "--max") { minimal = false; } else { std::cout << "Option '" << option << "' not recognized." << std::endl; return 1; } } // Construct pctlFormula if (!targetFormula.empty()) { assert(pctlFormula.empty()); pctlFormula = operatorType + (minimal ? "min" : "max") + "=?[" + targetFormula + "]"; } assert(!pctlFormula.empty()); storm::utility::setUp(); std::cout << "Running " << (parametric ? "parametric " : "") << "DFT analysis on file " << filename << " with property " << pctlFormula << std::endl; if (parametric) { analyzeDFT(filename, pctlFormula, symred); } else { analyzeDFT(filename, pctlFormula, symred); } }