Browse Source
First version of minimum expected time for Markov automata.
First version of minimum expected time for Markov automata.
Former-commit-id: 6053be896e
tempestpy_adaptions
dehnert
11 years ago
8 changed files with 184 additions and 8 deletions
-
4src/formula/Csl/AbstractNoBoundOperator.h
-
2src/formula/abstract/StateNoBoundOperator.h
-
2src/modelchecker/csl/AbstractModelChecker.h
-
153src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h
-
2src/modelchecker/prctl/AbstractModelChecker.h
-
13src/models/MarkovAutomaton.h
-
7src/parser/MarkovAutomatonSparseTransitionParser.cpp
-
7src/storm.cpp
@ -0,0 +1,153 @@ |
|||||
|
#ifndef STORM_MODELCHECKER_CSL_SPARSEMARKOVAUTOMATONCSLMODELCHECKER_H_ |
||||
|
#define STORM_MODELCHECKER_CSL_SPARSEMARKOVAUTOMATONCSLMODELCHECKER_H_ |
||||
|
|
||||
|
#include <stack> |
||||
|
|
||||
|
#include "src/modelchecker/csl/AbstractModelChecker.h" |
||||
|
#include "src/models/MarkovAutomaton.h" |
||||
|
#include "src/storage/BitVector.h" |
||||
|
#include "src/solver/AbstractNondeterministicLinearEquationSolver.h" |
||||
|
#include "src/exceptions/NotImplementedException.h" |
||||
|
|
||||
|
namespace storm { |
||||
|
namespace modelchecker { |
||||
|
namespace csl { |
||||
|
|
||||
|
template<typename ValueType> |
||||
|
class SparseMarkovAutomatonCslModelChecker : public AbstractModelChecker<ValueType> { |
||||
|
public: |
||||
|
|
||||
|
explicit SparseMarkovAutomatonCslModelChecker(storm::models::MarkovAutomaton<ValueType> const& model, storm::solver::AbstractNondeterministicLinearEquationSolver<ValueType>* linearEquationSolver) : AbstractModelChecker<ValueType>(model), minimumOperatorStack(), linearEquationSolver(linearEquationSolver) { |
||||
|
// Intentionally left empty. |
||||
|
} |
||||
|
|
||||
|
/*! |
||||
|
* Returns a constant reference to the MDP associated with this model checker. |
||||
|
* @returns A constant reference to the MDP associated with this model checker. |
||||
|
*/ |
||||
|
storm::models::MarkovAutomaton<ValueType> const& getModel() const { |
||||
|
return AbstractModelChecker<ValueType>::template getModel<storm::models::MarkovAutomaton<ValueType>>(); |
||||
|
} |
||||
|
|
||||
|
std::vector<ValueType> checkUntil(storm::property::csl::Until<ValueType> const& formula, bool qualitative) const { |
||||
|
throw storm::exceptions::NotImplementedException() << "Model checking Until formulas on Markov automata is not yet implemented."; |
||||
|
} |
||||
|
|
||||
|
std::vector<ValueType> checkTimeBoundedUntil(storm::property::csl::TimeBoundedUntil<ValueType> const& formula, bool qualitative) const { |
||||
|
throw storm::exceptions::NotImplementedException() << "Model checking Until formulas on Markov automata is not yet implemented."; |
||||
|
} |
||||
|
|
||||
|
std::vector<ValueType> checkTimeBoundedEventually(storm::property::csl::TimeBoundedEventually<ValueType> const& formula, bool qualitative) const { |
||||
|
throw storm::exceptions::NotImplementedException() << "Model checking time-bounded Until formulas on Markov automata is not yet implemented."; |
||||
|
} |
||||
|
|
||||
|
std::vector<ValueType> checkGlobally(storm::property::csl::Globally<ValueType> const& formula, bool qualitative) const { |
||||
|
throw storm::exceptions::NotImplementedException() << "Model checking Globally formulas on Markov automata is not yet implemented."; |
||||
|
} |
||||
|
|
||||
|
std::vector<ValueType> checkEventually(storm::property::csl::Eventually<ValueType> const& formula, bool qualitative) const { |
||||
|
throw storm::exceptions::NotImplementedException() << "Model checking Eventually formulas on Markov automata is not yet implemented."; |
||||
|
} |
||||
|
|
||||
|
std::vector<ValueType> checkNext(storm::property::csl::Next<ValueType> const& formula, bool qualitative) const { |
||||
|
throw storm::exceptions::NotImplementedException() << "Model checking Next formulas on Markov automata is not yet implemented."; |
||||
|
} |
||||
|
|
||||
|
std::vector<ValueType> checkNoBoundOperator(storm::property::csl::AbstractNoBoundOperator<ValueType> const& formula) const { |
||||
|
// Check if the operator was an non-optimality operator and report an error in that case. |
||||
|
if (!formula.isOptimalityOperator()) { |
||||
|
LOG4CPLUS_ERROR(logger, "Formula does not specify neither min nor max optimality, which is not meaningful for nondeterministic models."); |
||||
|
throw storm::exceptions::InvalidArgumentException() << "Formula does not specify neither min nor max optimality, which is not meaningful for nondeterministic models."; |
||||
|
} |
||||
|
minimumOperatorStack.push(formula.isMinimumOperator()); |
||||
|
std::vector<ValueType> result = formula.check(*this, false); |
||||
|
minimumOperatorStack.pop(); |
||||
|
return result; |
||||
|
} |
||||
|
|
||||
|
std::vector<ValueType> checkExpectedTime(bool min, storm::storage::BitVector const& goalStates) const { |
||||
|
// TODO: check whether the Markov automaton is closed once again? Or should that rather be done when constructing the model checker? |
||||
|
// For now we just assume that it is closed already. |
||||
|
|
||||
|
// Start by identifying the states for which values need to be computed. |
||||
|
storm::storage::BitVector maybeStates = ~goalStates; |
||||
|
|
||||
|
// Then, we can eliminate the rows and columns for all states whose values are already known to be 0. |
||||
|
std::vector<ValueType> x(maybeStates.getNumberOfSetBits()); |
||||
|
std::vector<uint_fast64_t> subNondeterministicChoiceIndices = this->computeNondeterministicChoiceIndicesForConstraint(maybeStates); |
||||
|
storm::storage::SparseMatrix<ValueType> submatrix = this->getModel().getTransitionMatrix().getSubmatrix(maybeStates, this->getModel().getNondeterministicChoiceIndices()); |
||||
|
|
||||
|
// Now prepare the mean sojourn times for all states so they can be used as the right-hand side of the equation system. |
||||
|
std::vector<ValueType> meanSojournTimes(this->getModel().getExitRates()); |
||||
|
for (auto state : this->getModel().getMarkovianStates()) { |
||||
|
meanSojournTimes[state] = storm::utility::constGetOne<ValueType>() / meanSojournTimes[state]; |
||||
|
} |
||||
|
|
||||
|
// Finally, prepare the actual right-hand side. |
||||
|
std::vector<ValueType> b(submatrix.getRowCount()); |
||||
|
storm::utility::vector::selectVectorValuesRepeatedly(b, maybeStates, this->getModel().getNondeterministicChoiceIndices(), meanSojournTimes); |
||||
|
|
||||
|
// Solve the corresponding system of equations. |
||||
|
if (linearEquationSolver != nullptr) { |
||||
|
this->linearEquationSolver->solveEquationSystem(min, submatrix, x, b, subNondeterministicChoiceIndices); |
||||
|
} else { |
||||
|
throw storm::exceptions::InvalidStateException() << "No valid linear equation solver available."; |
||||
|
} |
||||
|
|
||||
|
// Create resulting vector. |
||||
|
std::vector<ValueType> result(this->getModel().getNumberOfStates()); |
||||
|
|
||||
|
// Set values of resulting vector according to previous result (goal states will have a value of 0) and return the result. |
||||
|
storm::utility::vector::setVectorValues<ValueType>(result, maybeStates, x); |
||||
|
return result; |
||||
|
} |
||||
|
|
||||
|
protected: |
||||
|
/*! |
||||
|
* A stack used for storing whether we are currently computing min or max probabilities or rewards, respectively. |
||||
|
* The topmost element is true if and only if we are currently computing minimum probabilities or rewards. |
||||
|
*/ |
||||
|
mutable std::stack<bool> minimumOperatorStack; |
||||
|
|
||||
|
private: |
||||
|
/*! |
||||
|
* Computes the nondeterministic choice indices vector resulting from reducing the full system to the states given |
||||
|
* by the parameter constraint. |
||||
|
* |
||||
|
* @param constraint A bit vector specifying which states are kept. |
||||
|
* @returns A vector of the nondeterministic choice indices of the subsystem induced by the given constraint. |
||||
|
*/ |
||||
|
std::vector<uint_fast64_t> computeNondeterministicChoiceIndicesForConstraint(storm::storage::BitVector const& constraint) const { |
||||
|
// First, get a reference to the full nondeterministic choice indices. |
||||
|
std::vector<uint_fast64_t> const& nondeterministicChoiceIndices = this->getModel().getNondeterministicChoiceIndices(); |
||||
|
|
||||
|
// Reserve the known amount of slots for the resulting vector. |
||||
|
std::vector<uint_fast64_t> subNondeterministicChoiceIndices(constraint.getNumberOfSetBits() + 1); |
||||
|
uint_fast64_t currentRowCount = 0; |
||||
|
uint_fast64_t currentIndexCount = 1; |
||||
|
|
||||
|
// Set the first element as this will clearly begin at offset 0. |
||||
|
subNondeterministicChoiceIndices[0] = 0; |
||||
|
|
||||
|
// Loop over all states that need to be kept and copy the relative indices of the nondeterministic choices over |
||||
|
// to the resulting vector. |
||||
|
for (auto index : constraint) { |
||||
|
subNondeterministicChoiceIndices[currentIndexCount] = currentRowCount + nondeterministicChoiceIndices[index + 1] - nondeterministicChoiceIndices[index]; |
||||
|
currentRowCount += nondeterministicChoiceIndices[index + 1] - nondeterministicChoiceIndices[index]; |
||||
|
++currentIndexCount; |
||||
|
} |
||||
|
|
||||
|
// Put a sentinel element at the end. |
||||
|
subNondeterministicChoiceIndices[constraint.getNumberOfSetBits()] = currentRowCount; |
||||
|
|
||||
|
return subNondeterministicChoiceIndices; |
||||
|
} |
||||
|
|
||||
|
// An object that is used for solving linear equations and performing matrix-vector multiplication. |
||||
|
std::unique_ptr<storm::solver::AbstractNondeterministicLinearEquationSolver<ValueType>> linearEquationSolver; |
||||
|
}; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
#endif /* STORM_MODELCHECKER_CSL_SPARSEMARKOVAUTOMATONCSLMODELCHECKER_H_ */ |
Write
Preview
Loading…
Cancel
Save
Reference in new issue