From f68120639399c1253f63872eae77b2434b5c24db Mon Sep 17 00:00:00 2001 From: TimQu Date: Wed, 29 Jun 2016 16:49:56 +0200 Subject: [PATCH 1/6] building markov automata from prism code Former-commit-id: 791c49c7cf4c2c0ab467afdd1ee3fcdc87433294 --- src/builder/ExplicitModelBuilder.cpp | 70 +++++++++++++++++-- src/builder/ExplicitModelBuilder.h | 19 ++++- src/generator/PrismNextStateGenerator.cpp | 4 +- src/models/sparse/MarkovAutomaton.cpp | 51 +++++++++++--- src/models/sparse/MarkovAutomaton.h | 41 ++++++++++- src/storage/BitVector.cpp | 10 +++ src/storage/BitVector.h | 11 +++ src/storage/SparseMatrix.cpp | 64 +++++++++++++++++ src/storage/SparseMatrix.h | 8 +++ src/utility/ModelInstantiator.h | 2 +- .../builder/ExplicitPrismModelBuilderTest.cpp | 26 +++++++ test/functional/builder/hybrid_states.ma | 19 +++++ test/functional/builder/simple.ma | 15 ++++ test/functional/builder/stream2.ma | 50 +++++++++++++ 14 files changed, 369 insertions(+), 21 deletions(-) create mode 100644 test/functional/builder/hybrid_states.ma create mode 100644 test/functional/builder/simple.ma create mode 100644 test/functional/builder/stream2.ma diff --git a/src/builder/ExplicitModelBuilder.cpp b/src/builder/ExplicitModelBuilder.cpp index f008321ee..aaa000000 100644 --- a/src/builder/ExplicitModelBuilder.cpp +++ b/src/builder/ExplicitModelBuilder.cpp @@ -5,6 +5,7 @@ #include "src/models/sparse/Dtmc.h" #include "src/models/sparse/Ctmc.h" #include "src/models/sparse/Mdp.h" +#include "src/models/sparse/MarkovAutomaton.h" #include "src/models/sparse/StandardRewardModel.h" #include "src/storage/expressions/ExpressionManager.h" @@ -133,6 +134,9 @@ namespace storm { case storm::generator::ModelType::MDP: result = std::shared_ptr>(new storm::models::sparse::Mdp(std::move(modelComponents.transitionMatrix), std::move(modelComponents.stateLabeling), std::move(modelComponents.rewardModels), std::move(modelComponents.choiceLabeling))); break; + case storm::generator::ModelType::MA: + result = std::shared_ptr>(new storm::models::sparse::MarkovAutomaton(std::move(modelComponents.transitionMatrix), std::move(modelComponents.stateLabeling), *std::move(modelComponents.markovianStates), std::move(modelComponents.rewardModels), std::move(modelComponents.choiceLabeling))); + break; default: STORM_LOG_THROW(false, storm::exceptions::WrongFormatException, "Error while creating model: cannot handle this model type."); break; @@ -165,12 +169,17 @@ namespace storm { } template - boost::optional>> ExplicitModelBuilder::buildMatrices(storm::storage::SparseMatrixBuilder& transitionMatrixBuilder, std::vector>& rewardModelBuilders) { + void ExplicitModelBuilder::buildMatrices(storm::storage::SparseMatrixBuilder& transitionMatrixBuilder, std::vector>& rewardModelBuilders, boost::optional>>& choiceLabels , boost::optional& markovianChoices) { // Create choice labels, if requested, - boost::optional>> choiceLabels; if (generator->getOptions().isBuildChoiceLabelsSet()) { choiceLabels = std::vector>(); } + + // Create markovian states bit vector, if required + if (generator->getModelType() == storm::generator::ModelType::MA) { + // The BitVector will be resized when the correct size is known + markovianChoices = storm::storage::BitVector(64, false); + } // Create a callback for the next-state generator to enable it to request the index of states. std::function stateToIdCallback = std::bind(&ExplicitModelBuilder::getOrAddStateIndex, this, std::placeholders::_1); @@ -219,6 +228,12 @@ namespace storm { // Insert empty choice labeling for added self-loop transitions. choiceLabels.get().push_back(boost::container::flat_set()); } + + if (generator->getModelType() == storm::generator::ModelType::MA) { + markovianChoices->enlargeLiberally(currentRow, false); + markovianChoices->set(currentRow); + } + if (!generator->isDeterministicModel()) { transitionMatrixBuilder.newRowGroup(currentRow); } @@ -262,6 +277,12 @@ namespace storm { choiceLabels.get().push_back(choice.getChoiceLabels()); } + // If we keep track of the Markovian choices, store whether the current one is Markovian. + if( markovianChoices && choice.isMarkovian() ) { + markovianChoices->enlargeLiberally(currentRow, false); + markovianChoices->set(currentRow); + } + // Add the probabilistic behavior to the matrix. for (auto const& stateProbabilityPair : choice) { transitionMatrixBuilder.addNextValue(currentRow, stateProbabilityPair.first, stateProbabilityPair.second); @@ -280,6 +301,11 @@ namespace storm { ++currentRowGroup; } } + + if (markovianChoices) { + // We now know the correct size + markovianChoices->resize(currentRow, false); + } // If the exploration order was not breadth-first, we need to fix the entries in the matrix according to // (reversed) mapping of row groups to indices. @@ -304,8 +330,6 @@ namespace storm { // Fix (c). this->stateStorage.stateToId.remap([&remapping] (StateType const& state) { return remapping[state]; } ); } - - return choiceLabels; } template @@ -323,7 +347,8 @@ namespace storm { rewardModelBuilders.emplace_back(generator->getRewardModelInformation(i)); } - modelComponents.choiceLabeling = buildMatrices(transitionMatrixBuilder, rewardModelBuilders); + boost::optional markovianChoices; + buildMatrices(transitionMatrixBuilder, rewardModelBuilders, modelComponents.choiceLabeling, markovianChoices); modelComponents.transitionMatrix = transitionMatrixBuilder.build(); // Now finalize all reward models. @@ -342,9 +367,44 @@ namespace storm { } } + if (generator->getModelType() == storm::generator::ModelType::MA) { + STORM_LOG_ASSERT(markovianChoices, "No information regarding markovian choices available."); + buildMarkovianStates(modelComponents, *markovianChoices); + } + return modelComponents; } + template + void ExplicitModelBuilder::buildMarkovianStates(ModelComponents& modelComponents, storm::storage::BitVector const& markovianChoices) const { + modelComponents.markovianStates = storm::storage::BitVector(modelComponents.transitionMatrix.getRowGroupCount(), false); + // Check for each state whether it contains a markovian choice. + for (uint_fast64_t state = 0; state < modelComponents.transitionMatrix.getRowGroupCount(); ++state ) { + uint_fast64_t firstChoice = modelComponents.transitionMatrix.getRowGroupIndices()[state]; + uint_fast64_t markovianChoice = markovianChoices.getNextSetIndex(firstChoice); + if (markovianChoice < modelComponents.transitionMatrix.getRowGroupIndices()[state+1]) { + // Found a markovian choice. Assert that there is not a second one. + STORM_LOG_THROW(markovianChoices.getNextSetIndex(markovianChoice+1) >= modelComponents.transitionMatrix.getRowGroupIndices()[state+1], storm::exceptions::WrongFormatException, "Multiple Markovian choices defined for some state"); + modelComponents.markovianStates->set(state); + // Swap the first choice and the found markovian choice (if they are not equal) + if (firstChoice != markovianChoice) { + modelComponents.transitionMatrix.swapRows(firstChoice, markovianChoice); + for (auto& rewardModel : modelComponents.rewardModels) { + if (rewardModel.second.hasStateActionRewards()) { + std::swap(rewardModel.second.getStateActionRewardVector()[firstChoice], rewardModel.second.getStateActionRewardVector()[markovianChoice]); + } + if (rewardModel.second.hasTransitionRewards()) { + rewardModel.second.getTransitionRewardMatrix().swapRows(firstChoice, markovianChoice); + } + } + if (modelComponents.choiceLabeling) { + std::swap(modelComponents.choiceLabeling.get()[firstChoice], modelComponents.choiceLabeling.get()[markovianChoice]); + } + } + } + } + } + template storm::models::sparse::StateLabeling ExplicitModelBuilder::buildStateLabeling() { return generator->label(stateStorage.stateToId, stateStorage.initialStateIndices, stateStorage.deadlockStateIndices); diff --git a/src/builder/ExplicitModelBuilder.h b/src/builder/ExplicitModelBuilder.h index 296c15b7d..bb404f088 100644 --- a/src/builder/ExplicitModelBuilder.h +++ b/src/builder/ExplicitModelBuilder.h @@ -63,6 +63,9 @@ namespace storm { // A vector that stores a labeling for each choice. boost::optional>> choiceLabeling; + + // A vector that stores which states are markovian. + boost::optional markovianStates; }; struct Options { @@ -138,10 +141,10 @@ namespace storm { * * @param transitionMatrixBuilder The builder of the transition matrix. * @param rewardModelBuilders The builders for the selected reward models. - * @return A tuple containing a vector with all rows at which the nondeterministic choices of each state begin - * and a vector containing the labels associated with each choice. + * @param choiceLabels is set to a vector containing the labels associated with each choice (is only set if choice labels are requested). + * @param markovianChoices is set to a bit vector storing whether a choice is markovian (is only set if the model type requires this information). */ - boost::optional>> buildMatrices(storm::storage::SparseMatrixBuilder& transitionMatrixBuilder, std::vector>& rewardModelBuilders); + void buildMatrices(storm::storage::SparseMatrixBuilder& transitionMatrixBuilder, std::vector>& rewardModelBuilders, boost::optional>>& choiceLabels , boost::optional& markovianChoices); /*! * Explores the state space of the given program and returns the components of the model as a result. @@ -150,6 +153,16 @@ namespace storm { */ ModelComponents buildModelComponents(); + /*! + * Set the markovian states of the given modelComponents, + * makes sure that each state has at most one markovian choice, + * and makes this choice the first one of the corresponding state + * + * @param modelComponents The components of the model build so far + * @markovianChoices bit vector storing whether a choice is markovian + */ + void buildMarkovianStates(ModelComponents& modelComponents, storm::storage::BitVector const& markovianChoices) const; + /*! * Builds the state labeling for the given program. * diff --git a/src/generator/PrismNextStateGenerator.cpp b/src/generator/PrismNextStateGenerator.cpp index 1173fc5d2..552cd30a9 100644 --- a/src/generator/PrismNextStateGenerator.cpp +++ b/src/generator/PrismNextStateGenerator.cpp @@ -366,7 +366,7 @@ namespace storm { continue; } - result.push_back(Choice(command.getActionIndex())); + result.push_back(Choice(command.getActionIndex(), command.isMarkovian())); Choice& choice = result.back(); // Remember the command labels only if we were asked to. @@ -564,4 +564,4 @@ namespace storm { template class PrismNextStateGenerator; template class PrismNextStateGenerator; } -} \ No newline at end of file +} diff --git a/src/models/sparse/MarkovAutomaton.cpp b/src/models/sparse/MarkovAutomaton.cpp index 81f3bc5c9..f822de3c1 100644 --- a/src/models/sparse/MarkovAutomaton.cpp +++ b/src/models/sparse/MarkovAutomaton.cpp @@ -1,12 +1,14 @@ #include "src/models/sparse/MarkovAutomaton.h" -#include "src/models/sparse/StandardRewardModel.h" -#include "src/exceptions/InvalidArgumentException.h" -#include "src/utility/constants.h" + #include "src/adapters/CarlAdapter.h" -#include "src/storage/FlexibleSparseMatrix.h" -#include "src/models/sparse/Dtmc.h" +#include "src/models/sparse/StandardRewardModel.h" #include "src/solver/stateelimination/StateEliminator.h" +#include "src/storage/FlexibleSparseMatrix.h" +#include "src/utility/constants.h" #include "src/utility/vector.h" +#include "src/utility/macros.h" + +#include "src/exceptions/InvalidArgumentException.h" namespace storm { namespace models { @@ -33,6 +35,30 @@ namespace storm { : NondeterministicModel(storm::models::ModelType::MarkovAutomaton, std::move(transitionMatrix), std::move(stateLabeling), std::move(rewardModels), std::move(optionalChoiceLabeling)), markovianStates(markovianStates), exitRates(std::move(exitRates)), closed(this->checkIsClosed()) { this->turnRatesToProbabilities(); } + + + template + MarkovAutomaton::MarkovAutomaton(storm::storage::SparseMatrix const& rateMatrix, + storm::models::sparse::StateLabeling const& stateLabeling, + storm::storage::BitVector const& markovianStates, + std::unordered_map const& rewardModels, + boost::optional> const& optionalChoiceLabeling) + : NondeterministicModel(storm::models::ModelType::MarkovAutomaton, rateMatrix, stateLabeling, rewardModels, optionalChoiceLabeling), markovianStates(markovianStates) { + turnRatesToProbabilities(); + this->closed = checkIsClosed(); + } + + template + MarkovAutomaton::MarkovAutomaton(storm::storage::SparseMatrix&& rateMatrix, + storm::models::sparse::StateLabeling&& stateLabeling, + storm::storage::BitVector&& markovianStates, + std::unordered_map&& rewardModels, + boost::optional>&& optionalChoiceLabeling) + : NondeterministicModel(storm::models::ModelType::MarkovAutomaton, rateMatrix, stateLabeling, rewardModels, optionalChoiceLabeling), markovianStates(markovianStates) { + turnRatesToProbabilities(); + this->closed = checkIsClosed(); + } + template MarkovAutomaton::MarkovAutomaton(storm::storage::SparseMatrix&& transitionMatrix, @@ -230,9 +256,18 @@ namespace storm { template void MarkovAutomaton::turnRatesToProbabilities() { - for (auto state : this->markovianStates) { - for (auto& transition : this->getTransitionMatrix().getRow(this->getTransitionMatrix().getRowGroupIndices()[state])) { - transition.setValue(transition.getValue() / this->exitRates[state]); + this->exitRates.resize(this->getNumberOfStates()); + for (uint_fast64_t state = 0; state< this->getNumberOfStates(); ++state) { + uint_fast64_t row = this->getTransitionMatrix().getRowGroupIndices()[state]; + if(this->markovianStates.get(state)) { + this->exitRates[state] = this->getTransitionMatrix().getRowSum(row); + for (auto& transition : this->getTransitionMatrix().getRow(row)) { + transition.setValue(transition.getValue() / this->exitRates[state]); + } + ++row; + } + for(; row < this->getTransitionMatrix().getRowGroupIndices()[state+1]; ++row) { + STORM_LOG_THROW(storm::utility::isOne(this->getTransitionMatrix().getRowSum(row)), storm::exceptions::InvalidArgumentException, "Transitions of rateMatrix do not sum up to one for some non-Markovian choice."); } } } diff --git a/src/models/sparse/MarkovAutomaton.h b/src/models/sparse/MarkovAutomaton.h index 9864f4f85..f78dc073c 100644 --- a/src/models/sparse/MarkovAutomaton.h +++ b/src/models/sparse/MarkovAutomaton.h @@ -31,7 +31,43 @@ namespace storm { std::vector const& exitRates, std::unordered_map const& rewardModels = std::unordered_map(), boost::optional> const& optionalChoiceLabeling = boost::optional>()); + + /*! + * Constructs a model from the given data. + * + * For hybrid states (i.e., states with Markovian and probabilistic transitions), it is assumed that the first + * choice corresponds to the markovian transitions. + * + * @param rateMatrix The matrix representing the transitions in the model in terms of rates. + * @param stateLabeling The labeling of the states. + * @param markovianStates A bit vector indicating the Markovian states of the automaton (i.e., states with at least one markovian transition). + * @param rewardModels A mapping of reward model names to reward models. + * @param optionalChoiceLabeling A vector that represents the labels associated with the choices of each state. + */ + MarkovAutomaton(storm::storage::SparseMatrix const& rateMatrix, + storm::models::sparse::StateLabeling const& stateLabeling, + storm::storage::BitVector const& markovianStates, + std::unordered_map const& rewardModels = std::unordered_map(), + boost::optional> const& optionalChoiceLabeling = boost::optional>()); + /*! + * Constructs a model from the given data. + * + * For hybrid states (i.e., states with Markovian and probabilistic transitions), it is assumed that the first + * choice corresponds to the markovian transitions. + * + * @param rateMatrix The matrix representing the transitions in the model in terms of rates. + * @param stateLabeling The labeling of the states. + * @param markovianStates A bit vector indicating the Markovian states of the automaton (i.e., states with at least one markovian transition). + * @param rewardModels A mapping of reward model names to reward models. + * @param optionalChoiceLabeling A vector that represents the labels associated with the choices of each state. + */ + MarkovAutomaton(storm::storage::SparseMatrix&& rateMatrix, + storm::models::sparse::StateLabeling&& stateLabeling, + storm::storage::BitVector&& markovianStates, + std::unordered_map&& rewardModels = std::unordered_map(), + boost::optional>&& optionalChoiceLabeling = boost::optional>()); + /*! * Constructs a model by moving the given data. * @@ -48,7 +84,7 @@ namespace storm { std::vector const& exitRates, std::unordered_map&& rewardModels = std::unordered_map(), boost::optional>&& optionalChoiceLabeling = boost::optional>()); - + /*! * Constructs a model by moving the given data. * @@ -160,7 +196,8 @@ namespace storm { /*! * Under the assumption that the Markovian choices of this Markov automaton are expressed in terms of * rates in the transition matrix, this procedure turns the rates into the corresponding probabilities by - * dividing each entry by the exit rate of the state. + * dividing each entry by the sum of the rates for that choice. + * Also sets the exitRates accordingly and throws an exception if the values for a non-markovian choice do not sum up to one. */ void turnRatesToProbabilities(); diff --git a/src/storage/BitVector.cpp b/src/storage/BitVector.cpp index a60b8c10c..ccbcce134 100644 --- a/src/storage/BitVector.cpp +++ b/src/storage/BitVector.cpp @@ -252,6 +252,16 @@ namespace storm { truncateLastBucket(); } } + + void BitVector::enlargeLiberally(uint_fast64_t minimumLength, bool init) { + if(minimumLength > this->size()) { + uint_fast64_t newLength = this->bucketCount() << 6; + while(newLength < minimumLength) { + newLength = newLength << 1; + } + resize(newLength, init); + } + } BitVector BitVector::operator&(BitVector const& other) const { STORM_LOG_ASSERT(bitCount == other.bitCount, "Length of the bit vectors does not match."); diff --git a/src/storage/BitVector.h b/src/storage/BitVector.h index 62ae010be..6ae966379 100644 --- a/src/storage/BitVector.h +++ b/src/storage/BitVector.h @@ -234,6 +234,17 @@ namespace storm { */ void resize(uint_fast64_t newLength, bool init = false); + /*! + * Enlarges the bit vector such that it holds at least the given number of bits (but possibly more). + * This can be used to diminish reallocations when the final size of the bit vector is not known yet. + * The bit vector does not become smaller. + * New bits are initialized to the given value. + * + * @param minimumLength The minimum number of bits that the bit vector should hold. + * @param init The truth value to which to initialize newly created bits. + */ + void enlargeLiberally(uint_fast64_t minimumLength, bool init = false); + /*! * Performs a logical "and" with the given bit vector. In case the sizes of the bit vectors do not match, * only the matching portion is considered and the overlapping bits are set to 0. diff --git a/src/storage/SparseMatrix.cpp b/src/storage/SparseMatrix.cpp index e4e81e4c5..95f65ce99 100644 --- a/src/storage/SparseMatrix.cpp +++ b/src/storage/SparseMatrix.cpp @@ -658,6 +658,70 @@ namespace storm { return bv; } + template + void SparseMatrix::swapRows(index_type const& row1, index_type const& row2) { + if(row1==row2) { + return; + } + + // Get the index of the row that has more / less entries than the other + index_type largerRow = getRow(row1).getNumberOfEntries() > getRow(row2).getNumberOfEntries() ? row1 : row2; + index_type smallerRow = largerRow == row1 ? row2 : row1; + index_type rowSizeDifference = getRow(largerRow).getNumberOfEntries() - getRow(smallerRow).getNumberOfEntries(); + // Save contents of larger row + std::vector> largerRowContents(getRow(largerRow).begin(), getRow(largerRow).end()); + + if(largerRow < smallerRow) { + auto writeIt = getRows(largerRow, smallerRow+1).begin(); + // write smaller row in its new position + for(auto& smallerRowEntry : getRow(smallerRow)) { + *writeIt = std::move(smallerRowEntry); + ++writeIt; + } + if(!storm::utility::isZero(rowSizeDifference)) { + // write the intermediate rows into their correct position + for(auto& intermediateRowEntry : getRows(largerRow+1, smallerRow)) { + *writeIt = std::move(intermediateRowEntry); + ++writeIt; + } + } + // write the larger row + for(auto& largerRowEntry : largerRowContents) { + *writeIt = std::move(largerRowEntry); + ++writeIt; + } + STORM_LOG_ASSERT(writeIt == getRow(smallerRow).end(), "Unexpected position of write iterator"); + //Update row indications + for(index_type row = largerRow +1; row <= smallerRow; ++row) { + rowIndications[row] -= rowSizeDifference; + } + } else { + auto writeIt = getRows(smallerRow, largerRow+1).end() -1; + // write smaller row in its new position + for(auto smallerRowEntryIt = getRow(smallerRow).end() -1; smallerRowEntryIt != getRow(smallerRow).begin()-1; --smallerRowEntryIt) { + *writeIt = std::move(*smallerRowEntryIt); + --writeIt; + } + if(!storm::utility::isZero(rowSizeDifference)) { + // write the intermediate rows into their correct position + for(auto intermediateRowEntryIt = getRows(smallerRow+1, largerRow).end() -1; intermediateRowEntryIt != getRows(smallerRow+1, largerRow).begin()-1; --intermediateRowEntryIt) { + *writeIt = std::move(*intermediateRowEntryIt); + --writeIt; + } + } + // write the larger row + for(auto largerRowEntryIt = largerRowContents.rbegin(); largerRowEntryIt != largerRowContents.rend(); ++largerRowEntryIt) { + *writeIt = std::move(*largerRowEntryIt); + --writeIt; + } + STORM_LOG_ASSERT(writeIt == getRow(smallerRow).begin()-1, "Unexpected position of write iterator"); + //Update row indications + for(index_type row = smallerRow +1; row <= largerRow; ++row) { + rowIndications[row] += rowSizeDifference; + } + } + } + template ValueType SparseMatrix::getConstrainedRowSum(index_type row, storm::storage::BitVector const& constraint) const { ValueType result = storm::utility::zero(); diff --git a/src/storage/SparseMatrix.h b/src/storage/SparseMatrix.h index 95d209b07..8e386f446 100644 --- a/src/storage/SparseMatrix.h +++ b/src/storage/SparseMatrix.h @@ -655,11 +655,19 @@ namespace storm { * @return True if the rows have identical entries. */ bool compareRows(index_type i1, index_type i2) const; + /*! * Finds duplicate rows in a rowgroup. */ BitVector duplicateRowsInRowgroups() const; + /** + * Swaps the two rows. + * @param row1 Index of first row + * @param row2 Index of second row + */ + void swapRows(index_type const& row1, index_type const& row2); + /*! * Selects exactly one row from each row group of this matrix and returns the resulting matrix. * diff --git a/src/utility/ModelInstantiator.h b/src/utility/ModelInstantiator.h index 939e71e2d..2b116188e 100644 --- a/src/utility/ModelInstantiator.h +++ b/src/utility/ModelInstantiator.h @@ -82,7 +82,7 @@ namespace storm { auto markovianStatesCopy = parametricModel.getMarkovianStates(); auto choiceLabelingCopy = parametricModel.getOptionalChoiceLabeling(); std::vector exitRates(parametricModel.getExitRates().size(), storm::utility::one()); - this->instantiatedModel = std::make_shared(buildDummyMatrix(parametricModel.getTransitionMatrix()), std::move(stateLabelingCopy), std::move(markovianStatesCopy), std::move(exitRates), buildDummyRewardModels(parametricModel.getRewardModels()), std::move(choiceLabelingCopy)); + this->instantiatedModel = std::make_shared(buildDummyMatrix(parametricModel.getTransitionMatrix()), std::move(stateLabelingCopy), std::move(markovianStatesCopy), std::move(exitRates), true, buildDummyRewardModels(parametricModel.getRewardModels()), std::move(choiceLabelingCopy)); initializeVectorMapping(this->instantiatedModel->getExitRates(), this->functions, this->vectorMapping, parametricModel.getExitRates()); } diff --git a/test/functional/builder/ExplicitPrismModelBuilderTest.cpp b/test/functional/builder/ExplicitPrismModelBuilderTest.cpp index ac2b1f6fb..66906c2c5 100644 --- a/test/functional/builder/ExplicitPrismModelBuilderTest.cpp +++ b/test/functional/builder/ExplicitPrismModelBuilderTest.cpp @@ -1,6 +1,7 @@ #include "gtest/gtest.h" #include "storm-config.h" #include "src/models/sparse/StandardRewardModel.h" +#include "src/models/sparse/MarkovAutomaton.h" #include "src/settings/SettingMemento.h" #include "src/parser/PrismParser.h" #include "src/builder/ExplicitModelBuilder.h" @@ -99,6 +100,31 @@ TEST(ExplicitPrismModelBuilderTest, Mdp) { EXPECT_EQ(59ul, model->getNumberOfTransitions()); } +TEST(ExplicitPrismModelBuilderTest, Ma) { + storm::prism::Program program = storm::parser::PrismParser::parse(STORM_CPP_TESTS_BASE_PATH "/functional/builder/simple.ma"); + + std::shared_ptr> model = storm::builder::ExplicitModelBuilder(program).build(); + EXPECT_EQ(5ul, model->getNumberOfStates()); + EXPECT_EQ(8ul, model->getNumberOfTransitions()); + ASSERT_TRUE(model->isOfType(storm::models::ModelType::MarkovAutomaton)); + EXPECT_EQ(4ul, model->as>()->getMarkovianStates().getNumberOfSetBits()); + + program = storm::parser::PrismParser::parse(STORM_CPP_TESTS_BASE_PATH "/functional/builder/hybrid_states.ma"); + model = storm::builder::ExplicitModelBuilder(program).build(); + EXPECT_EQ(5ul, model->getNumberOfStates()); + EXPECT_EQ(13ul, model->getNumberOfTransitions()); + ASSERT_TRUE(model->isOfType(storm::models::ModelType::MarkovAutomaton)); + EXPECT_EQ(5ul, model->as>()->getMarkovianStates().getNumberOfSetBits()); + + program = storm::parser::PrismParser::parse(STORM_CPP_TESTS_BASE_PATH "/functional/builder/stream2.ma"); + model = storm::builder::ExplicitModelBuilder(program).build(); + EXPECT_EQ(12ul, model->getNumberOfStates()); + EXPECT_EQ(14ul, model->getNumberOfTransitions()); + ASSERT_TRUE(model->isOfType(storm::models::ModelType::MarkovAutomaton)); + EXPECT_EQ(7ul, model->as>()->getMarkovianStates().getNumberOfSetBits()); + +} + TEST(ExplicitPrismModelBuilderTest, FailComposition) { storm::prism::Program program = storm::parser::PrismParser::parse(STORM_CPP_TESTS_BASE_PATH "/functional/builder/system_composition.nm"); diff --git a/test/functional/builder/hybrid_states.ma b/test/functional/builder/hybrid_states.ma new file mode 100644 index 000000000..3c19d220a --- /dev/null +++ b/test/functional/builder/hybrid_states.ma @@ -0,0 +1,19 @@ + +ma + +module hybrid_states + + s : [0..4]; + + [] (s=0) -> 0.8 : (s'=0) + 0.2 : (s'=2); + [] (s=0) -> 1 : (s' = 1); + <> (s=0) -> 3 : (s' = 1); + <> (s=1) -> 9 : (s'=0) + 1 : (s'=3); + <> (s=2) -> 12 : (s'=4); + [] (s=3) -> 1 : true; + [] (s=3) -> 1 : (s'=4); + <> (s=3) -> 2 : (s'=3) + 2 : (s'=4); + [] (s=4) -> 1 : true; + <> (s=4) -> 3 : (s'=3); + +endmodule diff --git a/test/functional/builder/simple.ma b/test/functional/builder/simple.ma new file mode 100644 index 000000000..92a003555 --- /dev/null +++ b/test/functional/builder/simple.ma @@ -0,0 +1,15 @@ + +ma + +module simple + + s : [0..4]; + + + [alpha] (s=0) -> 1 : (s' = 1); + [beta] (s=0) -> 0.8 : (s'=0) + 0.2 : (s'=2); + <> (s=1) -> 9 : (s'=0) + 1 : (s'=3); + <> (s=2) -> 12 : (s'=4); + <> (s>2) -> 1 : true; + +endmodule \ No newline at end of file diff --git a/test/functional/builder/stream2.ma b/test/functional/builder/stream2.ma new file mode 100644 index 000000000..56378a490 --- /dev/null +++ b/test/functional/builder/stream2.ma @@ -0,0 +1,50 @@ + +ma + +const int N = 2; // num packages + +const double inRate = 4; +const double processingRate = 5; + +module streamingclient + + s : [0..3]; // current state: + // 0: decide whether to start + // 1: buffering + // 2: running + // 3: done + + n : [0..N]; // number of received packages + k : [0..N]; // number of processed packages + + [buffer] s=0 & n 1 : (s'=1); + [start] s=0 & k 1 : (s'=2) & (k'=k+1); + + <> s=1 -> inRate : (n'=n+1) & (s'=0); + + <> s=2 & n inRate : (n'=n+1) + processingRate : (k'=k+1); + <> s=2 & n inRate : (n'=n+1) + processingRate : (s'=0); + <> s=2 & n=N & k processingRate : (k'=k+1); + <> s=2 & n=N & k=N -> processingRate : (s'=3); + + <> s=3 -> 1 : true; +endmodule + +// All packages received and buffer empty +label "done" = (s=3); + +rewards "buffering" + s=1 : 1; +endrewards + +rewards "initialbuffering" + s=1 & k = 0: 1; +endrewards + +rewards "intermediatebuffering" + s=1 & k > 0: 1; +endrewards + +rewards "numRestarts" + [start] k > 0 : 1; +endrewards From 556b8e872641686e76545098ad3156ba75962f66 Mon Sep 17 00:00:00 2001 From: hbruintjes Date: Wed, 29 Jun 2016 17:15:20 +0200 Subject: [PATCH 2/6] Return reference to settings module, not a copy Former-commit-id: 0a5befee57d1f264268f19085aa859dad9987963 --- src/settings/SettingsManager.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/settings/SettingsManager.h b/src/settings/SettingsManager.h index 11c9d78d6..6b1afe5db 100644 --- a/src/settings/SettingsManager.h +++ b/src/settings/SettingsManager.h @@ -260,7 +260,7 @@ namespace storm { * @return The module. */ template - SettingsType getModule() { + SettingsType const& getModule() { static_assert(std::is_base_of::value, "Template argument must be derived from ModuleSettings"); return dynamic_cast(manager().getModule(SettingsType::moduleName)); } From a2140141a3bba6d628c5bd61b0cbcf09aaebe813 Mon Sep 17 00:00:00 2001 From: hbruintjes Date: Wed, 29 Jun 2016 18:05:56 +0200 Subject: [PATCH 3/6] Fix virtual destructor Former-commit-id: ff3ee6cafac3985054bedc09593d6c645dd4c0b1 --- src/settings/modules/ModuleSettings.h | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/settings/modules/ModuleSettings.h b/src/settings/modules/ModuleSettings.h index 08c92909d..2145d5267 100644 --- a/src/settings/modules/ModuleSettings.h +++ b/src/settings/modules/ModuleSettings.h @@ -30,6 +30,7 @@ namespace storm { * @param moduleName The name of the module for which to build the settings. */ ModuleSettings(std::string const& moduleName); + virtual ~ModuleSettings() {} /*! * Checks whether the settings are consistent. If they are inconsistent, an exception is thrown. @@ -140,4 +141,4 @@ namespace storm { } // namespace settings } // namespace storm -#endif /* STORM_SETTINGS_MODULES_MODULESETTINGS_H_ */ \ No newline at end of file +#endif /* STORM_SETTINGS_MODULES_MODULESETTINGS_H_ */ From ffe325b1964a63198d68e1521ff94c0c92496471 Mon Sep 17 00:00:00 2001 From: hbruintjes Date: Thu, 30 Jun 2016 11:29:00 +0200 Subject: [PATCH 4/6] Fix unitialized data in coresettings Former-commit-id: 7c0dac26ba4df2e22f5296a45862d3a6392c87b3 --- src/settings/modules/CoreSettings.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/settings/modules/CoreSettings.cpp b/src/settings/modules/CoreSettings.cpp index 5617d959c..b42a301c8 100644 --- a/src/settings/modules/CoreSettings.cpp +++ b/src/settings/modules/CoreSettings.cpp @@ -32,7 +32,7 @@ namespace storm { const std::string CoreSettings::cudaOptionName = "cuda"; const std::string CoreSettings::minMaxEquationSolvingTechniqueOptionName = "ndmethod"; - CoreSettings::CoreSettings() : ModuleSettings(moduleName) { + CoreSettings::CoreSettings() : ModuleSettings(moduleName), engine(CoreSettings::Engine::Sparse) { this->addOption(storm::settings::OptionBuilder(moduleName, counterexampleOptionName, false, "Generates a counterexample for the given PRCTL formulas if not satisfied by the model") .addArgument(storm::settings::ArgumentBuilder::createStringArgument("filename", "The name of the file to which the counterexample is to be written.").setDefaultValueString("-").setIsOptional(true).build()).setShortName(counterexampleOptionShortName).build()); this->addOption(storm::settings::OptionBuilder(moduleName, dontFixDeadlockOptionName, false, "If the model contains deadlock states, they need to be fixed by setting this option.").setShortName(dontFixDeadlockOptionShortName).build()); From 12287e509098263f944841dfeecbeff9ff774ab3 Mon Sep 17 00:00:00 2001 From: TimQu Date: Thu, 30 Jun 2016 12:43:02 +0200 Subject: [PATCH 5/6] small fix for building MAs Also fixed a test for explicit MA parsing, where the transitions of a probabilistic choice did not sum up to one Former-commit-id: c2f920a4d6d441ed80e8a64e1b1a25c28d736e74 --- src/builder/ExplicitModelBuilder.cpp | 6 +++--- src/models/sparse/MarkovAutomaton.cpp | 11 ++++++++++- src/models/sparse/MarkovAutomaton.h | 6 ++++-- src/storage/BitVector.cpp | 3 ++- 4 files changed, 19 insertions(+), 7 deletions(-) diff --git a/src/builder/ExplicitModelBuilder.cpp b/src/builder/ExplicitModelBuilder.cpp index aaa000000..bb62757a4 100644 --- a/src/builder/ExplicitModelBuilder.cpp +++ b/src/builder/ExplicitModelBuilder.cpp @@ -178,7 +178,7 @@ namespace storm { // Create markovian states bit vector, if required if (generator->getModelType() == storm::generator::ModelType::MA) { // The BitVector will be resized when the correct size is known - markovianChoices = storm::storage::BitVector(64, false); + markovianChoices = storm::storage::BitVector(); } // Create a callback for the next-state generator to enable it to request the index of states. @@ -230,7 +230,7 @@ namespace storm { } if (generator->getModelType() == storm::generator::ModelType::MA) { - markovianChoices->enlargeLiberally(currentRow, false); + markovianChoices->enlargeLiberally(currentRow+1, false); markovianChoices->set(currentRow); } @@ -279,7 +279,7 @@ namespace storm { // If we keep track of the Markovian choices, store whether the current one is Markovian. if( markovianChoices && choice.isMarkovian() ) { - markovianChoices->enlargeLiberally(currentRow, false); + markovianChoices->enlargeLiberally(currentRow+1, false); markovianChoices->set(currentRow); } diff --git a/src/models/sparse/MarkovAutomaton.cpp b/src/models/sparse/MarkovAutomaton.cpp index f822de3c1..b09da840e 100644 --- a/src/models/sparse/MarkovAutomaton.cpp +++ b/src/models/sparse/MarkovAutomaton.cpp @@ -267,7 +267,7 @@ namespace storm { ++row; } for(; row < this->getTransitionMatrix().getRowGroupIndices()[state+1]; ++row) { - STORM_LOG_THROW(storm::utility::isOne(this->getTransitionMatrix().getRowSum(row)), storm::exceptions::InvalidArgumentException, "Transitions of rateMatrix do not sum up to one for some non-Markovian choice."); + STORM_LOG_THROW(storm::utility::isOne(this->getTransitionMatrix().getRowSum(row)), storm::exceptions::InvalidArgumentException, "Transitions of rateMatrix do not sum up to one for some non-Markovian choice. Sum is " << this->getTransitionMatrix().getRowSum(row) << ". State is " << state << ". Choice is " << row << "."); } } } @@ -355,6 +355,15 @@ namespace storm { } + template + void MarkovAutomaton::printModelInformationToStream(std::ostream& out) const { + this->printModelInformationHeaderToStream(out); + out << "Choices: \t" << this->getNumberOfChoices() << std::endl; + out << "Markovian St.: \t" << this->getMarkovianStates().getNumberOfSetBits() << std::endl; + this->printModelInformationFooterToStream(out); + } + + template class MarkovAutomaton; // template class MarkovAutomaton; template class MarkovAutomaton; diff --git a/src/models/sparse/MarkovAutomaton.h b/src/models/sparse/MarkovAutomaton.h index f78dc073c..71d35b894 100644 --- a/src/models/sparse/MarkovAutomaton.h +++ b/src/models/sparse/MarkovAutomaton.h @@ -188,9 +188,11 @@ namespace storm { std::shared_ptr> convertToCTMC(); - virtual void writeDotToStream(std::ostream& outStream, bool includeLabeling = true, storm::storage::BitVector const* subsystem = nullptr, std::vector const* firstValue = nullptr, std::vector const* secondValue = nullptr, std::vector const* stateColoring = nullptr, std::vector const* colors = nullptr, std::vector* scheduler = nullptr, bool finalizeOutput = true) const; + virtual void writeDotToStream(std::ostream& outStream, bool includeLabeling = true, storm::storage::BitVector const* subsystem = nullptr, std::vector const* firstValue = nullptr, std::vector const* secondValue = nullptr, std::vector const* stateColoring = nullptr, std::vector const* colors = nullptr, std::vector* scheduler = nullptr, bool finalizeOutput = true) const override; - std::size_t getSizeInBytes() const; + std::size_t getSizeInBytes() const override; + + virtual void printModelInformationToStream(std::ostream& out) const override; private: /*! diff --git a/src/storage/BitVector.cpp b/src/storage/BitVector.cpp index ccbcce134..81705d431 100644 --- a/src/storage/BitVector.cpp +++ b/src/storage/BitVector.cpp @@ -255,7 +255,8 @@ namespace storm { void BitVector::enlargeLiberally(uint_fast64_t minimumLength, bool init) { if(minimumLength > this->size()) { - uint_fast64_t newLength = this->bucketCount() << 6; + uint_fast64_t newLength = this->bucketCount(); + newLength = std::max(newLength, 1ull) << 6; while(newLength < minimumLength) { newLength = newLength << 1; } From 8546786e1757e6140b64795dac29447ea6342ede Mon Sep 17 00:00:00 2001 From: hbruintjes Date: Thu, 30 Jun 2016 17:11:54 +0200 Subject: [PATCH 6/6] Fix missing template argument in builder utility Former-commit-id: 23d683f29ced31bd7a84061252fcbd62a609c476 --- src/utility/storm.h | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/utility/storm.h b/src/utility/storm.h index 41a17a4c7..49ef4bf04 100644 --- a/src/utility/storm.h +++ b/src/utility/storm.h @@ -113,10 +113,10 @@ namespace storm { template std::shared_ptr> buildSymbolicModel(storm::prism::Program const& program, std::vector> const& formulas) { - typename storm::builder::DdPrismModelBuilder::Options options; - options = typename storm::builder::DdPrismModelBuilder::Options(formulas); + typename storm::builder::DdPrismModelBuilder::Options options; + options = typename storm::builder::DdPrismModelBuilder::Options(formulas); - storm::builder::DdPrismModelBuilder builder; + storm::builder::DdPrismModelBuilder builder; return builder.build(program, options); }