diff --git a/src/storm/modelchecker/hints/ExplicitModelCheckerHint.cpp b/src/storm/modelchecker/hints/ExplicitModelCheckerHint.cpp index 67d233fcd..0b99ca27d 100644 --- a/src/storm/modelchecker/hints/ExplicitModelCheckerHint.cpp +++ b/src/storm/modelchecker/hints/ExplicitModelCheckerHint.cpp @@ -4,12 +4,12 @@ namespace storm { namespace modelchecker { template - ExplicitModelCheckerHint::ExplicitModelCheckerHint(boost::optional> const& resultHint, boost::optional const& schedulerHint) : resultHint(resultHint), schedulerHint(schedulerHint) { + ExplicitModelCheckerHint::ExplicitModelCheckerHint(boost::optional> const& resultHint, boost::optional const& schedulerHint) : resultHint(resultHint), schedulerHint(schedulerHint), forceApplicationOfHints(false) { // Intentionally left empty } template - ExplicitModelCheckerHint::ExplicitModelCheckerHint(boost::optional>&& resultHint, boost::optional&& schedulerHint) : resultHint(resultHint), schedulerHint(schedulerHint) { + ExplicitModelCheckerHint::ExplicitModelCheckerHint(boost::optional>&& resultHint, boost::optional&& schedulerHint) : resultHint(resultHint), schedulerHint(schedulerHint), forceApplicationOfHints(false) { // Intentionally left empty } @@ -73,6 +73,16 @@ namespace storm { this->schedulerHint = schedulerHint; } + template + bool ExplicitModelCheckerHint::getForceApplicationOfHints() const { + return forceApplicationOfHints; + } + + template + void ExplicitModelCheckerHint::setForceApplicationOfHints(bool value) { + forceApplicationOfHints = value; + } + template class ExplicitModelCheckerHint; template class ExplicitModelCheckerHint; template class ExplicitModelCheckerHint; diff --git a/src/storm/modelchecker/hints/ExplicitModelCheckerHint.h b/src/storm/modelchecker/hints/ExplicitModelCheckerHint.h index 533c551f8..0a514711e 100644 --- a/src/storm/modelchecker/hints/ExplicitModelCheckerHint.h +++ b/src/storm/modelchecker/hints/ExplicitModelCheckerHint.h @@ -10,6 +10,10 @@ namespace storm { namespace modelchecker { + /*! + * This class contains information that might accelerate the model checking process. + * @note The model checker has to make sure whether a given hint is actually applicable and thus a hint might be ignored. + */ template class ExplicitModelCheckerHint : public ModelCheckerHint { public: @@ -37,9 +41,16 @@ namespace storm { void setSchedulerHint(boost::optional const& schedulerHint); void setSchedulerHint(boost::optional&& schedulerHint); + // If set, this option forces the application of the specified hints. This means that the model checker may skip some checks to increase performance. + // This might yield wrong results on certain models, e.g., when computing maximal probabilities on an MDP that has an end component consisting only of maybestates. + bool getForceApplicationOfHints() const; + void setForceApplicationOfHints(bool value); + private: boost::optional> resultHint; boost::optional schedulerHint; + + bool forceApplicationOfHints; }; } diff --git a/src/storm/modelchecker/hints/ModelCheckerHint.h b/src/storm/modelchecker/hints/ModelCheckerHint.h index 4c3842c53..37fc7ec44 100644 --- a/src/storm/modelchecker/hints/ModelCheckerHint.h +++ b/src/storm/modelchecker/hints/ModelCheckerHint.h @@ -8,8 +8,9 @@ namespace storm { class ExplicitModelCheckerHint; - /* - * This class contains information that might accelerate the solving process + /*! + * This class contains information that might accelerate the model checking process. + * @note The model checker has to make sure whether a given hint is actually applicable and thus a hint might be ignored. */ class ModelCheckerHint { public: diff --git a/src/storm/modelchecker/parametric/SparseInstantiationModelChecker.cpp b/src/storm/modelchecker/parametric/SparseInstantiationModelChecker.cpp index c42e5cef6..205fea5fe 100644 --- a/src/storm/modelchecker/parametric/SparseInstantiationModelChecker.cpp +++ b/src/storm/modelchecker/parametric/SparseInstantiationModelChecker.cpp @@ -24,6 +24,17 @@ namespace storm { currentCheckTask->setProduceSchedulers(checkTask.isProduceSchedulersSet()); } + template + storm::modelchecker::ModelCheckerHint& SparseInstantiationModelChecker::getHint() { + return currentCheckTask->getHint(); + } + + template + storm::modelchecker::ModelCheckerHint const& SparseInstantiationModelChecker::getHint() const { + return currentCheckTask->getHint(); + } + + template class SparseInstantiationModelChecker, double>; template class SparseInstantiationModelChecker, double>; diff --git a/src/storm/modelchecker/parametric/SparseInstantiationModelChecker.h b/src/storm/modelchecker/parametric/SparseInstantiationModelChecker.h index b1513b667..fb39772e1 100644 --- a/src/storm/modelchecker/parametric/SparseInstantiationModelChecker.h +++ b/src/storm/modelchecker/parametric/SparseInstantiationModelChecker.h @@ -3,6 +3,7 @@ #include "storm/logic/Formulas.h" #include "storm/modelchecker/CheckTask.h" #include "storm/modelchecker/results/CheckResult.h" +#include "storm/modelchecker/hints/ModelCheckerHint.h" #include "storm/utility/parametric.h" namespace storm { @@ -20,7 +21,10 @@ namespace storm { void specifyFormula(CheckTask const& checkTask); virtual std::unique_ptr check(storm::utility::parametric::Valuation const& valuation) = 0; - + + storm::modelchecker::ModelCheckerHint& getHint(); + storm::modelchecker::ModelCheckerHint const& getHint() const; + protected: SparseModelType const& parametricModel; diff --git a/src/storm/modelchecker/prctl/SparseMdpPrctlModelChecker.cpp b/src/storm/modelchecker/prctl/SparseMdpPrctlModelChecker.cpp index c8878d74c..b505734ed 100644 --- a/src/storm/modelchecker/prctl/SparseMdpPrctlModelChecker.cpp +++ b/src/storm/modelchecker/prctl/SparseMdpPrctlModelChecker.cpp @@ -144,8 +144,12 @@ namespace storm { STORM_LOG_THROW(checkTask.isOptimizationDirectionSet(), storm::exceptions::InvalidPropertyException, "Formula needs to specify whether minimal or maximal values are to be computed on nondeterministic model."); std::unique_ptr subResultPointer = this->check(eventuallyFormula.getSubformula()); ExplicitQualitativeCheckResult const& subResult = subResultPointer->asExplicitQualitativeCheckResult(); - std::vector numericResult = storm::modelchecker::helper::SparseMdpPrctlHelper::computeReachabilityRewards(checkTask.getOptimizationDirection(), this->getModel().getTransitionMatrix(), this->getModel().getBackwardTransitions(), checkTask.isRewardModelSet() ? this->getModel().getRewardModel(checkTask.getRewardModel()) : this->getModel().getRewardModel(""), subResult.getTruthValuesVector(), checkTask.isQualitativeSet(), *minMaxLinearEquationSolverFactory, checkTask.getHint()); - return std::unique_ptr(new ExplicitQuantitativeCheckResult(std::move(numericResult))); + auto ret = storm::modelchecker::helper::SparseMdpPrctlHelper::computeReachabilityRewards(checkTask.getOptimizationDirection(), this->getModel().getTransitionMatrix(), this->getModel().getBackwardTransitions(), checkTask.isRewardModelSet() ? this->getModel().getRewardModel(checkTask.getRewardModel()) : this->getModel().getRewardModel(""), subResult.getTruthValuesVector(), checkTask.isQualitativeSet(), checkTask.isProduceSchedulersSet(), *minMaxLinearEquationSolverFactory, checkTask.getHint()); + std::unique_ptr result(new ExplicitQuantitativeCheckResult(std::move(ret.values))); + if (checkTask.isProduceSchedulersSet() && ret.scheduler) { + result->asExplicitQuantitativeCheckResult().setScheduler(std::move(ret.scheduler)); + } + return result; } template diff --git a/src/storm/modelchecker/prctl/helper/SparseMdpPrctlHelper.cpp b/src/storm/modelchecker/prctl/helper/SparseMdpPrctlHelper.cpp index 1296676f2..5164dbbf7 100644 --- a/src/storm/modelchecker/prctl/helper/SparseMdpPrctlHelper.cpp +++ b/src/storm/modelchecker/prctl/helper/SparseMdpPrctlHelper.cpp @@ -121,13 +121,8 @@ namespace storm { // the accumulated probability of going from state i to some 'yes' state. std::vector b = transitionMatrix.getConstrainedRowGroupSumVector(maybeStates, statesWithProbability1); - // Prepare the solution vector for the maybestates. Apply the hint, if given and applicable - std::vector x(submatrix.getRowGroupCount()); - if(!hint.isEmpty() && (goal.minimize() || (maybeStates & storm::utility::graph::performProb0E(transitionMatrix, transitionMatrix.getRowGroupIndices(), backwardTransitions, maybeStates, ~maybeStates)).empty())) { - storm::utility::vector::selectVectorValues(x, maybeStates, hint.template asExplicitModelCheckerHint().getResultHint()); - } - - MDPSparseModelCheckingHelperReturnType resultForMaybeStates = computeUntilProbabilitiesOnlyMaybeStates(goal, submatrix, b, produceScheduler, minMaxLinearEquationSolverFactory, std::move(x)); + // Now compute the results for the maybeStates + MDPSparseModelCheckingHelperReturnType resultForMaybeStates = computeValuesOnlyMaybeStates(goal, transitionMatrix, backwardTransitions, maybeStates, submatrix, b, produceScheduler, minMaxLinearEquationSolverFactory, hint, goal.minimize(), storm::utility::zero(), storm::utility::one()); // Set values of resulting vector according to result. storm::utility::vector::setVectorValues(result, maybeStates, resultForMaybeStates.values); @@ -161,23 +156,44 @@ namespace storm { } template - MDPSparseModelCheckingHelperReturnType SparseMdpPrctlHelper::computeUntilProbabilitiesOnlyMaybeStates(storm::solver::SolveGoal const& goal, storm::storage::SparseMatrix const& submatrix, std::vector const& b, bool produceScheduler, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, std::vector&& x) { - // If requested, we will produce a scheduler. - std::unique_ptr scheduler; - - // make sure that the solution vector has the correct size. - x.resize(submatrix.getRowGroupCount()); + MDPSparseModelCheckingHelperReturnType SparseMdpPrctlHelper::computeValuesOnlyMaybeStates(storm::solver::SolveGoal const& goal, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& maybeStates, storm::storage::SparseMatrix const& submatrix, std::vector const& b, bool produceScheduler, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint, bool skipECWithinMaybeStatesCheck, boost::optional const& lowerResultBound, boost::optional const& upperResultBound) { - // Solve the corresponding system of equations. + // Set up the solver std::unique_ptr> solver = storm::solver::configureMinMaxLinearEquationSolver(goal, minMaxLinearEquationSolverFactory, submatrix); + if (lowerResultBound) { + solver->setLowerBound(lowerResultBound.get()); + } + if (upperResultBound) { + solver->setUpperBound(upperResultBound.get()); + } + // the scheduler hint is only applicable if it induces no BSCC consisting of maybestates + if(hint.isExplicitModelCheckerHint() && hint.template asExplicitModelCheckerHint().hasSchedulerHint() && + (hint.template asExplicitModelCheckerHint().getForceApplicationOfHints() || skipECWithinMaybeStatesCheck || + storm::utility::graph::performProb1(transitionMatrix.transposeSelectedRowsFromRowGroups(hint.template asExplicitModelCheckerHint().getSchedulerHint().getChoices()), maybeStates, ~maybeStates).full())) { + solver->setSchedulerHint(hint.template asExplicitModelCheckerHint().getSchedulerHint().getSchedulerForSubsystem(maybeStates)); + } solver->setTrackScheduler(produceScheduler); + + // Create the solution vector. + std::vector x; + // The result hint is only applicable if there are no End Components consisting of maybestates + if(hint.isExplicitModelCheckerHint() && hint.template asExplicitModelCheckerHint().hasResultHint() && + (hint.template asExplicitModelCheckerHint().getForceApplicationOfHints() || skipECWithinMaybeStatesCheck || solver->hasSchedulerHint() || + storm::utility::graph::performProb1A(transitionMatrix, transitionMatrix.getRowGroupIndices(), backwardTransitions, maybeStates, ~maybeStates).full())) { + x = storm::utility::vector::filterVector(hint.template asExplicitModelCheckerHint().getResultHint(), maybeStates); + } else { + x = std::vector(submatrix.getRowGroupCount(), lowerResultBound ? lowerResultBound.get() : storm::utility::zero()); + } + + // Solve the corresponding system of equations. solver->solveEquations(x, b); + // If requested, a scheduler was produced if (produceScheduler) { - scheduler = solver->getScheduler(); + return MDPSparseModelCheckingHelperReturnType(std::move(x), std::move(solver->getScheduler())); + } else { + return MDPSparseModelCheckingHelperReturnType(std::move(x)); } - - return MDPSparseModelCheckingHelperReturnType(std::move(x), std::move(scheduler)); } template @@ -247,17 +263,28 @@ namespace storm { return result; } - template template - std::vector SparseMdpPrctlHelper::computeReachabilityRewards(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, RewardModelType const& rewardModel, storm::storage::BitVector const& targetStates, bool qualitative, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint) { + MDPSparseModelCheckingHelperReturnType SparseMdpPrctlHelper::computeReachabilityRewards(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, RewardModelType const& rewardModel, storm::storage::BitVector const& targetStates, bool qualitative, bool produceScheduler, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint) { // Only compute the result if the model has at least one reward this->getModel(). STORM_LOG_THROW(!rewardModel.empty(), storm::exceptions::InvalidPropertyException, "Missing reward model for formula. Skipping formula."); - return computeReachabilityRewardsHelper(dir, transitionMatrix, backwardTransitions, + return computeReachabilityRewardsHelper(storm::solver::SolveGoal(dir), transitionMatrix, backwardTransitions, [&rewardModel] (uint_fast64_t rowCount, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::BitVector const& maybeStates) { return rewardModel.getTotalRewardVector(rowCount, transitionMatrix, maybeStates); }, - targetStates, qualitative, minMaxLinearEquationSolverFactory, hint); + targetStates, qualitative, produceScheduler, minMaxLinearEquationSolverFactory, hint); + } + + template + template + MDPSparseModelCheckingHelperReturnType SparseMdpPrctlHelper::computeReachabilityRewards(storm::solver::SolveGoal const& goal, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, RewardModelType const& rewardModel, storm::storage::BitVector const& targetStates, bool qualitative, bool produceScheduler, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint) { + // Only compute the result if the model has at least one reward this->getModel(). + STORM_LOG_THROW(!rewardModel.empty(), storm::exceptions::InvalidPropertyException, "Missing reward model for formula. Skipping formula."); + return computeReachabilityRewardsHelper(goal, transitionMatrix, backwardTransitions, + [&rewardModel] (uint_fast64_t rowCount, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::BitVector const& maybeStates) { + return rewardModel.getTotalRewardVector(rowCount, transitionMatrix, maybeStates); + }, + targetStates, qualitative, produceScheduler, minMaxLinearEquationSolverFactory, hint); } #ifdef STORM_HAVE_CARL @@ -265,7 +292,7 @@ namespace storm { std::vector SparseMdpPrctlHelper::computeReachabilityRewards(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, storm::models::sparse::StandardRewardModel const& intervalRewardModel, bool lowerBoundOfIntervals, storm::storage::BitVector const& targetStates, bool qualitative, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory) { // Only compute the result if the reward model is not empty. STORM_LOG_THROW(!intervalRewardModel.empty(), storm::exceptions::InvalidPropertyException, "Missing reward model for formula. Skipping formula."); - return computeReachabilityRewardsHelper(dir, transitionMatrix, backwardTransitions, \ + return computeReachabilityRewardsHelper(storm::solver::SolveGoal(dir), transitionMatrix, backwardTransitions, \ [&] (uint_fast64_t rowCount, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::BitVector const& maybeStates) { std::vector result; result.reserve(rowCount); @@ -275,7 +302,7 @@ namespace storm { } return result; }, \ - targetStates, qualitative, minMaxLinearEquationSolverFactory); + targetStates, qualitative, false, minMaxLinearEquationSolverFactory).values; } template<> @@ -285,14 +312,14 @@ namespace storm { #endif template - std::vector SparseMdpPrctlHelper::computeReachabilityRewardsHelper(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, std::function(uint_fast64_t, storm::storage::SparseMatrix const&, storm::storage::BitVector const&)> const& totalStateRewardVectorGetter, storm::storage::BitVector const& targetStates, bool qualitative, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint) { + MDPSparseModelCheckingHelperReturnType SparseMdpPrctlHelper::computeReachabilityRewardsHelper(storm::solver::SolveGoal const& goal, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, std::function(uint_fast64_t, storm::storage::SparseMatrix const&, storm::storage::BitVector const&)> const& totalStateRewardVectorGetter, storm::storage::BitVector const& targetStates, bool qualitative, bool produceScheduler, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint) { std::vector const& nondeterministicChoiceIndices = transitionMatrix.getRowGroupIndices(); // Determine which states have a reward of infinity by definition. storm::storage::BitVector infinityStates; storm::storage::BitVector trueStates(transitionMatrix.getRowGroupCount(), true); - if (dir == OptimizationDirection::Minimize) { + if (goal.minimize()) { STORM_LOG_WARN("Results of reward computation may be too low, because of zero-reward loops."); infinityStates = storm::utility::graph::performProb1E(transitionMatrix, nondeterministicChoiceIndices, backwardTransitions, trueStates, targetStates); } else { @@ -307,6 +334,15 @@ namespace storm { // Create resulting vector. std::vector result(transitionMatrix.getRowGroupCount(), storm::utility::zero()); + // Set values of resulting vector that are known exactly. + storm::utility::vector::setVectorValues(result, infinityStates, storm::utility::infinity()); + + // If requested, we will produce a scheduler. + std::unique_ptr scheduler; + if (produceScheduler) { + scheduler = std::make_unique(transitionMatrix.getRowGroupCount()); + } + // Check whether we need to compute exact rewards for some states. if (qualitative) { STORM_LOG_INFO("The rewards for the initial states were determined in a preprocessing step. No exact rewards were computed."); @@ -337,36 +373,37 @@ namespace storm { } } } - - // Create vector for results for maybe states and a solver object. - std::vector x; - std::unique_ptr> solver = minMaxLinearEquationSolverFactory.create(std::move(submatrix)); - - // If applicable, consider the given hint(s) - if(hint.isExplicitModelCheckerHint() && hint.template asExplicitModelCheckerHint().hasResultHint() && - (hint.template asExplicitModelCheckerHint().hasSchedulerHint() || - dir == OptimizationDirection::Maximize || - (maybeStates & storm::utility::graph::performProb0E(transitionMatrix, transitionMatrix.getRowGroupIndices(), backwardTransitions, maybeStates, ~maybeStates)).empty())) { - x = storm::utility::vector::filterVector(hint.template asExplicitModelCheckerHint().getResultHint(), maybeStates); - } else { - x = std::vector(maybeStates.getNumberOfSetBits(), storm::utility::zero()); - } - if(hint.isExplicitModelCheckerHint() && hint.template asExplicitModelCheckerHint().hasSchedulerHint()) { - solver->setSchedulerHint(hint.template asExplicitModelCheckerHint().getSchedulerHint().getSchedulerForSubsystem(maybeStates)); - } - - // Solve the corresponding system of equations. - solver->solveEquations(dir, x, b); + // Now compute the results for the maybeStates + MDPSparseModelCheckingHelperReturnType resultForMaybeStates = computeValuesOnlyMaybeStates(goal, transitionMatrix, backwardTransitions, maybeStates, submatrix, b, produceScheduler, minMaxLinearEquationSolverFactory, hint, !goal.minimize(), storm::utility::zero()); + // Set values of resulting vector according to result. - storm::utility::vector::setVectorValues(result, maybeStates, x); + storm::utility::vector::setVectorValues(result, maybeStates, resultForMaybeStates.values); + + if (produceScheduler) { + storm::storage::Scheduler const& subscheduler = *resultForMaybeStates.scheduler; + uint_fast64_t currentSubState = 0; + for (auto maybeState : maybeStates) { + scheduler->setChoice(maybeState, subscheduler.getChoice(currentSubState)); + ++currentSubState; + } + } } } - // Set values of resulting vector that are known exactly. - storm::utility::vector::setVectorValues(result, infinityStates, storm::utility::infinity()); + // Finally, if we need to produce a scheduler, we also need to figure out the parts of the scheduler for + // the states with reward infinity. + if (produceScheduler) { + storm::storage::PartialScheduler relevantQualitativeScheduler; + if (!goal.minimize()) { + relevantQualitativeScheduler = storm::utility::graph::computeSchedulerProb0E(infinityStates, transitionMatrix); + } + for (auto const& stateChoicePair : relevantQualitativeScheduler) { + scheduler->setChoice(stateChoicePair.first, stateChoicePair.second); + } + } - return result; + return MDPSparseModelCheckingHelperReturnType(std::move(result), std::move(scheduler)); } template @@ -669,13 +706,15 @@ namespace storm { template class SparseMdpPrctlHelper; template std::vector SparseMdpPrctlHelper::computeInstantaneousRewards(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::models::sparse::StandardRewardModel const& rewardModel, uint_fast64_t stepCount, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory); template std::vector SparseMdpPrctlHelper::computeCumulativeRewards(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::models::sparse::StandardRewardModel const& rewardModel, uint_fast64_t stepBound, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory); - template std::vector SparseMdpPrctlHelper::computeReachabilityRewards(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, storm::models::sparse::StandardRewardModel const& rewardModel, storm::storage::BitVector const& targetStates, bool qualitative, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint); + template MDPSparseModelCheckingHelperReturnType SparseMdpPrctlHelper::computeReachabilityRewards(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, storm::models::sparse::StandardRewardModel const& rewardModel, storm::storage::BitVector const& targetStates, bool qualitative, bool produceScheduler, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint); + template MDPSparseModelCheckingHelperReturnType SparseMdpPrctlHelper::computeReachabilityRewards(storm::solver::SolveGoal const& goal, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, storm::models::sparse::StandardRewardModel const& rewardModel, storm::storage::BitVector const& targetStates, bool qualitative, bool produceScheduler, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint); #ifdef STORM_HAVE_CARL template class SparseMdpPrctlHelper; template std::vector SparseMdpPrctlHelper::computeInstantaneousRewards(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::models::sparse::StandardRewardModel const& rewardModel, uint_fast64_t stepCount, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory); template std::vector SparseMdpPrctlHelper::computeCumulativeRewards(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::models::sparse::StandardRewardModel const& rewardModel, uint_fast64_t stepBound, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory); - template std::vector SparseMdpPrctlHelper::computeReachabilityRewards(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, storm::models::sparse::StandardRewardModel const& rewardModel, storm::storage::BitVector const& targetStates, bool qualitative, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint); + template MDPSparseModelCheckingHelperReturnType SparseMdpPrctlHelper::computeReachabilityRewards(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, storm::models::sparse::StandardRewardModel const& rewardModel, storm::storage::BitVector const& targetStates, bool qualitative, bool produceScheduler, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint); + template MDPSparseModelCheckingHelperReturnType SparseMdpPrctlHelper::computeReachabilityRewards(storm::solver::SolveGoal const& goal, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, storm::models::sparse::StandardRewardModel const& rewardModel, storm::storage::BitVector const& targetStates, bool qualitative, bool produceScheduler, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint); #endif } } diff --git a/src/storm/modelchecker/prctl/helper/SparseMdpPrctlHelper.h b/src/storm/modelchecker/prctl/helper/SparseMdpPrctlHelper.h index 7210cd6c8..2eb4ae3ae 100644 --- a/src/storm/modelchecker/prctl/helper/SparseMdpPrctlHelper.h +++ b/src/storm/modelchecker/prctl/helper/SparseMdpPrctlHelper.h @@ -39,7 +39,7 @@ namespace storm { static MDPSparseModelCheckingHelperReturnType computeUntilProbabilities(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates, bool qualitative, bool produceScheduler, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint = ModelCheckerHint()); - static MDPSparseModelCheckingHelperReturnType computeUntilProbabilitiesOnlyMaybeStates(storm::solver::SolveGoal const& goal, storm::storage::SparseMatrix const& submatrix, std::vector const& b, bool produceScheduler, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, std::vector&& x = std::vector()); + static MDPSparseModelCheckingHelperReturnType computeValuesOnlyMaybeStates(storm::solver::SolveGoal const& goal, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& maybeStates, storm::storage::SparseMatrix const& submatrix, std::vector const& b, bool produceScheduler, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint = ModelCheckerHint(), bool skipECWithinMaybeStatesCheck = false, boost::optional const& lowerResultBound = boost::none, boost::optional const& upperResultBound = boost::none); static MDPSparseModelCheckingHelperReturnType computeUntilProbabilities(storm::solver::SolveGoal const& goal, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates, bool qualitative, bool produceScheduler, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint = ModelCheckerHint()); @@ -52,7 +52,10 @@ namespace storm { static std::vector computeCumulativeRewards(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, RewardModelType const& rewardModel, uint_fast64_t stepBound, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory); template - static std::vector computeReachabilityRewards(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, RewardModelType const& rewardModel, storm::storage::BitVector const& targetStates, bool qualitative, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint = ModelCheckerHint()); + static MDPSparseModelCheckingHelperReturnType computeReachabilityRewards(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, RewardModelType const& rewardModel, storm::storage::BitVector const& targetStates, bool qualitative, bool produceScheduler, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint = ModelCheckerHint()); + + template + static MDPSparseModelCheckingHelperReturnType computeReachabilityRewards(storm::solver::SolveGoal const& goal, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, RewardModelType const& rewardModel, storm::storage::BitVector const& targetStates, bool qualitative, bool produceScheduler, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint = ModelCheckerHint()); #ifdef STORM_HAVE_CARL static std::vector computeReachabilityRewards(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, storm::models::sparse::StandardRewardModel const& intervalRewardModel, bool lowerBoundOfIntervals, storm::storage::BitVector const& targetStates, bool qualitative, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory); @@ -63,7 +66,7 @@ namespace storm { static std::unique_ptr computeConditionalProbabilities(OptimizationDirection dir, storm::storage::sparse::state_type initialState, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& targetStates, storm::storage::BitVector const& conditionStates, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory); private: - static std::vector computeReachabilityRewardsHelper(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, std::function(uint_fast64_t, storm::storage::SparseMatrix const&, storm::storage::BitVector const&)> const& totalStateRewardVectorGetter, storm::storage::BitVector const& targetStates, bool qualitative, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint = ModelCheckerHint()); + static MDPSparseModelCheckingHelperReturnType computeReachabilityRewardsHelper(storm::solver::SolveGoal const& goal, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::SparseMatrix const& backwardTransitions, std::function(uint_fast64_t, storm::storage::SparseMatrix const&, storm::storage::BitVector const&)> const& totalStateRewardVectorGetter, storm::storage::BitVector const& targetStates, bool qualitative, bool produceScheduler, storm::solver::MinMaxLinearEquationSolverFactory const& minMaxLinearEquationSolverFactory, ModelCheckerHint const& hint = ModelCheckerHint()); static ValueType computeLraForMaximalEndComponent(OptimizationDirection dir, storm::storage::SparseMatrix const& transitionMatrix, storm::storage::BitVector const& goalStates, storm::storage::MaximalEndComponent const& mec); diff --git a/src/storm/solver/MinMaxLinearEquationSolver.cpp b/src/storm/solver/MinMaxLinearEquationSolver.cpp index 02269a248..b159d730f 100644 --- a/src/storm/solver/MinMaxLinearEquationSolver.cpp +++ b/src/storm/solver/MinMaxLinearEquationSolver.cpp @@ -114,6 +114,16 @@ namespace storm { setUpperBound(upper); } + template + void MinMaxLinearEquationSolver::setSchedulerHint(storm::storage::TotalScheduler&& scheduler) { + schedulerHint = scheduler; + } + + template + bool MinMaxLinearEquationSolver::hasSchedulerHint() const { + return schedulerHint.is_initialized(); + } + template MinMaxLinearEquationSolverFactory::MinMaxLinearEquationSolverFactory(bool trackScheduler) : trackScheduler(trackScheduler) { // Intentionally left empty. diff --git a/src/storm/solver/MinMaxLinearEquationSolver.h b/src/storm/solver/MinMaxLinearEquationSolver.h index 473438a38..aa5a92bc8 100644 --- a/src/storm/solver/MinMaxLinearEquationSolver.h +++ b/src/storm/solver/MinMaxLinearEquationSolver.h @@ -159,6 +159,16 @@ namespace storm { * Sets bounds for the solution that can potentially used by the solver. */ void setBounds(ValueType const& lower, ValueType const& upper); + + /*! + * Sets a scheduler that might be considered by the solver as an initial guess + */ + void setSchedulerHint(storm::storage::TotalScheduler&& scheduler); + + /*! + * Returns true iff a scheduler hint was defined + */ + bool hasSchedulerHint() const; protected: /// The optimization direction to use for calls to functions that do not provide it explicitly. Can also be unset. @@ -176,6 +186,9 @@ namespace storm { // An upper bound if one was set. boost::optional upperBound; + // A scheduler that might be considered by the solver as an initial guess + boost::optional schedulerHint; + private: /// Whether some of the generated data during solver calls should be cached. bool cachingEnabled; diff --git a/src/storm/solver/StandardMinMaxLinearEquationSolver.cpp b/src/storm/solver/StandardMinMaxLinearEquationSolver.cpp index c9c405341..d5e25df46 100644 --- a/src/storm/solver/StandardMinMaxLinearEquationSolver.cpp +++ b/src/storm/solver/StandardMinMaxLinearEquationSolver.cpp @@ -97,7 +97,7 @@ namespace storm { template bool StandardMinMaxLinearEquationSolver::solveEquationsPolicyIteration(OptimizationDirection dir, std::vector& x, std::vector const& b) const { // Create the initial scheduler. - std::vector scheduler(this->A.getRowGroupCount()); + std::vector scheduler = this->schedulerHint ? this->schedulerHint->getChoices() : std::vector(this->A.getRowGroupCount()); // Get a vector for storing the right-hand side of the inner equation system. if(!auxiliaryRowGroupVector) { @@ -229,6 +229,19 @@ namespace storm { if (!auxiliaryRowGroupVector.get()) { auxiliaryRowGroupVector = std::make_unique>(A.getRowGroupCount()); } + + if(this->schedulerHint) { + // Resolve the nondeterminism according to the scheduler hint and solve the resulting equation system. + storm::storage::SparseMatrix submatrix = this->A.selectRowsFromRowGroups(this->schedulerHint->getChoices(), true); + submatrix.convertToEquationSystem(); + storm::utility::vector::selectVectorValues(*auxiliaryRowGroupVector, this->schedulerHint->getChoices(), this->A.getRowGroupIndices(), b); + + auto submatrixSolver = linearEquationSolverFactory->create(std::move(submatrix)); + if (this->lowerBound) { submatrixSolver->setLowerBound(this->lowerBound.get()); } + if (this->upperBound) { submatrixSolver->setUpperBound(this->upperBound.get()); } + submatrixSolver->solveEquations(x, *auxiliaryRowGroupVector); + } + std::vector* newX = auxiliaryRowGroupVector.get(); std::vector* currentX = &x; diff --git a/src/storm/storage/SparseMatrix.cpp b/src/storm/storage/SparseMatrix.cpp index 42b762b64..4bb080952 100644 --- a/src/storm/storage/SparseMatrix.cpp +++ b/src/storm/storage/SparseMatrix.cpp @@ -1043,6 +1043,54 @@ namespace storm { return transposedMatrix; } + + template + SparseMatrix SparseMatrix::transposeSelectedRowsFromRowGroups(std::vector const& rowGroupChoices, bool keepZeros) const { + index_type rowCount = this->getColumnCount(); + index_type columnCount = this->getRowGroupCount(); + + // Get the overall entry count as well as the number of entries of each column + index_type entryCount = 0; + std::vector rowIndications(columnCount + 1); + auto rowGroupChoiceIt = rowGroupChoices.begin(); + for (index_type rowGroup = 0; rowGroup < columnCount; ++rowGroup, ++rowGroupChoiceIt) { + for(auto const& entry : this->getRow(rowGroup, *rowGroupChoiceIt)) { + if(keepZeros || !storm::utility::isZero(entry.getValue())) { + ++entryCount; + ++rowIndications[entry.getColumn() + 1]; + } + } + } + + // Now compute the accumulated offsets. + for (index_type i = 1; i < rowCount + 1; ++i) { + rowIndications[i] = rowIndications[i - 1] + rowIndications[i]; + } + + std::vector> columnsAndValues(entryCount); + + // Create an array that stores the index for the next value to be added for + // each row in the transposed matrix. Initially this corresponds to the previously + // computed accumulated offsets. + std::vector nextIndices = rowIndications; + + // Now we are ready to actually fill in the values of the transposed matrix. + rowGroupChoiceIt = rowGroupChoices.begin(); + for (index_type rowGroup = 0; rowGroup < columnCount; ++rowGroup, ++rowGroupChoiceIt) { + for(auto const& entry : this->getRow(rowGroup, *rowGroupChoiceIt)) { + if(keepZeros || !storm::utility::isZero(entry.getValue())) { + columnsAndValues[nextIndices[entry.getColumn()]] = std::make_pair(rowGroup, entry.getValue()); + ++nextIndices[entry.getColumn()]; + } + } + } + + // TODO: remove this assertion + auto result = storm::storage::SparseMatrix(std::move(columnCount), std::move(rowIndications), std::move(columnsAndValues), boost::none); + STORM_LOG_ASSERT(result == selectRowsFromRowGroups(rowGroupChoices, false).transpose(false, keepZeros), "Expected that the two matrices are equal"); + return result; + // return storm::storage::SparseMatrix(std::move(columnCount), std::move(rowIndications), std::move(columnsAndValues), boost::none); + } template void SparseMatrix::convertToEquationSystem() { diff --git a/src/storm/storage/SparseMatrix.h b/src/storm/storage/SparseMatrix.h index f897e1606..1b6da0454 100644 --- a/src/storm/storage/SparseMatrix.h +++ b/src/storm/storage/SparseMatrix.h @@ -673,8 +673,7 @@ namespace storm { /*! * Selects exactly one row from each row group of this matrix and returns the resulting matrix. * - * @param rowGroupToRowIndexMapping A mapping from each row group index to a selected row in this group. - * @param insertDiagonalEntries If set to true, the resulting matrix will have zero entries in column i for +s * @param insertDiagonalEntries If set to true, the resulting matrix will have zero entries in column i for * each row in row group i. This can then be used for inserting other values later. * @return A submatrix of the current matrix by selecting one row out of each row group. */ @@ -701,6 +700,17 @@ namespace storm { */ storm::storage::SparseMatrix transpose(bool joinGroups = false, bool keepZeros = false) const; + + /*! + * Transposes the matrix w.r.t. the selected rows. + * This is equivalent to selectRowsFromRowGroups(rowGroupChoices, false).transpose(false, keepZeros) but avoids creating one intermediate matrix. + * + * @param rowGroupChoices A mapping from each row group index to a selected row in this group. + * @param keepZeros A flag indicating whether entries with value zero should be kept. + * + */ + SparseMatrix transposeSelectedRowsFromRowGroups(std::vector const& rowGroupChoices, bool keepZeros = false) const; + /*! * Transforms the matrix into an equation system. That is, it transforms the matrix A into a matrix (1-A). */