From ab12e4ff3d9a63d64fa8b80e2661b0501d6ba7f3 Mon Sep 17 00:00:00 2001 From: dehnert Date: Wed, 18 Oct 2017 22:29:48 +0200 Subject: [PATCH] started on partial quotient extraction in symbolic bisimulation --- .../storage/dd/BisimulationDecomposition.cpp | 72 +++++++-- .../storage/dd/BisimulationDecomposition.h | 15 +- src/storm/storage/dd/DdManager.h | 4 +- .../bisimulation/PartialQuotientExtractor.cpp | 63 ++++++++ .../bisimulation/PartialQuotientExtractor.h | 39 +++++ .../SymbolicBisimulationDecompositionTest.cpp | 139 ++++++++++++++++++ 6 files changed, 318 insertions(+), 14 deletions(-) create mode 100644 src/storm/storage/dd/bisimulation/PartialQuotientExtractor.cpp create mode 100644 src/storm/storage/dd/bisimulation/PartialQuotientExtractor.h diff --git a/src/storm/storage/dd/BisimulationDecomposition.cpp b/src/storm/storage/dd/BisimulationDecomposition.cpp index 9476e39da..b5cca9ec5 100644 --- a/src/storm/storage/dd/BisimulationDecomposition.cpp +++ b/src/storm/storage/dd/BisimulationDecomposition.cpp @@ -4,6 +4,7 @@ #include "storm/storage/dd/bisimulation/PartitionRefiner.h" #include "storm/storage/dd/bisimulation/MdpPartitionRefiner.h" #include "storm/storage/dd/bisimulation/QuotientExtractor.h" +#include "storm/storage/dd/bisimulation/PartialQuotientExtractor.h" #include "storm/models/symbolic/Model.h" #include "storm/models/symbolic/Mdp.h" @@ -36,6 +37,11 @@ namespace storm { showProgress = generalSettings.isVerboseSet(); showProgressDelay = generalSettings.getShowProgressDelay(); this->refineWrtRewardModels(); + + STORM_LOG_TRACE("Initial partition has " << refiner->getStatePartition().getNumberOfBlocks() << " blocks."); +#ifndef NDEBUG + STORM_LOG_TRACE("Initial partition has " << refiner->getStatePartition().getNodeCount() << " nodes."); +#endif } template @@ -44,6 +50,11 @@ namespace storm { showProgress = generalSettings.isVerboseSet(); showProgressDelay = generalSettings.getShowProgressDelay(); this->refineWrtRewardModels(); + + STORM_LOG_TRACE("Initial partition has " << refiner->getStatePartition().getNumberOfBlocks() << " blocks."); +#ifndef NDEBUG + STORM_LOG_TRACE("Initial partition has " << refiner->getStatePartition().getNodeCount() << " nodes."); +#endif } template @@ -52,6 +63,11 @@ namespace storm { showProgress = generalSettings.isVerboseSet(); showProgressDelay = generalSettings.getShowProgressDelay(); this->refineWrtRewardModels(); + + STORM_LOG_TRACE("Initial partition has " << refiner->getStatePartition().getNumberOfBlocks() << " blocks."); +#ifndef NDEBUG + STORM_LOG_TRACE("Initial partition has " << refiner->getStatePartition().getNodeCount() << " nodes."); +#endif } template @@ -60,11 +76,7 @@ namespace storm { template void BisimulationDecomposition::compute(bisimulation::SignatureMode const& mode) { STORM_LOG_ASSERT(refiner, "No suitable refiner."); - - STORM_LOG_TRACE("Initial partition has " << refiner->getStatePartition().getNumberOfBlocks() << " blocks."); -#ifndef NDEBUG - STORM_LOG_TRACE("Initial partition has " << refiner->getStatePartition().getNodeCount() << " nodes."); -#endif + STORM_LOG_ASSERT(this->refiner->getStatus() != Status::FixedPoint, "Can only proceed if no fixpoint has been reached yet."); auto start = std::chrono::high_resolution_clock::now(); auto timeOfLastMessage = start; @@ -90,16 +102,54 @@ namespace storm { STORM_LOG_DEBUG("Partition refinement completed in " << std::chrono::duration_cast(end - start).count() << "s (" << iterations << " iterations)."); } + + template + bool BisimulationDecomposition::compute(uint64_t steps, bisimulation::SignatureMode const& mode) { + STORM_LOG_ASSERT(refiner, "No suitable refiner."); + STORM_LOG_ASSERT(this->refiner->getStatus() != Status::FixedPoint, "Can only proceed if no fixpoint has been reached yet."); + STORM_LOG_ASSERT(steps > 0, "Can only perform positive number of steps."); + + auto start = std::chrono::high_resolution_clock::now(); + auto timeOfLastMessage = start; + uint64_t iterations = 0; + bool refined = true; + while (refined && iterations < steps) { + refined = refiner->refine(mode); + + ++iterations; + + if (showProgress) { + auto now = std::chrono::high_resolution_clock::now(); + auto durationSinceLastMessage = std::chrono::duration_cast(now - timeOfLastMessage).count(); + if (static_cast(durationSinceLastMessage) >= showProgressDelay) { + auto durationSinceStart = std::chrono::duration_cast(now - start).count(); + STORM_LOG_INFO("State partition after " << iterations << " iterations (" << durationSinceStart << "s) has " << refiner->getStatePartition().getNumberOfBlocks() << " blocks."); + timeOfLastMessage = std::chrono::high_resolution_clock::now(); + } + } + } + + return !refined; + } template std::shared_ptr> BisimulationDecomposition::getQuotient() const { - STORM_LOG_THROW(this->refiner->getStatus() == Status::FixedPoint, storm::exceptions::InvalidOperationException, "Cannot extract quotient, because bisimulation decomposition was not completed."); - - STORM_LOG_TRACE("Starting quotient extraction."); - QuotientExtractor extractor; - std::shared_ptr> quotient = extractor.extract(model, refiner->getStatePartition(), preservationInformation); + std::shared_ptr> quotient; + if (this->refiner->getStatus() == Status::FixedPoint) { + STORM_LOG_TRACE("Starting full quotient extraction."); + QuotientExtractor extractor; + quotient = extractor.extract(model, refiner->getStatePartition(), preservationInformation); + } else { + STORM_LOG_TRACE("Starting partial quotient extraction."); + if (!partialQuotientExtractor) { + partialQuotientExtractor = std::make_unique>(model); + } + + quotient = partialQuotientExtractor->extract(refiner->getStatePartition(), preservationInformation); + STORM_LOG_TRACE("Quotient extraction done."); + } + STORM_LOG_TRACE("Quotient extraction done."); - return quotient; } diff --git a/src/storm/storage/dd/BisimulationDecomposition.h b/src/storm/storage/dd/BisimulationDecomposition.h index 2a62c9fa4..d4ddfced4 100644 --- a/src/storm/storage/dd/BisimulationDecomposition.h +++ b/src/storm/storage/dd/BisimulationDecomposition.h @@ -28,6 +28,9 @@ namespace storm { template class PartitionRefiner; + + template + class PartialQuotientExtractor; } template @@ -40,10 +43,17 @@ namespace storm { ~BisimulationDecomposition(); /*! - * Computes the decomposition. + * Performs partition refinement until a fixpoint has been reached. */ void compute(bisimulation::SignatureMode const& mode = bisimulation::SignatureMode::Eager); + /*! + * Performs the given number of refinement steps. + * + * @return True iff the computation arrived at a fixpoint. + */ + bool compute(uint64_t steps, bisimulation::SignatureMode const& mode); + /*! * Retrieves the quotient model after the bisimulation decomposition was computed. */ @@ -61,6 +71,9 @@ namespace storm { // The refiner to use. std::unique_ptr> refiner; + // A quotient extractor that is used when the fixpoint has not been reached yet. + mutable std::unique_ptr> partialQuotientExtractor; + // A flag indicating whether progress is reported. bool showProgress; diff --git a/src/storm/storage/dd/DdManager.h b/src/storm/storage/dd/DdManager.h index 055bbfb1a..60af0b4a3 100644 --- a/src/storm/storage/dd/DdManager.h +++ b/src/storm/storage/dd/DdManager.h @@ -168,8 +168,8 @@ namespace storm { * Clones the given meta variable and optionally changes the number of layers of the variable. * * @param variable The variable to clone. - * @param newVariableName The name of the variable to crate. - * @param numberOfLayers The number of layers of the variable to crate + * @param newVariableName The name of the variable to create. + * @param numberOfLayers The number of layers of the variable to create. */ std::vector cloneVariable(storm::expressions::Variable const& variable, std::string const& newVariableName, boost::optional const& numberOfLayers = boost::none); diff --git a/src/storm/storage/dd/bisimulation/PartialQuotientExtractor.cpp b/src/storm/storage/dd/bisimulation/PartialQuotientExtractor.cpp new file mode 100644 index 000000000..cd179dce3 --- /dev/null +++ b/src/storm/storage/dd/bisimulation/PartialQuotientExtractor.cpp @@ -0,0 +1,63 @@ +#include "storm/storage/dd/bisimulation/PartialQuotientExtractor.h" + +#include "storm/storage/dd/DdManager.h" + +#include "storm/settings/SettingsManager.h" + +#include "storm/utility/macros.h" +#include "storm/exceptions/NotSupportedException.h" + +namespace storm { + namespace dd { + namespace bisimulation { + + template + PartialQuotientExtractor::PartialQuotientExtractor(storm::models::symbolic::Model const& model) : model(model) { + auto const& settings = storm::settings::getModule(); + this->quotientFormat = settings.getQuotientFormat(); + + STORM_LOG_THROW(this->quotientFormat == storm::settings::modules::BisimulationSettings::QuotientFormat::Dd, storm::exceptions::NotSupportedException, "Only DD-based partial quotient extraction is currently supported."); + + // We need to create auxiliary variables that encode the nondeterminism arising from partion quotient extraction. + auto& manager = model.getManager(); + uint64_t numberOfDdVariables = 0; + for (auto const& metaVariable : model.getRowVariables()) { + auto const& ddMetaVariable = manager.getMetaVariable(metaVariable); + numberOfDdVariables += ddMetaVariable.getNumberOfDdVariables(); + } + std::vector auxVariables = manager.addBitVectorMetaVariable("quot", numberOfDdVariables, 1); + STORM_LOG_ASSERT(auxVariables.size() == 1, "Expected exactly one meta variable."); + nondeterminismVariable = auxVariables.front(); + } + + template + std::shared_ptr> PartialQuotientExtractor::extract(Partition const& partition, PreservationInformation const& preservationInformation) { + auto start = std::chrono::high_resolution_clock::now(); + std::shared_ptr> result; + + STORM_LOG_THROW(this->quotientFormat == storm::settings::modules::BisimulationSettings::QuotientFormat::Dd, storm::exceptions::NotSupportedException, "Only DD-based partial quotient extraction is currently supported."); + result = extractDdQuotient(partition, preservationInformation); + auto end = std::chrono::high_resolution_clock::now(); + STORM_LOG_TRACE("Quotient extraction completed in " << std::chrono::duration_cast(end - start).count() << "ms."); + + STORM_LOG_THROW(result, storm::exceptions::NotSupportedException, "Quotient could not be extracted."); + + return result; + } + + template + std::shared_ptr> PartialQuotientExtractor::extractDdQuotient(Partition const& partition, PreservationInformation const& preservationInformation) { + return nullptr; + } + + template class PartialQuotientExtractor; + template class PartialQuotientExtractor; + +#ifdef STORM_HAVE_CARL + template class PartialQuotientExtractor; + template class PartialQuotientExtractor; +#endif + + } + } +} diff --git a/src/storm/storage/dd/bisimulation/PartialQuotientExtractor.h b/src/storm/storage/dd/bisimulation/PartialQuotientExtractor.h new file mode 100644 index 000000000..92ab45ffb --- /dev/null +++ b/src/storm/storage/dd/bisimulation/PartialQuotientExtractor.h @@ -0,0 +1,39 @@ +#pragma once + +#include + +#include "storm/storage/dd/DdType.h" + +#include "storm/models/symbolic/Model.h" +#include "storm/models/sparse/Model.h" + +#include "storm/storage/dd/bisimulation/Partition.h" +#include "storm/storage/dd/bisimulation/PreservationInformation.h" + +#include "storm/settings/modules/BisimulationSettings.h" + +namespace storm { + namespace dd { + namespace bisimulation { + + template + class PartialQuotientExtractor { + public: + PartialQuotientExtractor(storm::models::symbolic::Model const& model); + + std::shared_ptr> extract(Partition const& partition, PreservationInformation const& preservationInformation); + + private: + std::shared_ptr> extractDdQuotient(Partition const& partition, PreservationInformation const& preservationInformation); + + // The model for which to compute the partial quotient. + storm::models::symbolic::Model const& model; + + storm::expressions::Variable nondeterminismVariable; + + storm::settings::modules::BisimulationSettings::QuotientFormat quotientFormat; + }; + + } + } +} diff --git a/src/test/storm/storage/SymbolicBisimulationDecompositionTest.cpp b/src/test/storm/storage/SymbolicBisimulationDecompositionTest.cpp index 2f9b217e2..402410933 100644 --- a/src/test/storm/storage/SymbolicBisimulationDecompositionTest.cpp +++ b/src/test/storm/storage/SymbolicBisimulationDecompositionTest.cpp @@ -45,6 +45,45 @@ TEST(SymbolicModelBisimulationDecomposition, Die_Cudd) { EXPECT_TRUE(quotient->isSymbolicModel()); } +TEST(SymbolicModelBisimulationDecomposition, DiePartialQuotient_Cudd) { + storm::prism::Program program = storm::parser::PrismParser::parse(STORM_TEST_RESOURCES_DIR "/dtmc/die.pm"); + + std::shared_ptr> model = storm::builder::DdPrismModelBuilder().build(program); + + storm::dd::BisimulationDecomposition decomposition(*model, storm::storage::BisimulationType::Strong); + std::shared_ptr> quotient = decomposition.getQuotient(); +} + +TEST(SymbolicModelBisimulationDecomposition, Die_Sylvan) { + storm::prism::Program program = storm::parser::PrismParser::parse(STORM_TEST_RESOURCES_DIR "/dtmc/die.pm"); + + std::shared_ptr> model = storm::builder::DdPrismModelBuilder().build(program); + + storm::dd::BisimulationDecomposition decomposition(*model, storm::storage::BisimulationType::Strong); + decomposition.compute(); + std::shared_ptr> quotient = decomposition.getQuotient(); + + EXPECT_EQ(11ul, quotient->getNumberOfStates()); + EXPECT_EQ(17ul, quotient->getNumberOfTransitions()); + EXPECT_EQ(storm::models::ModelType::Dtmc, quotient->getType()); + EXPECT_TRUE(quotient->isSymbolicModel()); + + storm::parser::FormulaParser formulaParser; + std::shared_ptr formula = formulaParser.parseSingleFormulaFromString("P=? [F \"two\"]"); + + std::vector> formulas; + formulas.push_back(formula); + + storm::dd::BisimulationDecomposition decomposition2(*model, formulas, storm::storage::BisimulationType::Strong); + decomposition2.compute(); + quotient = decomposition2.getQuotient(); + + EXPECT_EQ(5ul, quotient->getNumberOfStates()); + EXPECT_EQ(8ul, quotient->getNumberOfTransitions()); + EXPECT_EQ(storm::models::ModelType::Dtmc, quotient->getType()); + EXPECT_TRUE(quotient->isSymbolicModel()); +} + TEST(SymbolicModelBisimulationDecomposition, Crowds_Cudd) { storm::storage::SymbolicModelDescription smd = storm::parser::PrismParser::parse(STORM_TEST_RESOURCES_DIR "/dtmc/crowds5_5.pm"); @@ -78,6 +117,39 @@ TEST(SymbolicModelBisimulationDecomposition, Crowds_Cudd) { EXPECT_TRUE(quotient->isSymbolicModel()); } +TEST(SymbolicModelBisimulationDecomposition, Crowds_Sylvan) { + storm::storage::SymbolicModelDescription smd = storm::parser::PrismParser::parse(STORM_TEST_RESOURCES_DIR "/dtmc/crowds5_5.pm"); + + // Preprocess model to substitute all constants. + smd = smd.preprocess(); + + std::shared_ptr> model = storm::builder::DdPrismModelBuilder().build(smd.asPrismProgram()); + + storm::dd::BisimulationDecomposition decomposition(*model, storm::storage::BisimulationType::Strong); + decomposition.compute(); + std::shared_ptr> quotient = decomposition.getQuotient(); + + EXPECT_EQ(2007ul, quotient->getNumberOfStates()); + EXPECT_EQ(3738ul, quotient->getNumberOfTransitions()); + EXPECT_EQ(storm::models::ModelType::Dtmc, quotient->getType()); + EXPECT_TRUE(quotient->isSymbolicModel()); + + storm::parser::FormulaParser formulaParser; + std::shared_ptr formula = formulaParser.parseSingleFormulaFromString("P=? [F \"observe0Greater1\"]"); + + std::vector> formulas; + formulas.push_back(formula); + + storm::dd::BisimulationDecomposition decomposition2(*model, formulas, storm::storage::BisimulationType::Strong); + decomposition2.compute(); + quotient = decomposition2.getQuotient(); + + EXPECT_EQ(65ul, quotient->getNumberOfStates()); + EXPECT_EQ(105ul, quotient->getNumberOfTransitions()); + EXPECT_EQ(storm::models::ModelType::Dtmc, quotient->getType()); + EXPECT_TRUE(quotient->isSymbolicModel()); +} + TEST(SymbolicModelBisimulationDecomposition, TwoDice_Cudd) { storm::prism::Program program = storm::parser::PrismParser::parse(STORM_TEST_RESOURCES_DIR "/mdp/two_dice.nm"); @@ -110,6 +182,38 @@ TEST(SymbolicModelBisimulationDecomposition, TwoDice_Cudd) { EXPECT_EQ(19ul, (quotient->as>()->getNumberOfChoices())); } +TEST(SymbolicModelBisimulationDecomposition, TwoDice_Sylvan) { + storm::prism::Program program = storm::parser::PrismParser::parse(STORM_TEST_RESOURCES_DIR "/mdp/two_dice.nm"); + + std::shared_ptr> model = storm::builder::DdPrismModelBuilder().build(program); + + storm::dd::BisimulationDecomposition decomposition(*model, storm::storage::BisimulationType::Strong); + decomposition.compute(); + std::shared_ptr> quotient = decomposition.getQuotient(); + + EXPECT_EQ(77ul, quotient->getNumberOfStates()); + EXPECT_EQ(210ul, quotient->getNumberOfTransitions()); + EXPECT_EQ(storm::models::ModelType::Mdp, quotient->getType()); + EXPECT_TRUE(quotient->isSymbolicModel()); + EXPECT_EQ(116ul, (quotient->as>()->getNumberOfChoices())); + + storm::parser::FormulaParser formulaParser; + std::shared_ptr formula = formulaParser.parseSingleFormulaFromString("Pmin=? [F \"two\"]"); + + std::vector> formulas; + formulas.push_back(formula); + + storm::dd::BisimulationDecomposition decomposition2(*model, formulas, storm::storage::BisimulationType::Strong); + decomposition2.compute(); + quotient = decomposition2.getQuotient(); + + EXPECT_EQ(11ul, quotient->getNumberOfStates()); + EXPECT_EQ(34ul, quotient->getNumberOfTransitions()); + EXPECT_EQ(storm::models::ModelType::Mdp, quotient->getType()); + EXPECT_TRUE(quotient->isSymbolicModel()); + EXPECT_EQ(19ul, (quotient->as>()->getNumberOfChoices())); +} + TEST(SymbolicModelBisimulationDecomposition, AsynchronousLeader_Cudd) { storm::storage::SymbolicModelDescription smd = storm::parser::PrismParser::parse(STORM_TEST_RESOURCES_DIR "/mdp/leader4.nm"); @@ -144,3 +248,38 @@ TEST(SymbolicModelBisimulationDecomposition, AsynchronousLeader_Cudd) { EXPECT_TRUE(quotient->isSymbolicModel()); EXPECT_EQ(2152ul, (quotient->as>()->getNumberOfChoices())); } + +TEST(SymbolicModelBisimulationDecomposition, AsynchronousLeader_Sylvan) { + storm::storage::SymbolicModelDescription smd = storm::parser::PrismParser::parse(STORM_TEST_RESOURCES_DIR "/mdp/leader4.nm"); + + // Preprocess model to substitute all constants. + smd = smd.preprocess(); + + storm::parser::FormulaParser formulaParser; + std::shared_ptr formula = formulaParser.parseSingleFormulaFromString("Rmax=? [F \"elected\"]"); + + std::shared_ptr> model = storm::builder::DdPrismModelBuilder().build(smd.asPrismProgram(), *formula); + + storm::dd::BisimulationDecomposition decomposition(*model, storm::storage::BisimulationType::Strong); + decomposition.compute(); + std::shared_ptr> quotient = decomposition.getQuotient(); + + EXPECT_EQ(252ul, quotient->getNumberOfStates()); + EXPECT_EQ(624ul, quotient->getNumberOfTransitions()); + EXPECT_EQ(storm::models::ModelType::Mdp, quotient->getType()); + EXPECT_TRUE(quotient->isSymbolicModel()); + EXPECT_EQ(500ul, (quotient->as>()->getNumberOfChoices())); + + std::vector> formulas; + formulas.push_back(formula); + + storm::dd::BisimulationDecomposition decomposition2(*model, formulas, storm::storage::BisimulationType::Strong); + decomposition2.compute(); + quotient = decomposition2.getQuotient(); + + EXPECT_EQ(1107ul, quotient->getNumberOfStates()); + EXPECT_EQ(2684ul, quotient->getNumberOfTransitions()); + EXPECT_EQ(storm::models::ModelType::Mdp, quotient->getType()); + EXPECT_TRUE(quotient->isSymbolicModel()); + EXPECT_EQ(2152ul, (quotient->as>()->getNumberOfChoices())); +}