From dce43d78e7ba35c505b1d68102e46e06212f36e4 Mon Sep 17 00:00:00 2001 From: dehnert Date: Sat, 30 Nov 2013 21:09:51 +0100 Subject: [PATCH 1/9] Started implementation of time-bounded reachability of Markov automata. Former-commit-id: 512bb117a62098a8c41d9b29f22a65c3f8660478 --- .../SparseMarkovAutomatonCslModelChecker.h | 93 +++++++- src/models/MarkovAutomaton.h | 8 + src/storage/SparseMatrix.h | 217 ++++++++++++++++-- src/storm.cpp | 2 + 4 files changed, 296 insertions(+), 24 deletions(-) diff --git a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h index c48e3b6b3..e35c9bca1 100644 --- a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h +++ b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h @@ -2,6 +2,7 @@ #define STORM_MODELCHECKER_CSL_SPARSEMARKOVAUTOMATONCSLMODELCHECKER_H_ #include +#include #include "src/modelchecker/csl/AbstractModelChecker.h" #include "src/models/MarkovAutomaton.h" @@ -69,35 +70,105 @@ namespace storm { return result; } - std::vector checkTimeBoundedEventually(bool min, storm::storage::BitVector const& goalStates) const { + std::vector checkTimeBoundedEventually(bool min, storm::storage::BitVector const& goalStates, uint_fast64_t lowerBound, uint_fast64_t upperBound) const { if (!this->getModel().isClosed()) { throw storm::exceptions::InvalidArgumentException() << "Unable to compute time-bounded reachability on non-closed Markov automaton."; } // (1) Compute the number of steps we need to take. + ValueType lambda = this->getModel().getMaximalExitRate(); + ValueType delta = (2 * storm::settings::Settings::getInstance()->getOptionByLongName("precision").getArgument(0).getValueAsDouble()) / (upperBound * lambda * lambda); + LOG4CPLUS_INFO(logger, "Determined delta to be " << delta << "."); + + // Get some data fields for convenient access. + std::vector const& nondeterministicChoiceIndices = this->getModel().getNondeterministicChoiceIndices(); + std::vector const& exitRates = this->getModel().getExitRates(); + typename storm::storage::SparseMatrix const& transitionMatrix = this->getModel().getTransitionMatrix(); // (2) Compute four sparse matrices: - // * a matrix A_MSwG with all (discretized!) transitions from Markovian non-goal states to *all other* non-goal states. Note: this matrix has more columns than rows. - // * a matrix A_PSwg with all (non-discretized) transitions from probabilistic non-goal states to other probabilistic non-goal states. This matrix has more rows than columns. + // * a matrix A_MS with all (discretized) transitions from Markovian non-goal states to all Markovian non-goal states. + // * a matrix A_MStoPS with all (discretized) transitions from Markovian non-goal states to all probabilistic non-goal states. + // * a matrix A_PS with all (non-discretized) transitions from probabilistic non-goal states to other probabilistic non-goal states. This matrix has more rows than columns. // * a matrix A_PStoMS with all (non-discretized) transitions from probabilistic non-goal states to all Markovian non-goal states. This matrix may have any shape. + storm::storage::BitVector const& markovianNonGoalStates = this->getModel().getMarkovianStates() & ~goalStates; + storm::storage::BitVector const& probabilisticNonGoalStates = ~this->getModel().getMarkovianStates() & ~goalStates; + + typename storm::storage::SparseMatrix aMarkovian = this->getModel().getTransitionMatrix().getSubmatrix(markovianNonGoalStates, this->getModel().getNondeterministicChoiceIndices(), true); + typename storm::storage::SparseMatrix aMarkovianToProbabilistic = this->getModel().getTransitionMatrix().getSubmatrix(markovianNonGoalStates, probabilisticNonGoalStates, nondeterministicChoiceIndices); + typename storm::storage::SparseMatrix aProbabilistic = this->getModel().getTransitionMatrix().getSubmatrix(probabilisticNonGoalStates, this->getModel().getNondeterministicChoiceIndices()); + typename storm::storage::SparseMatrix aProbabilisticToMarkovian = this->getModel().getTransitionMatrix().getSubmatrix(probabilisticNonGoalStates, markovianNonGoalStates, nondeterministicChoiceIndices); + + // The matrices with transitions from Markovian states need to be digitized. + + // Digitize aMarkovian. Based on whether the transition is a self-loop or not, we apply the two digitization rules. + uint_fast64_t rowIndex = 0; + for (auto state : markovianNonGoalStates) { + typename storm::storage::SparseMatrix::MutableRows row = aMarkovian.getRow(rowIndex); + for (auto element : row) { + ValueType eTerm = std::exp(-exitRates[state] * delta); + if (element.column() == rowIndex) { + element.value() = (storm::utility::constGetOne() - eTerm) * element.value() + eTerm; + } else { + element.value() = (storm::utility::constGetOne() - eTerm) * element.value(); + } + } + ++rowIndex; + } - // (3) Initialize three used vectors: + // Digitize aMarkovianToProbabilistic. As there are no self-loops in this case, we only need to apply the digitization formula for regular successors. + rowIndex = 0; + for (auto state : markovianNonGoalStates) { + typename storm::storage::SparseMatrix::MutableRows row = aMarkovianToProbabilistic.getRow(rowIndex); + for (auto element : row) { + element.value() = (1 - std::exp(-exitRates[state] * delta)) * element.value(); + } + ++rowIndex; + } + + // (3) Initialize two vectors: // * v_PS holds the probability values of probabilistic non-goal states. // * v_MS holds the probability values of Markovian non-goal states. - // * v_all holds the probability values of all non-goal states. This vector is needed for the Markov step. + std::vector vProbabilistic(probabilisticNonGoalStates.getNumberOfSetBits()); + std::vector vMarkovian(markovianNonGoalStates.getNumberOfSetBits()); + + // (4) Compute the two fixed right-hand side vectors, one for Markovian states and one for the probabilistic ones. + std::vector bProbabilistic = this->getModel().getTransitionMatrix().getConstrainedRowSumVector(probabilisticNonGoalStates, nondeterministicChoiceIndices, ~this->getModel().getMarkovianStates() & goalStates, aProbabilistic.getRowCount()); + storm::storage::BitVector probabilisticGoalStates = ~this->getModel().getMarkovianStates() & goalStates; + std::vector bMarkovian; + bMarkovian.reserve(markovianNonGoalStates.getNumberOfSetBits()); + for (auto state : markovianNonGoalStates) { + bMarkovian.push_back(storm::utility::constGetZero()); + + typename storm::storage::SparseMatrix::Rows row = transitionMatrix.getRow(rowIndex); + for (auto element : row) { + bMarkovian.back() += (1 - std::exp(-exitRates[state] * delta)) * element.value(); + } + } + + std::cout << delta << std::endl; + std::cout << markovianNonGoalStates.toString() << std::endl; + std::cout << aMarkovian.toString() << std::endl; + std::cout << aMarkovianToProbabilistic.toString() << std::endl; + std::cout << probabilisticNonGoalStates.toString() << std::endl; + std::cout << aProbabilistic.toString() << std::endl; + std::cout << aProbabilisticToMarkovian.toString() << std::endl; + std::cout << bProbabilistic << std::endl; + std::cout << bMarkovian << std::endl; // (3) Perform value iteration - // * initialize the vectors v_PS, v_MS and v_all. // * loop until the step bound has been reached // * in the loop: - // * perform value iteration using A_PSwG, v_PS and the vector x where x = x_PS + x_MS, x_PS = (A * 1_G)|PS with x_MS = A_PStoMS * v_MS + // * perform value iteration using A_PSwG, v_PS and the vector b where b = (A * 1_G)|PS + A_PStoMS * v_MS // and 1_G being the characteristic vector for all goal states. - // * copy the values from v_PS to the correct positions into v_all - // * perform one timed-step using A_MSwG, v_all and x_MS = (A * 1_G)|MS and obtain v_MS - // * copy the values from v_MS to the correct positions into v_all + // * perform one timed-step using v_MS := A_MSwG * v_MS + A_MStoPS * v_PS + (A * 1_G)|MS // // After the loop, perform one more step of the value iteration for PS states. - // Finally, create the result vector out of 1_G and v_all. + + + + + + // (4) Finally, create the result vector out of 1_G and v_all. // Return dummy vector for the time being. return std::vector(); diff --git a/src/models/MarkovAutomaton.h b/src/models/MarkovAutomaton.h index 67e2a8acf..2d53745f2 100644 --- a/src/models/MarkovAutomaton.h +++ b/src/models/MarkovAutomaton.h @@ -100,6 +100,14 @@ namespace storm { return this->exitRates[state]; } + T const& getMaximalExitRate() const { + T result = storm::utility::constGetZero(); + for (auto markovianState : this->markovianStates) { + result = std::max(result, this->exitRates[markovianState]); + } + return result; + } + storm::storage::BitVector const& getMarkovianStates() const { return this->markovianStates; } diff --git a/src/storage/SparseMatrix.h b/src/storage/SparseMatrix.h index b52a63269..db277459c 100644 --- a/src/storage/SparseMatrix.h +++ b/src/storage/SparseMatrix.h @@ -15,6 +15,7 @@ #include #include #include +#include #include #include #include @@ -172,6 +173,89 @@ public: uint_fast64_t const* columnPtr; }; + /*! + * A class representing an iterator over a continuous number of rows of the matrix. + */ + class Iterator { + public: + /*! + * Constructs an iterator from the given parameters. + * + * @param valuePtr A pointer to the value of the first element that is to be iterated over. + * @param columnPtr A pointer to the column of the first element that is to be iterated over. + */ + Iterator(T* valuePtr, uint_fast64_t* columnPtr) : valuePtr(valuePtr), columnPtr(columnPtr) { + // Intentionally left empty. + } + + /*! + * Moves the iterator to the next non-zero element. + * + * @return A reference to itself. + */ + Iterator& operator++() { + ++valuePtr; + ++columnPtr; + return *this; + } + + /*! + * Dereferences the iterator by returning a reference to itself. This is needed for making use of the range-based + * for loop over transitions. + * + * @return A reference to itself. + */ + Iterator& operator*() { + return *this; + } + + /*! + * Compares the two iterators for inequality. + * + * @return True iff the given iterator points to a different index as the current iterator. + */ + bool operator!=(Iterator const& other) const { + return this->valuePtr != other.valuePtr; + } + + /*! + * Assigns the position of the given iterator to the current iterator. + * + * @return A reference to itself. + */ + Iterator& operator=(Iterator const& other) { + this->valuePtr = other.valuePtr; + this->columnPtr = other.columnPtr; + return *this; + } + + /*! + * Retrieves the column that is associated with the current non-zero element to which this iterator + * points. + * + * @return The column of the current non-zero element to which this iterator points. + */ + uint_fast64_t column() { + return *columnPtr; + } + + /*! + * Retrieves the value of the current non-zero element to which this iterator points. + * + * @return The value of the current non-zero element to which this iterator points. + */ + T& value() { + return *valuePtr; + } + + private: + // A pointer to the value of the current non-zero element. + T* valuePtr; + + // A pointer to the column of the current non-zero element. + uint_fast64_t* columnPtr; + }; + /*! * This class represents a number of consecutive rows of the matrix. */ @@ -216,6 +300,51 @@ public: // The number of non-zero entries in the rows. uint_fast64_t entryCount; }; + + /*! + * This class represents a number of consecutive rows of the matrix. + */ + class MutableRows { + public: + /*! + * Constructs a row from the given parameters. + * + * @param valuePtr A pointer to the value of the first non-zero element of the rows. + * @param columnPtr A pointer to the column of the first non-zero element of the rows. + * @param entryCount The number of non-zero elements of the rows. + */ + MutableRows(T* valuePtr, uint_fast64_t* columnPtr, uint_fast64_t entryCount) : valuePtr(valuePtr), columnPtr(columnPtr), entryCount(entryCount) { + // Intentionally left empty. + } + + /*! + * Retrieves an iterator that points to the beginning of the rows. + * + * @return An iterator that points to the beginning of the rows. + */ + Iterator begin() { + return Iterator(valuePtr, columnPtr); + } + + /*! + * Retrieves an iterator that points past the last element of the rows. + * + * @return An iterator that points past the last element of the rows. + */ + Iterator end() { + return Iterator(valuePtr + entryCount, columnPtr + entryCount); + } + + private: + // The pointer to the value of the first element. + T* valuePtr; + + // The pointer to the column of the first element. + uint_fast64_t* columnPtr; + + // The number of non-zero entries in the rows. + uint_fast64_t entryCount; + }; /*! * This class represents an iterator to all rows of the matrix. @@ -919,29 +1048,53 @@ public: * @returns A matrix corresponding to a submatrix of the current matrix in which only row groups * and columns given by the row group constraint are kept and all others are dropped. */ - SparseMatrix getSubmatrix(storm::storage::BitVector const& rowGroupConstraint, std::vector const& rowGroupIndices) const { + SparseMatrix getSubmatrix(storm::storage::BitVector const& rowGroupConstraint, std::vector const& rowGroupIndices, bool insertDiagonalEntries = false) const { + return getSubmatrix(rowGroupConstraint, rowGroupConstraint, rowGroupIndices, insertDiagonalEntries); + } + + /*! + * Creates a submatrix of the current matrix by keeping only row groups and columns in the given + * row group and column constraint, respectively. + * + * @param rowGroupConstraint A bit vector indicating which row groups to keep. + * @param columnConstraint A bit vector indicating which columns to keep. + * @param rowGroupIndices A vector indicating which rows belong to a given row group. + * @returns A matrix corresponding to a submatrix of the current matrix in which only row groups + * and columns given by the row group constraint are kept and all others are dropped. + */ + SparseMatrix getSubmatrix(storm::storage::BitVector const& rowGroupConstraint, storm::storage::BitVector const& columnConstraint, std::vector const& rowGroupIndices, bool insertDiagonalEntries = false) const { LOG4CPLUS_DEBUG(logger, "Creating a sub-matrix (of unknown size)."); - + // First, we need to determine the number of non-zero entries and the number of rows of the sub-matrix. uint_fast64_t subNonZeroEntries = 0; uint_fast64_t subRowCount = 0; for (auto index : rowGroupConstraint) { subRowCount += rowGroupIndices[index + 1] - rowGroupIndices[index]; for (uint_fast64_t i = rowGroupIndices[index]; i < rowGroupIndices[index + 1]; ++i) { + bool foundDiagonalElement = false; + for (uint_fast64_t j = rowIndications[i]; j < rowIndications[i + 1]; ++j) { - if (rowGroupConstraint.get(columnIndications[j])) { + if (columnConstraint.get(columnIndications[j])) { ++subNonZeroEntries; + + if (index == columnIndications[j]) { + foundDiagonalElement = true; + } } } + + if (insertDiagonalEntries && !foundDiagonalElement) { + ++subNonZeroEntries; + } } } - + LOG4CPLUS_DEBUG(logger, "Determined size of submatrix to be " << subRowCount << "x" << rowGroupConstraint.getNumberOfSetBits() << "."); - + // Create and initialize resulting matrix. SparseMatrix result(subRowCount, rowGroupConstraint.getNumberOfSetBits()); result.initialize(subNonZeroEntries); - + // Create a temporary vector that stores for each index whose bit is set // to true the number of bits that were set before that particular index. std::vector bitsSetBeforeIndex; @@ -950,27 +1103,45 @@ public: // Compute the information to fill this vector. uint_fast64_t lastIndex = 0; uint_fast64_t currentNumberOfSetBits = 0; - for (auto index : rowGroupConstraint) { + + // If we are requested to add missing diagonal entries, we need to make sure the corresponding rows + storm::storage::BitVector columnBitCountConstraint = columnConstraint; + if (insertDiagonalEntries) { + columnBitCountConstraint |= rowGroupConstraint; + } + for (auto index : columnBitCountConstraint) { while (lastIndex <= index) { bitsSetBeforeIndex.push_back(currentNumberOfSetBits); ++lastIndex; } ++currentNumberOfSetBits; } - + // Copy over selected entries. uint_fast64_t rowCount = 0; for (auto index : rowGroupConstraint) { for (uint_fast64_t i = rowGroupIndices[index]; i < rowGroupIndices[index + 1]; ++i) { + bool insertedDiagonalElement = false; + for (uint_fast64_t j = rowIndications[i]; j < rowIndications[i + 1]; ++j) { - if (rowGroupConstraint.get(columnIndications[j])) { + if (columnConstraint.get(columnIndications[j])) { + if (index == columnIndications[j]) { + insertedDiagonalElement = true; + } else if (insertDiagonalEntries && !insertedDiagonalElement && columnIndications[j] > index) { + result.addNextValue(rowCount, bitsSetBeforeIndex[index], storm::utility::constGetZero()); + insertedDiagonalElement = true; + } result.addNextValue(rowCount, bitsSetBeforeIndex[columnIndications[j]], valueStorage[j]); } } + if (insertDiagonalEntries && !insertedDiagonalElement) { + result.addNextValue(rowCount, bitsSetBeforeIndex[index], storm::utility::constGetZero()); + } + ++rowCount; } } - + // Finalize sub-matrix and return result. result.finalize(); LOG4CPLUS_DEBUG(logger, "Done creating sub-matrix."); @@ -980,8 +1151,7 @@ public: SparseMatrix getSubmatrix(std::vector const& rowGroupToRowIndexMapping, std::vector const& rowGroupIndices, bool insertDiagonalEntries = true) const { LOG4CPLUS_DEBUG(logger, "Creating a sub-matrix (of unknown size)."); - // First, we need to count how many non-zero entries the resulting matrix will have and reserve space for diagonal - // entries. + // First, we need to count how many non-zero entries the resulting matrix will have and reserve space for diagonal entries. uint_fast64_t subNonZeroEntries = 0; for (uint_fast64_t rowGroupIndex = 0, rowGroupIndexEnd = rowGroupToRowIndexMapping.size(); rowGroupIndex < rowGroupIndexEnd; ++rowGroupIndex) { // Determine which row we need to select from the current row group. @@ -1227,6 +1397,27 @@ public: return getRows(row, row); } + /*! + * Returns an object representing the consecutive rows given by the parameters. + * + * @param startRow The starting row. + * @param endRow The ending row (which is included in the result). + * @return An object representing the consecutive rows given by the parameters. + */ + MutableRows getRows(uint_fast64_t startRow, uint_fast64_t endRow) { + return MutableRows(this->valueStorage.data() + this->rowIndications[startRow], this->columnIndications.data() + this->rowIndications[startRow], this->rowIndications[endRow + 1] - this->rowIndications[startRow]); + } + + /*! + * Returns an object representing the given row. + * + * @param row The chosen row. + * @return An object representing the given row. + */ + MutableRows getRow(uint_fast64_t row) { + return getRows(row, row); + } + /*! * Returns a const iterator to the rows of the matrix. * @@ -1435,7 +1626,7 @@ public: uint_fast64_t currentRealIndex = 0; while (currentRealIndex < colCount) { if (nextIndex < rowIndications[i + 1] && currentRealIndex == columnIndications[nextIndex]) { - result << valueStorage[nextIndex] << "\t"; + result << std::setprecision(8) << valueStorage[nextIndex] << "\t"; ++nextIndex; } else { result << "0\t"; diff --git a/src/storm.cpp b/src/storm.cpp index 8668f01f5..97e5c1b6b 100644 --- a/src/storm.cpp +++ b/src/storm.cpp @@ -478,6 +478,8 @@ int main(const int argc, const char* argv[]) { std::cout << mc.checkExpectedTime(false, markovAutomaton->getLabeledStates("goal")) << std::endl; std::cout << mc.checkLongRunAverage(true, markovAutomaton->getLabeledStates("goal")) << std::endl; std::cout << mc.checkLongRunAverage(false, markovAutomaton->getLabeledStates("goal")) << std::endl; + std::cout << mc.checkTimeBoundedEventually(true, markovAutomaton->getLabeledStates("goal"), 0, 1) << std::endl; + std::cout << mc.checkTimeBoundedEventually(false, markovAutomaton->getLabeledStates("goal"), 0, 1) << std::endl; break; } case storm::models::Unknown: From 18711c01a333047551ba8765bc937c2ac6e7a734 Mon Sep 17 00:00:00 2001 From: dehnert Date: Sun, 1 Dec 2013 20:08:48 +0100 Subject: [PATCH 2/9] First working version of time-bounded reachability for Markov automata. Former-commit-id: 6501cbfca48b689beb7893cd7f667ada923ef49d --- .../SparseMarkovAutomatonCslModelChecker.h | 203 +++++++++++------- ...ractNondeterministicLinearEquationSolver.h | 77 ++++--- ...mmxxNondeterministicLinearEquationSolver.h | 39 ++-- src/storage/SparseMatrix.cpp | 2 +- src/storm.cpp | 12 +- src/utility/solver.h | 8 + 6 files changed, 207 insertions(+), 134 deletions(-) diff --git a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h index f11f765f3..75b3e81cb 100644 --- a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h +++ b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h @@ -22,7 +22,7 @@ namespace storm { class SparseMarkovAutomatonCslModelChecker : public AbstractModelChecker { public: - explicit SparseMarkovAutomatonCslModelChecker(storm::models::MarkovAutomaton const& model, storm::solver::AbstractNondeterministicLinearEquationSolver* linearEquationSolver) : AbstractModelChecker(model), minimumOperatorStack(), linearEquationSolver(linearEquationSolver) { + explicit SparseMarkovAutomatonCslModelChecker(storm::models::MarkovAutomaton const& model) : AbstractModelChecker(model), minimumOperatorStack() { // Intentionally left empty. } @@ -70,36 +70,20 @@ namespace storm { return result; } - std::vector checkTimeBoundedEventually(bool min, storm::storage::BitVector const& goalStates, uint_fast64_t lowerBound, uint_fast64_t upperBound) const { - if (!this->getModel().isClosed()) { - throw storm::exceptions::InvalidArgumentException() << "Unable to compute time-bounded reachability on non-closed Markov automaton."; - } - - // (1) Compute the number of steps we need to take. - ValueType lambda = this->getModel().getMaximalExitRate(); - ValueType delta = (2 * storm::settings::Settings::getInstance()->getOptionByLongName("precision").getArgument(0).getValueAsDouble()) / (upperBound * lambda * lambda); - LOG4CPLUS_INFO(logger, "Determined delta to be " << delta << "."); - - // Get some data fields for convenient access. - std::vector const& nondeterministicChoiceIndices = this->getModel().getNondeterministicChoiceIndices(); - std::vector const& exitRates = this->getModel().getExitRates(); - typename storm::storage::SparseMatrix const& transitionMatrix = this->getModel().getTransitionMatrix(); - - // (2) Compute four sparse matrices: - // * a matrix A_MS with all (discretized) transitions from Markovian non-goal states to all Markovian non-goal states. - // * a matrix A_MStoPS with all (discretized) transitions from Markovian non-goal states to all probabilistic non-goal states. - // * a matrix A_PS with all (non-discretized) transitions from probabilistic non-goal states to other probabilistic non-goal states. This matrix has more rows than columns. - // * a matrix A_PStoMS with all (non-discretized) transitions from probabilistic non-goal states to all Markovian non-goal states. This matrix may have any shape. - storm::storage::BitVector const& markovianNonGoalStates = this->getModel().getMarkovianStates() & ~goalStates; - storm::storage::BitVector const& probabilisticNonGoalStates = ~this->getModel().getMarkovianStates() & ~goalStates; - - typename storm::storage::SparseMatrix aMarkovian = this->getModel().getTransitionMatrix().getSubmatrix(markovianNonGoalStates, this->getModel().getNondeterministicChoiceIndices(), true); - typename storm::storage::SparseMatrix aMarkovianToProbabilistic = this->getModel().getTransitionMatrix().getSubmatrix(markovianNonGoalStates, probabilisticNonGoalStates, nondeterministicChoiceIndices); - typename storm::storage::SparseMatrix aProbabilistic = this->getModel().getTransitionMatrix().getSubmatrix(probabilisticNonGoalStates, this->getModel().getNondeterministicChoiceIndices()); - typename storm::storage::SparseMatrix aProbabilisticToMarkovian = this->getModel().getTransitionMatrix().getSubmatrix(probabilisticNonGoalStates, markovianNonGoalStates, nondeterministicChoiceIndices); + static void computeBoundedReachability(bool min, storm::storage::SparseMatrix const& transitionMatrix, std::vector const& nondeterministicChoiceIndices, std::vector const& exitRates, storm::storage::BitVector const& markovianStates, storm::storage::BitVector const& goalStates, storm::storage::BitVector const& markovianNonGoalStates, storm::storage::BitVector const& probabilisticNonGoalStates, std::vector& markovianNonGoalValues, std::vector& probabilisticNonGoalValues, ValueType delta, uint_fast64_t numberOfSteps) { + // Start by computing four sparse matrices: + // * a matrix aMarkovian with all (discretized) transitions from Markovian non-goal states to all Markovian non-goal states. + // * a matrix aMarkovianToProbabilistic with all (discretized) transitions from Markovian non-goal states to all probabilistic non-goal states. + // * a matrix aProbabilistic with all (non-discretized) transitions from probabilistic non-goal states to other probabilistic non-goal states. + // * a matrix aProbabilisticToMarkovian with all (non-discretized) transitions from probabilistic non-goal states to all Markovian non-goal states. + typename storm::storage::SparseMatrix aMarkovian = transitionMatrix.getSubmatrix(markovianNonGoalStates, nondeterministicChoiceIndices, true); + typename storm::storage::SparseMatrix aMarkovianToProbabilistic = transitionMatrix.getSubmatrix(markovianNonGoalStates, probabilisticNonGoalStates, nondeterministicChoiceIndices); + std::vector markovianNondeterministicChoiceIndices = computeNondeterministicChoiceIndicesForConstraint(nondeterministicChoiceIndices, markovianNonGoalStates); + typename storm::storage::SparseMatrix aProbabilistic = transitionMatrix.getSubmatrix(probabilisticNonGoalStates, nondeterministicChoiceIndices); + typename storm::storage::SparseMatrix aProbabilisticToMarkovian = transitionMatrix.getSubmatrix(probabilisticNonGoalStates, markovianNonGoalStates, nondeterministicChoiceIndices); + std::vector probabilisticNondeterministicChoiceIndices = computeNondeterministicChoiceIndicesForConstraint(nondeterministicChoiceIndices, probabilisticNonGoalStates); // The matrices with transitions from Markovian states need to be digitized. - // Digitize aMarkovian. Based on whether the transition is a self-loop or not, we apply the two digitization rules. uint_fast64_t rowIndex = 0; for (auto state : markovianNonGoalStates) { @@ -125,53 +109,128 @@ namespace storm { ++rowIndex; } - // (3) Initialize two vectors: - // * v_PS holds the probability values of probabilistic non-goal states. - // * v_MS holds the probability values of Markovian non-goal states. - std::vector vProbabilistic(probabilisticNonGoalStates.getNumberOfSetBits()); - std::vector vMarkovian(markovianNonGoalStates.getNumberOfSetBits()); + // Initialize the two vectors that hold the variable one-step probabilities to all target states for probabilistic and Markovian (non-goal) states. + std::vector bProbabilistic(aProbabilistic.getRowCount()); + std::vector bMarkovian(markovianNonGoalStates.getNumberOfSetBits()); - // (4) Compute the two fixed right-hand side vectors, one for Markovian states and one for the probabilistic ones. - std::vector bProbabilistic = this->getModel().getTransitionMatrix().getConstrainedRowSumVector(probabilisticNonGoalStates, nondeterministicChoiceIndices, ~this->getModel().getMarkovianStates() & goalStates, aProbabilistic.getRowCount()); - storm::storage::BitVector probabilisticGoalStates = ~this->getModel().getMarkovianStates() & goalStates; - std::vector bMarkovian; - bMarkovian.reserve(markovianNonGoalStates.getNumberOfSetBits()); + // Compute the two fixed right-hand side vectors, one for Markovian states and one for the probabilistic ones. + std::vector bProbabilisticFixed = transitionMatrix.getConstrainedRowSumVector(probabilisticNonGoalStates, nondeterministicChoiceIndices, goalStates, aProbabilistic.getRowCount()); + std::vector bMarkovianFixed; + bMarkovianFixed.reserve(markovianNonGoalStates.getNumberOfSetBits()); for (auto state : markovianNonGoalStates) { - bMarkovian.push_back(storm::utility::constGetZero()); + bMarkovianFixed.push_back(storm::utility::constGetZero()); - typename storm::storage::SparseMatrix::Rows row = transitionMatrix.getRow(rowIndex); + typename storm::storage::SparseMatrix::Rows row = transitionMatrix.getRow(nondeterministicChoiceIndices[state]); for (auto element : row) { - bMarkovian.back() += (1 - std::exp(-exitRates[state] * delta)) * element.value(); + if (goalStates.get(element.column())) { + bMarkovianFixed.back() += (1 - std::exp(-exitRates[state] * delta)) * element.value(); + } } } - std::cout << delta << std::endl; - std::cout << markovianNonGoalStates.toString() << std::endl; - std::cout << aMarkovian.toString() << std::endl; - std::cout << aMarkovianToProbabilistic.toString() << std::endl; - std::cout << probabilisticNonGoalStates.toString() << std::endl; - std::cout << aProbabilistic.toString() << std::endl; - std::cout << aProbabilisticToMarkovian.toString() << std::endl; - std::cout << bProbabilistic << std::endl; - std::cout << bMarkovian << std::endl; + std::unique_ptr> nondeterministiclinearEquationSolver = storm::utility::solver::getNondeterministicLinearEquationSolver(); - // (3) Perform value iteration + // Perform the actual value iteration // * loop until the step bound has been reached // * in the loop: // * perform value iteration using A_PSwG, v_PS and the vector b where b = (A * 1_G)|PS + A_PStoMS * v_MS // and 1_G being the characteristic vector for all goal states. // * perform one timed-step using v_MS := A_MSwG * v_MS + A_MStoPS * v_PS + (A * 1_G)|MS - // - // After the loop, perform one more step of the value iteration for PS states. - - - + std::vector markovianNonGoalValuesSwap(markovianNonGoalValues); + std::vector multiplicationResultScratchMemory(aProbabilistic.getRowCount()); + std::vector aProbabilisticScratchMemory(probabilisticNonGoalValues.size()); + for (uint_fast64_t currentStep = 0; currentStep < numberOfSteps; ++currentStep) { + // Start by (re-)computing bProbabilistic = bProbabilisticFixed + aProbabilisticToMarkovian * vMarkovian. + aProbabilisticToMarkovian.multiplyWithVector(markovianNonGoalValues, bProbabilistic); + storm::utility::vector::addVectorsInPlace(bProbabilistic, bProbabilisticFixed); + + // Now perform the inner value iteration for probabilistic states. + nondeterministiclinearEquationSolver->solveEquationSystem(min, aProbabilistic, probabilisticNonGoalValues, bProbabilistic, probabilisticNondeterministicChoiceIndices, &multiplicationResultScratchMemory, &aProbabilisticScratchMemory); + + // (Re-)compute bMarkovian = bMarkovianFixed + aMarkovianToProbabilistic * vProbabilistic. + aMarkovianToProbabilistic.multiplyWithVector(probabilisticNonGoalValues, bMarkovian); + storm::utility::vector::addVectorsInPlace(bMarkovian, bMarkovianFixed); + aMarkovian.multiplyWithVector(markovianNonGoalValues, markovianNonGoalValuesSwap); + std::swap(markovianNonGoalValues, markovianNonGoalValuesSwap); + storm::utility::vector::addVectorsInPlace(markovianNonGoalValues, bMarkovian); + } + // After the loop, perform one more step of the value iteration for PS states. + aProbabilisticToMarkovian.multiplyWithVector(probabilisticNonGoalValues, bProbabilistic); + storm::utility::vector::addVectorsInPlace(bProbabilistic, bProbabilisticFixed); + nondeterministiclinearEquationSolver->solveEquationSystem(min, aProbabilistic, probabilisticNonGoalValues, bProbabilistic, probabilisticNondeterministicChoiceIndices); + } + + std::vector checkTimeBoundedEventually(bool min, storm::storage::BitVector const& goalStates, ValueType lowerBound, ValueType upperBound) const { + // Check whether the automaton is closed. + if (!this->getModel().isClosed()) { + throw storm::exceptions::InvalidArgumentException() << "Unable to compute time-bounded reachability on non-closed Markov automaton."; + } - // (4) Finally, create the result vector out of 1_G and v_all. + // Check whether the given bounds were valid. + if (lowerBound < storm::utility::constGetZero() || upperBound < storm::utility::constGetZero() || upperBound < lowerBound) { + throw storm::exceptions::InvalidArgumentException() << "Illegal interval ["; + } + + // Get some data fields for convenient access. + std::vector const& nondeterministicChoiceIndices = this->getModel().getNondeterministicChoiceIndices(); + typename storm::storage::SparseMatrix const& transitionMatrix = this->getModel().getTransitionMatrix(); + std::vector const& exitRates = this->getModel().getExitRates(); + storm::storage::BitVector const& markovianStates = this->getModel().getMarkovianStates(); + + // (1) Compute the accuracy we need to achieve the required error bound. + ValueType maxExitRate = this->getModel().getMaximalExitRate(); + ValueType delta = (2 * storm::settings::Settings::getInstance()->getOptionByLongName("precision").getArgument(0).getValueAsDouble()) / (upperBound * maxExitRate * maxExitRate); + + // (2) Compute the number of steps we need to make for the interval. + uint_fast64_t numberOfSteps = static_cast(std::ceil((upperBound - lowerBound) / delta)); + std::cout << "Performing " << numberOfSteps << " iterations (delta=" << delta << ") for interval [" << lowerBound << ", " << upperBound << "]." << std::endl; + + // (3) Compute the non-goal states and initialize two vectors + // * vProbabilistic holds the probability values of probabilistic non-goal states. + // * vMarkovian holds the probability values of Markovian non-goal states. + storm::storage::BitVector const& markovianNonGoalStates = markovianStates & ~goalStates; + storm::storage::BitVector const& probabilisticNonGoalStates = ~markovianStates & ~goalStates; + std::vector vProbabilistic(probabilisticNonGoalStates.getNumberOfSetBits()); + std::vector vMarkovian(markovianNonGoalStates.getNumberOfSetBits()); - // Return dummy vector for the time being. - return std::vector(); + computeBoundedReachability(min, transitionMatrix, nondeterministicChoiceIndices, exitRates, markovianStates, goalStates, markovianNonGoalStates, probabilisticNonGoalStates, vMarkovian, vProbabilistic, delta, numberOfSteps); + + // (4) If the lower bound of interval was non-zero, we need to take the current values as the starting values for a subsequent value iteration. + if (lowerBound != storm::utility::constGetZero()) { + std::vector vAllProbabilistic((~markovianStates).getNumberOfSetBits()); + std::vector vAllMarkovian(markovianStates.getNumberOfSetBits()); + + // Create the starting value vectors for the next value iteration based on the results of the previous one. + storm::utility::vector::setVectorValues(vAllProbabilistic, ~markovianStates % goalStates, storm::utility::constGetOne()); + storm::utility::vector::setVectorValues(vAllProbabilistic, ~markovianStates % ~goalStates, vProbabilistic); + std::cout << vAllProbabilistic << std::endl; + storm::utility::vector::setVectorValues(vAllMarkovian, markovianStates % goalStates, storm::utility::constGetOne()); + storm::utility::vector::setVectorValues(vAllMarkovian, markovianStates % ~goalStates, vMarkovian); + std::cout << vAllMarkovian << std::endl; + + // Compute the number of steps to reach the target interval. + numberOfSteps = static_cast(std::ceil(lowerBound / delta)); + std::cout << "Performing " << numberOfSteps << " iterations (delta=" << delta << ") for interval [0, " << lowerBound << "]." << std::endl; + // FIXME: According to IMCA, the value of delta needs to be recomputed here, but using the equations from the source this doesn't make sense, because they always evaluate to the original value of delta. + + // Compute the bounded reachability for interval [0, b-a]. + computeBoundedReachability(min, transitionMatrix, nondeterministicChoiceIndices, exitRates, markovianStates, storm::storage::BitVector(this->getModel().getNumberOfStates()), markovianStates, ~markovianStates, vAllMarkovian, vAllProbabilistic, delta, numberOfSteps); + + // Create the result vector out of vAllProbabilistic and vAllMarkovian and return it. + std::vector result(this->getModel().getNumberOfStates()); + storm::utility::vector::setVectorValues(result, ~markovianStates, vAllProbabilistic); + storm::utility::vector::setVectorValues(result, markovianStates, vAllMarkovian); + + return result; + } else { + // Create the result vector out of 1_G, vProbabilistic and vMarkovian and return it. + std::vector result(this->getModel().getNumberOfStates()); + storm::utility::vector::setVectorValues(result, goalStates, storm::utility::constGetOne()); + storm::utility::vector::setVectorValues(result, probabilisticNonGoalStates, vProbabilistic); + storm::utility::vector::setVectorValues(result, markovianNonGoalStates, vMarkovian); + return result; + } } std::vector checkLongRunAverage(bool min, storm::storage::BitVector const& goalStates) const { @@ -370,11 +429,8 @@ namespace storm { // Finalize the matrix and solve the corresponding system of equations. sspMatrix.finalize(); std::vector x(numberOfStatesNotInMecs + mecDecomposition.size()); - if (linearEquationSolver != nullptr) { - this->linearEquationSolver->solveEquationSystem(min, sspMatrix, x, b, sspNondeterministicChoiceIndices); - } else { - throw storm::exceptions::InvalidStateException() << "No valid linear equation solver available."; - } + std::unique_ptr> nondeterministiclinearEquationSolver = storm::utility::solver::getNondeterministicLinearEquationSolver(); + nondeterministiclinearEquationSolver->solveEquationSystem(min, sspMatrix, x, b, sspNondeterministicChoiceIndices); // Prepare result vector. std::vector result(this->getModel().getNumberOfStates()); @@ -451,7 +507,7 @@ namespace storm { // Then, we can eliminate the rows and columns for all states whose values are already known to be 0. std::vector x(maybeStates.getNumberOfSetBits()); - std::vector subNondeterministicChoiceIndices = this->computeNondeterministicChoiceIndicesForConstraint(maybeStates); + std::vector subNondeterministicChoiceIndices = computeNondeterministicChoiceIndicesForConstraint(this->getModel().getNondeterministicChoiceIndices(), maybeStates); storm::storage::SparseMatrix 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. @@ -465,11 +521,8 @@ namespace storm { 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."; - } + std::unique_ptr> nondeterministiclinearEquationSolver = storm::utility::solver::getNondeterministicLinearEquationSolver(); + nondeterministiclinearEquationSolver->solveEquationSystem(min, submatrix, x, b, subNondeterministicChoiceIndices); // Create resulting vector. std::vector result(this->getModel().getNumberOfStates()); @@ -497,10 +550,7 @@ namespace storm { * @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 computeNondeterministicChoiceIndicesForConstraint(storm::storage::BitVector const& constraint) const { - // First, get a reference to the full nondeterministic choice indices. - std::vector const& nondeterministicChoiceIndices = this->getModel().getNondeterministicChoiceIndices(); - + static std::vector computeNondeterministicChoiceIndicesForConstraint(std::vector const& nondeterministicChoiceIndices, storm::storage::BitVector const& constraint) { // Reserve the known amount of slots for the resulting vector. std::vector subNondeterministicChoiceIndices(constraint.getNumberOfSetBits() + 1); uint_fast64_t currentRowCount = 0; @@ -522,9 +572,6 @@ namespace storm { return subNondeterministicChoiceIndices; } - - // An object that is used for solving linear equations and performing matrix-vector multiplication. - std::unique_ptr> linearEquationSolver; }; } } diff --git a/src/solver/AbstractNondeterministicLinearEquationSolver.h b/src/solver/AbstractNondeterministicLinearEquationSolver.h index 4a63828cb..f0036a397 100644 --- a/src/solver/AbstractNondeterministicLinearEquationSolver.h +++ b/src/solver/AbstractNondeterministicLinearEquationSolver.h @@ -13,9 +13,19 @@ namespace storm { template class AbstractNondeterministicLinearEquationSolver { public: + AbstractNondeterministicLinearEquationSolver() { + storm::settings::Settings* s = storm::settings::Settings::getInstance(); + precision = s->getOptionByLongName("precision").getArgument(0).getValueAsDouble(); + maxIterations = s->getOptionByLongName("maxIterations").getArgument(0).getValueAsUnsignedInteger(); + relative = s->getOptionByLongName("relative").getArgument(0).getValueAsBoolean(); + } + + AbstractNondeterministicLinearEquationSolver(double precision, uint_fast64_t maxIterations, bool relative) : precision(precision), maxIterations(maxIterations), relative(relative) { + // Intentionally left empty. + } virtual AbstractNondeterministicLinearEquationSolver* clone() const { - return new AbstractNondeterministicLinearEquationSolver(); + return new AbstractNondeterministicLinearEquationSolver(this->precision, this->maxIterations, this->relative); } /*! @@ -69,38 +79,39 @@ namespace storm { * as there are states in the MDP. * @returns The solution vector x of the system of linear equations as the content of the parameter x. */ - virtual void solveEquationSystem(bool minimize, storm::storage::SparseMatrix const& A, std::vector& x, std::vector const& b, std::vector const& nondeterministicChoiceIndices) const { - LOG4CPLUS_INFO(logger, "Starting iterative solver."); - - // Get the settings object to customize solving. - storm::settings::Settings* s = storm::settings::Settings::getInstance(); - - // Get relevant user-defined settings for solving the equations. - double precision = s->getOptionByLongName("precision").getArgument(0).getValueAsDouble(); - uint_fast64_t maxIterations = s->getOptionByLongName("maxIterations").getArgument(0).getValueAsUnsignedInteger(); - bool relative = s->getOptionByLongName("relative").getArgument(0).getValueAsBoolean(); - + virtual void solveEquationSystem(bool minimize, storm::storage::SparseMatrix const& A, std::vector& x, std::vector const& b, std::vector const& nondeterministicChoiceIndices, std::vector* multiplyResult = nullptr, std::vector* newX = nullptr) const { +// LOG4CPLUS_INFO(logger, "Starting iterative solver."); + // Set up the environment for the power method. - std::vector multiplyResult(A.getRowCount()); +// auto startTime = std::chrono::high_resolution_clock::now(); +// std::chrono::nanoseconds totalTime(0); + + if (multiplyResult == nullptr) { + multiplyResult = new std::vector(A.getRowCount()); + } std::vector* currentX = &x; - std::vector* newX = new std::vector(x.size()); + bool xMemoryProvided = true; + if (newX == nullptr) { + newX = new std::vector(x.size()); + xMemoryProvided = false; + } std::vector* swap = nullptr; uint_fast64_t iterations = 0; bool converged = false; - + // Proceed with the iterations as long as the method did not converge or reach the // user-specified maximum number of iterations. while (!converged && iterations < maxIterations) { // Compute x' = A*x + b. - A.multiplyWithVector(*currentX, multiplyResult); - storm::utility::vector::addVectorsInPlace(multiplyResult, b); + A.multiplyWithVector(*currentX, *multiplyResult); + storm::utility::vector::addVectorsInPlace(*multiplyResult, b); // Reduce the vector x' by applying min/max for all non-deterministic choices as given by the topmost // element of the min/max operator stack. if (minimize) { - storm::utility::vector::reduceVectorMin(multiplyResult, *newX, nondeterministicChoiceIndices); + storm::utility::vector::reduceVectorMin(*multiplyResult, *newX, nondeterministicChoiceIndices); } else { - storm::utility::vector::reduceVectorMax(multiplyResult, *newX, nondeterministicChoiceIndices); + storm::utility::vector::reduceVectorMax(*multiplyResult, *newX, nondeterministicChoiceIndices); } // Determine whether the method converged. @@ -112,24 +123,32 @@ namespace storm { newX = swap; ++iterations; } - + // If we performed an odd number of iterations, we need to swap the x and currentX, because the newest result // is currently stored in currentX, but x is the output vector. if (iterations % 2 == 1) { std::swap(x, *currentX); - delete currentX; - } else { + if (!xMemoryProvided) { + delete currentX; + } + } else if (!xMemoryProvided) { delete newX; } - - // Check if the solver converged and issue a warning otherwise. - if (converged) { - LOG4CPLUS_INFO(logger, "Iterative solver converged after " << iterations << " iterations."); - } else { - LOG4CPLUS_WARN(logger, "Iterative solver did not converge."); - } + +// // Check if the solver converged and issue a warning otherwise. +// if (converged) { +// LOG4CPLUS_INFO(logger, "Iterative solver converged after " << iterations << " iterations."); +// } else { +// LOG4CPLUS_WARN(logger, "Iterative solver did not converge."); +// } + } + protected: + double precision; + uint_fast64_t maxIterations; + bool relative; + }; } // namespace solver diff --git a/src/solver/GmmxxNondeterministicLinearEquationSolver.h b/src/solver/GmmxxNondeterministicLinearEquationSolver.h index 953e826b0..bef5c838f 100644 --- a/src/solver/GmmxxNondeterministicLinearEquationSolver.h +++ b/src/solver/GmmxxNondeterministicLinearEquationSolver.h @@ -47,42 +47,41 @@ namespace storm { delete gmmxxMatrix; } - virtual void solveEquationSystem(bool minimize, storm::storage::SparseMatrix const& A, std::vector& x, std::vector const& b, std::vector const& nondeterministicChoiceIndices) const override { - // Get the settings object to customize solving. - storm::settings::Settings* s = storm::settings::Settings::getInstance(); - - // Get relevant user-defined settings for solving the equations. - double precision = s->getOptionByLongName("precision").getArgument(0).getValueAsDouble(); - uint_fast64_t maxIterations = s->getOptionByLongName("maxIterations").getArgument(0).getValueAsUnsignedInteger(); - bool relative = s->getOptionByLongName("relative").getArgument(0).getValueAsBoolean(); - + virtual void solveEquationSystem(bool minimize, storm::storage::SparseMatrix const& A, std::vector& x, std::vector const& b, std::vector const& nondeterministicChoiceIndices, std::vector* multiplyResult = nullptr, std::vector* newX = nullptr) const override { // Transform the transition probability matrix to the gmm++ format to use its arithmetic. gmm::csr_matrix* gmmxxMatrix = storm::adapters::GmmxxAdapter::toGmmxxSparseMatrix(A); // Set up the environment for the power method. - std::vector multiplyResult(A.getRowCount()); + if (multiplyResult == nullptr) { + multiplyResult = new std::vector(A.getRowCount()); + } + std::vector* currentX = &x; - std::vector* newX = new std::vector(x.size()); + bool xMemoryProvided = true; + if (newX == nullptr) { + newX = new std::vector(x.size()); + xMemoryProvided = false; + } std::vector* swap = nullptr; uint_fast64_t iterations = 0; bool converged = false; // Proceed with the iterations as long as the method did not converge or reach the user-specified maximum number // of iterations. - while (!converged && iterations < maxIterations) { + while (!converged && iterations < this->maxIterations) { // Compute x' = A*x + b. - gmm::mult(*gmmxxMatrix, *currentX, multiplyResult); - gmm::add(b, multiplyResult); + gmm::mult(*gmmxxMatrix, *currentX, *multiplyResult); + gmm::add(b, *multiplyResult); // Reduce the vector x' by applying min/max for all non-deterministic choices. if (minimize) { - storm::utility::vector::reduceVectorMin(multiplyResult, *newX, nondeterministicChoiceIndices); + storm::utility::vector::reduceVectorMin(*multiplyResult, *newX, nondeterministicChoiceIndices); } else { - storm::utility::vector::reduceVectorMax(multiplyResult, *newX, nondeterministicChoiceIndices); + storm::utility::vector::reduceVectorMax(*multiplyResult, *newX, nondeterministicChoiceIndices); } // Determine whether the method converged. - converged = storm::utility::vector::equalModuloPrecision(*currentX, *newX, precision, relative); + converged = storm::utility::vector::equalModuloPrecision(*currentX, *newX, this->precision, this->relative); // Update environment variables. swap = currentX; @@ -95,8 +94,10 @@ namespace storm { // is currently stored in currentX, but x is the output vector. if (iterations % 2 == 1) { std::swap(x, *currentX); - delete currentX; - } else { + if (!xMemoryProvided) { + delete currentX; + } + } else if (!xMemoryProvided) { delete newX; } delete gmmxxMatrix; diff --git a/src/storage/SparseMatrix.cpp b/src/storage/SparseMatrix.cpp index 8c4e31ce5..0a1813ccb 100644 --- a/src/storage/SparseMatrix.cpp +++ b/src/storage/SparseMatrix.cpp @@ -577,7 +577,7 @@ namespace storage { LOG4CPLUS_DEBUG(logger, "Determined size of submatrix to be " << subRowCount << "x" << rowGroupConstraint.getNumberOfSetBits() << "."); // Create and initialize resulting matrix. - SparseMatrix result(subRowCount, rowGroupConstraint.getNumberOfSetBits()); + SparseMatrix result(subRowCount, columnConstraint.getNumberOfSetBits()); result.initialize(subNonZeroEntries); // Create a temporary vector that stores for each index whose bit is set diff --git a/src/storm.cpp b/src/storm.cpp index 97e5c1b6b..820b2be7f 100644 --- a/src/storm.cpp +++ b/src/storm.cpp @@ -473,13 +473,11 @@ int main(const int argc, const char* argv[]) { LOG4CPLUS_INFO(logger, "Model is a Markov automaton."); std::shared_ptr> markovAutomaton = parser.getModel>(); markovAutomaton->close(); - storm::modelchecker::csl::SparseMarkovAutomatonCslModelChecker mc(*markovAutomaton, new storm::solver::AbstractNondeterministicLinearEquationSolver()); - std::cout << mc.checkExpectedTime(true, markovAutomaton->getLabeledStates("goal")) << std::endl; - std::cout << mc.checkExpectedTime(false, markovAutomaton->getLabeledStates("goal")) << std::endl; - std::cout << mc.checkLongRunAverage(true, markovAutomaton->getLabeledStates("goal")) << std::endl; - std::cout << mc.checkLongRunAverage(false, markovAutomaton->getLabeledStates("goal")) << std::endl; - std::cout << mc.checkTimeBoundedEventually(true, markovAutomaton->getLabeledStates("goal"), 0, 1) << std::endl; - std::cout << mc.checkTimeBoundedEventually(false, markovAutomaton->getLabeledStates("goal"), 0, 1) << std::endl; + storm::modelchecker::csl::SparseMarkovAutomatonCslModelChecker mc(*markovAutomaton); +// std::cout << mc.checkExpectedTime(true, markovAutomaton->getLabeledStates("goal")) << std::endl; +// std::cout << mc.checkLongRunAverage(true, markovAutomaton->getLabeledStates("goal")) << std::endl; +// std::cout << mc.checkTimeBoundedEventually(true, markovAutomaton->getLabeledStates("goal"), 0, 1) << std::endl; + std::cout << mc.checkTimeBoundedEventually(true, markovAutomaton->getLabeledStates("goal"), 1, 2) << std::endl; break; } case storm::models::Unknown: diff --git a/src/utility/solver.h b/src/utility/solver.h index d0047d212..fdf838174 100644 --- a/src/utility/solver.h +++ b/src/utility/solver.h @@ -1,6 +1,8 @@ #ifndef STORM_UTILITY_SOLVER_H_ #define STORM_UTILITY_SOLVER_H_ +#include "src/solver/AbstractNondeterministicLinearEquationSolver.h" +#include "src/solver/GmmxxNondeterministicLinearEquationSolver.h" #include "src/solver/GurobiLpSolver.h" namespace storm { @@ -10,6 +12,12 @@ namespace storm { std::unique_ptr getLpSolver(std::string const& name) { return std::unique_ptr(new storm::solver::GurobiLpSolver(name)); } + + template + std::unique_ptr> getNondeterministicLinearEquationSolver() { + return std::unique_ptr>(new storm::solver::AbstractNondeterministicLinearEquationSolver()); +// return std::unique_ptr>(new storm::solver::GmmxxNondeterministicLinearEquationSolver()); + } } } } From 0a89d65f93b48d8c1f3a202b5e0d517880e488f7 Mon Sep 17 00:00:00 2001 From: dehnert Date: Mon, 2 Dec 2013 14:05:49 +0100 Subject: [PATCH 3/9] Started refactoring Markov automaton model checker. Former-commit-id: c4278de4f0773364b7cf0a85e77cd3310dd49812 --- .../MILPMinimalLabelSetGenerator.h | 6 +- .../SMTMinimalCommandSetGenerator.h | 6 +- .../SparseMarkovAutomatonCslModelChecker.h | 228 ++++--- .../prctl/SparseMdpPrctlModelChecker.h | 85 +-- ...ractNondeterministicLinearEquationSolver.h | 6 + ...mmxxNondeterministicLinearEquationSolver.h | 8 +- src/storm.cpp | 21 +- src/utility/solver.cpp | 25 + src/utility/solver.h | 11 +- src/utility/vector.h | 600 +++++++++--------- .../GmmxxMdpPrctlModelCheckerTest.cpp | 8 +- .../SparseMdpPrctlModelCheckerTest.cpp | 8 +- .../GmmxxMdpPrctModelCheckerTest.cpp | 4 +- .../SparseMdpPrctlModelCheckerTest.cpp | 4 +- 14 files changed, 513 insertions(+), 507 deletions(-) create mode 100644 src/utility/solver.cpp diff --git a/src/counterexamples/MILPMinimalLabelSetGenerator.h b/src/counterexamples/MILPMinimalLabelSetGenerator.h index 910948391..5023c342e 100644 --- a/src/counterexamples/MILPMinimalLabelSetGenerator.h +++ b/src/counterexamples/MILPMinimalLabelSetGenerator.h @@ -961,7 +961,7 @@ namespace storm { // (1) Check whether its possible to exceed the threshold if checkThresholdFeasible is set. double maximalReachabilityProbability = 0; - storm::modelchecker::prctl::SparseMdpPrctlModelChecker modelchecker(labeledMdp, new storm::solver::GmmxxNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker modelchecker(labeledMdp); std::vector result = modelchecker.checkUntil(false, phiStates, psiStates, false).first; for (auto state : labeledMdp.getInitialStates()) { maximalReachabilityProbability = std::max(maximalReachabilityProbability, result[state]); @@ -977,7 +977,7 @@ namespace storm { ChoiceInformation choiceInformation = determineRelevantAndProblematicChoices(labeledMdp, stateInformation, psiStates); // (4) Encode resulting system as MILP problem. - std::unique_ptr solver = storm::utility::solver::getLpSolver("MinimalCommandSetCounterexample"); + std::shared_ptr solver = storm::utility::solver::getLpSolver("MinimalCommandSetCounterexample"); // (4.1) Create variables. VariableInformation variableInformation = createVariables(*solver, labeledMdp, stateInformation, choiceInformation); @@ -1025,7 +1025,7 @@ namespace storm { storm::property::prctl::AbstractPathFormula const& pathFormula = probBoundFormula->getPathFormula(); storm::storage::BitVector phiStates; storm::storage::BitVector psiStates; - storm::modelchecker::prctl::SparseMdpPrctlModelChecker modelchecker(labeledMdp, new storm::solver::GmmxxNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker modelchecker(labeledMdp); try { storm::property::prctl::Until const& untilFormula = dynamic_cast const&>(pathFormula); diff --git a/src/counterexamples/SMTMinimalCommandSetGenerator.h b/src/counterexamples/SMTMinimalCommandSetGenerator.h index 2bbc1357c..a4041a070 100644 --- a/src/counterexamples/SMTMinimalCommandSetGenerator.h +++ b/src/counterexamples/SMTMinimalCommandSetGenerator.h @@ -1635,7 +1635,7 @@ namespace storm { // (1) Check whether its possible to exceed the threshold if checkThresholdFeasible is set. double maximalReachabilityProbability = 0; - storm::modelchecker::prctl::SparseMdpPrctlModelChecker modelchecker(labeledMdp, new storm::solver::GmmxxNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker modelchecker(labeledMdp); std::vector result = modelchecker.checkUntil(false, phiStates, psiStates, false).first; for (auto state : labeledMdp.getInitialStates()) { maximalReachabilityProbability = std::max(maximalReachabilityProbability, result[state]); @@ -1701,7 +1701,7 @@ namespace storm { modelCheckingClock = std::chrono::high_resolution_clock::now(); commandSet.insert(relevancyInformation.knownLabels); storm::models::Mdp subMdp = labeledMdp.restrictChoiceLabels(commandSet); - storm::modelchecker::prctl::SparseMdpPrctlModelChecker modelchecker(subMdp, new storm::solver::GmmxxNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker modelchecker(subMdp); LOG4CPLUS_DEBUG(logger, "Invoking model checker."); std::vector result = modelchecker.checkUntil(false, phiStates, psiStates, false).first; LOG4CPLUS_DEBUG(logger, "Computed model checking results."); @@ -1786,7 +1786,7 @@ namespace storm { storm::property::prctl::AbstractPathFormula const& pathFormula = probBoundFormula->getPathFormula(); storm::storage::BitVector phiStates; storm::storage::BitVector psiStates; - storm::modelchecker::prctl::SparseMdpPrctlModelChecker modelchecker(labeledMdp, new storm::solver::GmmxxNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker modelchecker(labeledMdp); try { storm::property::prctl::Until const& untilFormula = dynamic_cast const&>(pathFormula); diff --git a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h index 75b3e81cb..123503cac 100644 --- a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h +++ b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h @@ -2,7 +2,6 @@ #define STORM_MODELCHECKER_CSL_SPARSEMARKOVAUTOMATONCSLMODELCHECKER_H_ #include -#include #include "src/modelchecker/csl/AbstractModelChecker.h" #include "src/models/MarkovAutomaton.h" @@ -22,7 +21,7 @@ namespace storm { class SparseMarkovAutomatonCslModelChecker : public AbstractModelChecker { public: - explicit SparseMarkovAutomatonCslModelChecker(storm::models::MarkovAutomaton const& model) : AbstractModelChecker(model), minimumOperatorStack() { + explicit SparseMarkovAutomatonCslModelChecker(storm::models::MarkovAutomaton const& model, std::shared_ptr> nondeterministicLinearEquationSolver = storm::utility::solver::getNondeterministicLinearEquationSolver()) : AbstractModelChecker(model), minimumOperatorStack(), nondeterministicLinearEquationSolver(nondeterministicLinearEquationSolver) { // Intentionally left empty. } @@ -78,10 +77,10 @@ namespace storm { // * a matrix aProbabilisticToMarkovian with all (non-discretized) transitions from probabilistic non-goal states to all Markovian non-goal states. typename storm::storage::SparseMatrix aMarkovian = transitionMatrix.getSubmatrix(markovianNonGoalStates, nondeterministicChoiceIndices, true); typename storm::storage::SparseMatrix aMarkovianToProbabilistic = transitionMatrix.getSubmatrix(markovianNonGoalStates, probabilisticNonGoalStates, nondeterministicChoiceIndices); - std::vector markovianNondeterministicChoiceIndices = computeNondeterministicChoiceIndicesForConstraint(nondeterministicChoiceIndices, markovianNonGoalStates); + std::vector markovianNondeterministicChoiceIndices = storm::utility::vector::getConstrainedOffsetVector(nondeterministicChoiceIndices, markovianNonGoalStates); typename storm::storage::SparseMatrix aProbabilistic = transitionMatrix.getSubmatrix(probabilisticNonGoalStates, nondeterministicChoiceIndices); typename storm::storage::SparseMatrix aProbabilisticToMarkovian = transitionMatrix.getSubmatrix(probabilisticNonGoalStates, markovianNonGoalStates, nondeterministicChoiceIndices); - std::vector probabilisticNondeterministicChoiceIndices = computeNondeterministicChoiceIndicesForConstraint(nondeterministicChoiceIndices, probabilisticNonGoalStates); + std::vector probabilisticNondeterministicChoiceIndices = storm::utility::vector::getConstrainedOffsetVector(nondeterministicChoiceIndices, probabilisticNonGoalStates); // The matrices with transitions from Markovian states need to be digitized. // Digitize aMarkovian. Based on whether the transition is a self-loop or not, we apply the two digitization rules. @@ -128,7 +127,7 @@ namespace storm { } } - std::unique_ptr> nondeterministiclinearEquationSolver = storm::utility::solver::getNondeterministicLinearEquationSolver(); + std::shared_ptr> nondeterministiclinearEquationSolver = storm::utility::solver::getNondeterministicLinearEquationSolver(); // Perform the actual value iteration // * loop until the step bound has been reached @@ -234,9 +233,20 @@ namespace storm { } std::vector checkLongRunAverage(bool min, storm::storage::BitVector const& goalStates) const { + // Check whether the automaton is closed. if (!this->getModel().isClosed()) { throw storm::exceptions::InvalidArgumentException() << "Unable to compute long-run average on non-closed Markov automaton."; } + + // If there are no goal states, we avoid the computation and directly return zero. + if (goalStates.empty()) { + return std::vector(this->getModel().getNumberOfStates(), storm::utility::constGetZero()); + } + + // Likewise, if all bits are set, we can avoid the computation and set. + if ((~goalStates).empty()) { + return std::vector(this->getModel().getNumberOfStates(), storm::utility::constGetOne()); + } // Start by decomposing the Markov automaton into its MECs. storm::storage::MaximalEndComponentDecomposition mecDecomposition(this->getModel()); @@ -245,7 +255,7 @@ namespace storm { typename storm::storage::SparseMatrix const& transitionMatrix = this->getModel().getTransitionMatrix(); std::vector const& nondeterministicChoiceIndices = this->getModel().getNondeterministicChoiceIndices(); - // Now compute the long-run average for all end components in isolation. + // Now start with compute the long-run average for all end components in isolation. std::vector lraValuesForEndComponents; // While doing so, we already gather some information for the following steps. @@ -255,67 +265,16 @@ namespace storm { for (uint_fast64_t currentMecIndex = 0; currentMecIndex < mecDecomposition.size(); ++currentMecIndex) { storm::storage::MaximalEndComponent const& mec = mecDecomposition[currentMecIndex]; - std::unique_ptr solver = storm::utility::solver::getLpSolver("LRA for MEC " + std::to_string(currentMecIndex)); - solver->setModelSense(min ? storm::solver::LpSolver::MAXIMIZE : storm::solver::LpSolver::MINIMIZE); - - // First, we need to create the variables for the problem. - std::map stateToVariableIndexMap; - for (auto const& stateChoicesPair : mec) { - stateToVariableIndexMap[stateChoicesPair.first] = solver->createContinuousVariable("x" + std::to_string(stateChoicesPair.first), storm::solver::LpSolver::UNBOUNDED, 0, 0, 0); - } - uint_fast64_t lraValueVariableIndex = solver->createContinuousVariable("k", storm::solver::LpSolver::UNBOUNDED, 0, 0, 1); - - // Now we encode the problem as constraints. - std::vector variables; - std::vector coefficients; + // Gather information for later use. for (auto const& stateChoicesPair : mec) { uint_fast64_t state = stateChoicesPair.first; - // Gather information for later use. statesInMecs.set(state); stateToMecIndexMap[state] = currentMecIndex; - - // Now, based on the type of the state, create a suitable constraint. - if (this->getModel().isMarkovianState(state)) { - variables.clear(); - coefficients.clear(); - - variables.push_back(stateToVariableIndexMap.at(state)); - coefficients.push_back(1); - - typename storm::storage::SparseMatrix::Rows row = transitionMatrix.getRow(nondeterministicChoiceIndices[state]); - for (auto element : row) { - variables.push_back(stateToVariableIndexMap.at(element.column())); - coefficients.push_back(-element.value()); - } - - variables.push_back(lraValueVariableIndex); - coefficients.push_back(storm::utility::constGetOne() / this->getModel().getExitRate(state)); - - solver->addConstraint("state" + std::to_string(state), variables, coefficients, min ? storm::solver::LpSolver::LESS_EQUAL : storm::solver::LpSolver::GREATER_EQUAL, goalStates.get(state) ? storm::utility::constGetOne() / this->getModel().getExitRate(state) : storm::utility::constGetZero()); - } else { - // For probabilistic states, we want to add the constraint x_s <= sum P(s, a, s') * x_s' where a is the current action - // and the sum ranges over all states s'. - for (auto choice : stateChoicesPair.second) { - variables.clear(); - coefficients.clear(); - - variables.push_back(stateToVariableIndexMap.at(state)); - coefficients.push_back(1); - - typename storm::storage::SparseMatrix::Rows row = transitionMatrix.getRow(choice); - for (auto element : row) { - variables.push_back(stateToVariableIndexMap.at(element.column())); - coefficients.push_back(-element.value()); - } - - solver->addConstraint("state" + std::to_string(state), variables, coefficients, min ? storm::solver::LpSolver::LESS_EQUAL : storm::solver::LpSolver::GREATER_EQUAL, storm::utility::constGetZero()); - } - } } - - solver->optimize(); - lraValuesForEndComponents.push_back(solver->getContinuousValue(lraValueVariableIndex)); + + // Compute the LRA value for the current MEC. + lraValuesForEndComponents.push_back(this->computeLraForMaximalEndComponent(min, transitionMatrix, nondeterministicChoiceIndices, this->getModel().getMarkovianStates(), this->getModel().getExitRates(), goalStates, mec)); } // For fast transition rewriting, we build some auxiliary data structures. @@ -429,7 +388,7 @@ namespace storm { // Finalize the matrix and solve the corresponding system of equations. sspMatrix.finalize(); std::vector x(numberOfStatesNotInMecs + mecDecomposition.size()); - std::unique_ptr> nondeterministiclinearEquationSolver = storm::utility::solver::getNondeterministicLinearEquationSolver(); + std::shared_ptr> nondeterministiclinearEquationSolver = storm::utility::solver::getNondeterministicLinearEquationSolver(); nondeterministiclinearEquationSolver->solveEquationSystem(min, sspMatrix, x, b, sspNondeterministicChoiceIndices); // Prepare result vector. @@ -451,10 +410,102 @@ namespace storm { } std::vector checkExpectedTime(bool min, storm::storage::BitVector const& goalStates) const { + // Reduce the problem of computing the expected time to computing expected rewards where the rewards + // for all probabilistic states are zero and the reward values of Markovian states is 1. + std::vector rewardValues(this->getModel().getNumberOfStates(), storm::utility::constGetZero()); + storm::utility::vector::setVectorValues(rewardValues, this->getModel().getMarkovianStates(), storm::utility::constGetOne()); + return this->computeExpectedRewards(min, goalStates, rewardValues); + } + + protected: + /*! + * Computes the long-run average value for the given maximal end component of a Markov automaton. + * + * @param min Sets whether the long-run average is to be minimized or maximized. + * @param transitionMatrix The transition matrix of the underlying Markov automaton. + * @param nondeterministicChoiceIndices A vector indicating at which row the choice of a given state begins. + * @param markovianStates A bit vector storing all markovian states. + * @param exitRates A vector with exit rates for all states. Exit rates of probabilistic states are assumed to be zero. + * @param goalStates A bit vector indicating which states are to be considered as goal states. + * @param mec The maximal end component to consider for computing the long-run average. + * @return The long-run average of being in a goal state for the given MEC. + */ + static ValueType computeLraForMaximalEndComponent(bool min, storm::storage::SparseMatrix const& transitionMatrix, std::vector const& nondeterministicChoiceIndices, storm::storage::BitVector const& markovianStates, std::vector const& exitRates, storm::storage::BitVector const& goalStates, storm::storage::MaximalEndComponent const& mec) { + std::shared_ptr solver = storm::utility::solver::getLpSolver("LRA for MEC"); + solver->setModelSense(min ? storm::solver::LpSolver::MAXIMIZE : storm::solver::LpSolver::MINIMIZE); + + // First, we need to create the variables for the problem. + std::map stateToVariableIndexMap; + for (auto const& stateChoicesPair : mec) { + stateToVariableIndexMap[stateChoicesPair.first] = solver->createContinuousVariable("x" + std::to_string(stateChoicesPair.first), storm::solver::LpSolver::UNBOUNDED, 0, 0, 0); + } + uint_fast64_t lraValueVariableIndex = solver->createContinuousVariable("k", storm::solver::LpSolver::UNBOUNDED, 0, 0, 1); + + // Now we encode the problem as constraints. + std::vector variables; + std::vector coefficients; + for (auto const& stateChoicesPair : mec) { + uint_fast64_t state = stateChoicesPair.first; + + // Now, based on the type of the state, create a suitable constraint. + if (markovianStates.get(state)) { + variables.clear(); + coefficients.clear(); + + variables.push_back(stateToVariableIndexMap.at(state)); + coefficients.push_back(1); + + typename storm::storage::SparseMatrix::Rows row = transitionMatrix.getRow(nondeterministicChoiceIndices[state]); + for (auto element : row) { + variables.push_back(stateToVariableIndexMap.at(element.column())); + coefficients.push_back(-element.value()); + } + + variables.push_back(lraValueVariableIndex); + coefficients.push_back(storm::utility::constGetOne() / exitRates[state]); + + solver->addConstraint("state" + std::to_string(state), variables, coefficients, min ? storm::solver::LpSolver::LESS_EQUAL : storm::solver::LpSolver::GREATER_EQUAL, goalStates.get(state) ? storm::utility::constGetOne() / exitRates[state] : storm::utility::constGetZero()); + } else { + // For probabilistic states, we want to add the constraint x_s <= sum P(s, a, s') * x_s' where a is the current action + // and the sum ranges over all states s'. + for (auto choice : stateChoicesPair.second) { + variables.clear(); + coefficients.clear(); + + variables.push_back(stateToVariableIndexMap.at(state)); + coefficients.push_back(1); + + typename storm::storage::SparseMatrix::Rows row = transitionMatrix.getRow(choice); + for (auto element : row) { + variables.push_back(stateToVariableIndexMap.at(element.column())); + coefficients.push_back(-element.value()); + } + + solver->addConstraint("state" + std::to_string(state), variables, coefficients, min ? storm::solver::LpSolver::LESS_EQUAL : storm::solver::LpSolver::GREATER_EQUAL, storm::utility::constGetZero()); + } + } + } + + solver->optimize(); + return solver->getContinuousValue(lraValueVariableIndex); + } + + /*! + * Computes the expected reward that is gained from each state before entering any of the goal states. + * + * @param min Indicates whether minimal or maximal rewards are to be computed. + * @param goalStates The goal states that define until which point rewards are gained. + * @param stateRewards A vector that defines the reward gained in each state. For probabilistic states, this is an instantaneous reward + * that is fully gained and for Markovian states the actually gained reward is dependent on the expected time to stay in the + * state, i.e. it is gouverned by the exit rate of the state. + * @return A vector that contains the expected reward for each state of the model. + */ + std::vector computeExpectedRewards(bool min, storm::storage::BitVector const& goalStates, std::vector const& stateRewards) const { + // Check whether the automaton is closed. if (!this->getModel().isClosed()) { throw storm::exceptions::InvalidArgumentException() << "Unable to compute expected time on non-closed Markov automaton."; } - + // First, we need to check which states have infinite expected time (by definition). storm::storage::BitVector infinityStates; if (min) { @@ -501,77 +552,50 @@ namespace storm { infinityStates = storm::storage::BitVector(this->getModel().getNumberOfStates()); } } - + // Now we identify the states for which values need to be computed. storm::storage::BitVector maybeStates = ~(goalStates | infinityStates); // Then, we can eliminate the rows and columns for all states whose values are already known to be 0. std::vector x(maybeStates.getNumberOfSetBits()); - std::vector subNondeterministicChoiceIndices = computeNondeterministicChoiceIndicesForConstraint(this->getModel().getNondeterministicChoiceIndices(), maybeStates); + std::vector subNondeterministicChoiceIndices = storm::utility::vector::getConstrainedOffsetVector(this->getModel().getNondeterministicChoiceIndices(), maybeStates); storm::storage::SparseMatrix 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 meanSojournTimes(this->getModel().getExitRates()); + + // Now prepare the expected reward values for all states so they can be used as the right-hand side of the equation system. + std::vector rewardValues(stateRewards); for (auto state : this->getModel().getMarkovianStates()) { - meanSojournTimes[state] = storm::utility::constGetOne() / meanSojournTimes[state]; + rewardValues[state] = rewardValues[state] / this->getModel().getExitRates()[state]; } // Finally, prepare the actual right-hand side. std::vector b(submatrix.getRowCount()); - storm::utility::vector::selectVectorValuesRepeatedly(b, maybeStates, this->getModel().getNondeterministicChoiceIndices(), meanSojournTimes); + storm::utility::vector::selectVectorValuesRepeatedly(b, maybeStates, this->getModel().getNondeterministicChoiceIndices(), rewardValues); // Solve the corresponding system of equations. - std::unique_ptr> nondeterministiclinearEquationSolver = storm::utility::solver::getNondeterministicLinearEquationSolver(); + std::shared_ptr> nondeterministiclinearEquationSolver = storm::utility::solver::getNondeterministicLinearEquationSolver(); nondeterministiclinearEquationSolver->solveEquationSystem(min, submatrix, x, b, subNondeterministicChoiceIndices); // Create resulting vector. std::vector result(this->getModel().getNumberOfStates()); - + // Set values of resulting vector according to previous result and return the result. storm::utility::vector::setVectorValues(result, maybeStates, x); storm::utility::vector::setVectorValues(result, goalStates, storm::utility::constGetZero()); storm::utility::vector::setVectorValues(result, infinityStates, storm::utility::constGetInfinity()); - + 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 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. + * A solver that is used for solving systems of linear equations that are the result of nondeterministic choices. */ - static std::vector computeNondeterministicChoiceIndicesForConstraint(std::vector const& nondeterministicChoiceIndices, storm::storage::BitVector const& constraint) { - // Reserve the known amount of slots for the resulting vector. - std::vector 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; - } + std::shared_ptr> nondeterministicLinearEquationSolver; }; } } diff --git a/src/modelchecker/prctl/SparseMdpPrctlModelChecker.h b/src/modelchecker/prctl/SparseMdpPrctlModelChecker.h index 59f910895..b51bb5565 100644 --- a/src/modelchecker/prctl/SparseMdpPrctlModelChecker.h +++ b/src/modelchecker/prctl/SparseMdpPrctlModelChecker.h @@ -18,6 +18,7 @@ #include "src/models/Mdp.h" #include "src/utility/vector.h" #include "src/utility/graph.h" +#include "src/utility/solver.h" #include "src/settings/Settings.h" #include "src/storage/TotalScheduler.h" @@ -37,7 +38,7 @@ namespace storm { * * @param model The MDP to be checked. */ - explicit SparseMdpPrctlModelChecker(storm::models::Mdp const& model, storm::solver::AbstractNondeterministicLinearEquationSolver* linearEquationSolver) : AbstractModelChecker(model), minimumOperatorStack(), linearEquationSolver(linearEquationSolver) { + explicit SparseMdpPrctlModelChecker(storm::models::Mdp const& model, std::shared_ptr> nondeterministicLinearEquationSolver = storm::utility::solver::getNondeterministicLinearEquationSolver()) : AbstractModelChecker(model), minimumOperatorStack(), nondeterministicLinearEquationSolver(nondeterministicLinearEquationSolver) { // Intentionally left empty. } @@ -46,7 +47,7 @@ namespace storm { * constructed model checker will have the model of the given model checker as its associated model. */ explicit SparseMdpPrctlModelChecker(storm::modelchecker::prctl::SparseMdpPrctlModelChecker const& modelchecker) - : AbstractModelChecker(modelchecker), minimumOperatorStack(), linearEquationSolver(new storm::solver::AbstractNondeterministicLinearEquationSolver()) { + : AbstractModelChecker(modelchecker), minimumOperatorStack(), nondeterministicLinearEquationSolver(storm::utility::solver::getNondeterministicLinearEquationSolver()) { // Intentionally left empty. } @@ -115,7 +116,7 @@ namespace storm { storm::storage::SparseMatrix submatrix = this->getModel().getTransitionMatrix().getSubmatrix(statesWithProbabilityGreater0, this->getModel().getNondeterministicChoiceIndices()); // Get the "new" nondeterministic choice indices for the submatrix. - std::vector subNondeterministicChoiceIndices = this->computeNondeterministicChoiceIndicesForConstraint(statesWithProbabilityGreater0); + std::vector subNondeterministicChoiceIndices = storm::utility::vector::getConstrainedOffsetVector(this->getModel().getNondeterministicChoiceIndices(), statesWithProbabilityGreater0); // Compute the new set of target states in the reduced system. storm::storage::BitVector rightStatesInReducedSystem = statesWithProbabilityGreater0 % psiStates; @@ -127,11 +128,7 @@ namespace storm { std::vector subresult(statesWithProbabilityGreater0.getNumberOfSetBits()); storm::utility::vector::setVectorValues(subresult, rightStatesInReducedSystem, storm::utility::constGetOne()); - if (linearEquationSolver != nullptr) { - this->linearEquationSolver->performMatrixVectorMultiplication(this->minimumOperatorStack.top(), submatrix, subresult, subNondeterministicChoiceIndices, nullptr, stepBound); - } else { - throw storm::exceptions::InvalidStateException() << "No valid linear equation solver available."; - } + this->nondeterministicLinearEquationSolver->performMatrixVectorMultiplication(this->minimumOperatorStack.top(), submatrix, subresult, subNondeterministicChoiceIndices, nullptr, stepBound); // Set the values of the resulting vector accordingly. storm::utility::vector::setVectorValues(result, statesWithProbabilityGreater0, subresult); @@ -172,11 +169,7 @@ namespace storm { std::vector result(this->getModel().getNumberOfStates()); storm::utility::vector::setVectorValues(result, nextStates, storm::utility::constGetOne()); - if (linearEquationSolver != nullptr) { - this->linearEquationSolver->performMatrixVectorMultiplication(this->minimumOperatorStack.top(), this->getModel().getTransitionMatrix(), result, this->getModel().getNondeterministicChoiceIndices()); - } else { - throw storm::exceptions::InvalidStateException() << "No valid linear equation solver available."; - } + this->nondeterministicLinearEquationSolver->performMatrixVectorMultiplication(this->minimumOperatorStack.top(), this->getModel().getTransitionMatrix(), result, this->getModel().getNondeterministicChoiceIndices()); return result; } @@ -325,7 +318,7 @@ namespace storm { storm::storage::SparseMatrix submatrix = this->getModel().getTransitionMatrix().getSubmatrix(maybeStates, this->getModel().getNondeterministicChoiceIndices()); // Get the "new" nondeterministic choice indices for the submatrix. - std::vector subNondeterministicChoiceIndices = this->computeNondeterministicChoiceIndicesForConstraint(maybeStates); + std::vector subNondeterministicChoiceIndices = storm::utility::vector::getConstrainedOffsetVector(this->getModel().getNondeterministicChoiceIndices(), maybeStates); // Prepare the right-hand side of the equation system. For entry i this corresponds to // the accumulated probability of going from state i to some 'yes' state. @@ -335,11 +328,7 @@ namespace storm { std::vector x(maybeStates.getNumberOfSetBits()); // Solve the corresponding system of equations. - if (linearEquationSolver != nullptr) { - this->linearEquationSolver->solveEquationSystem(minimize, submatrix, x, b, subNondeterministicChoiceIndices); - } else { - throw storm::exceptions::InvalidStateException() << "No valid linear equation solver available."; - } + this->nondeterministicLinearEquationSolver->solveEquationSystem(minimize, submatrix, x, b, subNondeterministicChoiceIndices); // Set values of resulting vector according to result. storm::utility::vector::setVectorValues(result, maybeStates, x); @@ -376,11 +365,7 @@ namespace storm { // Initialize result to state rewards of the model. std::vector result(this->getModel().getStateRewardVector()); - if (linearEquationSolver != nullptr) { - this->linearEquationSolver->performMatrixVectorMultiplication(this->minimumOperatorStack.top(), this->getModel().getTransitionMatrix(), result, this->getModel().getNondeterministicChoiceIndices(), nullptr, formula.getBound()); - } else { - throw storm::exceptions::InvalidStateException() << "No valid linear equation solver available."; - } + this->nondeterministicLinearEquationSolver->performMatrixVectorMultiplication(this->minimumOperatorStack.top(), this->getModel().getTransitionMatrix(), result, this->getModel().getNondeterministicChoiceIndices(), nullptr, formula.getBound()); return result; } @@ -422,11 +407,7 @@ namespace storm { result.resize(this->getModel().getNumberOfStates()); } - if (linearEquationSolver != nullptr) { - this->linearEquationSolver->performMatrixVectorMultiplication(this->minimumOperatorStack.top(), this->getModel().getTransitionMatrix(), result, this->getModel().getNondeterministicChoiceIndices(), &totalRewardVector, formula.getBound()); - } else { - throw storm::exceptions::InvalidStateException() << "No valid linear equation solver available."; - } + this->nondeterministicLinearEquationSolver->performMatrixVectorMultiplication(this->minimumOperatorStack.top(), this->getModel().getTransitionMatrix(), result, this->getModel().getNondeterministicChoiceIndices(), &totalRewardVector, formula.getBound()); return result; } @@ -500,7 +481,7 @@ namespace storm { storm::storage::SparseMatrix submatrix = this->getModel().getTransitionMatrix().getSubmatrix(maybeStates, this->getModel().getNondeterministicChoiceIndices()); // Get the "new" nondeterministic choice indices for the submatrix. - std::vector subNondeterministicChoiceIndices = this->computeNondeterministicChoiceIndicesForConstraint(maybeStates); + std::vector subNondeterministicChoiceIndices = storm::utility::vector::getConstrainedOffsetVector(this->getModel().getNondeterministicChoiceIndices(), maybeStates); // Prepare the right-hand side of the equation system. For entry i this corresponds to // the accumulated probability of going from state i to some 'yes' state. @@ -534,11 +515,7 @@ namespace storm { std::vector x(maybeStates.getNumberOfSetBits()); // Solve the corresponding system of equations. - if (linearEquationSolver != nullptr) { - this->linearEquationSolver->solveEquationSystem(minimize, submatrix, x, b, subNondeterministicChoiceIndices); - } else { - throw storm::exceptions::InvalidStateException() << "No valid linear equation solver available."; - } + this->nondeterministicLinearEquationSolver->solveEquationSystem(minimize, submatrix, x, b, subNondeterministicChoiceIndices); // Set values of resulting vector according to result. storm::utility::vector::setVectorValues(result, maybeStates, x); @@ -601,45 +578,11 @@ namespace storm { */ mutable std::stack 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. + * A solver that is used for solving systems of linear equations that are the result of nondeterministic choices. */ - std::vector computeNondeterministicChoiceIndicesForConstraint(storm::storage::BitVector const& constraint) const { - // First, get a reference to the full nondeterministic choice indices. - std::vector const& nondeterministicChoiceIndices = this->getModel().getNondeterministicChoiceIndices(); - - // Reserve the known amount of slots for the resulting vector. - std::vector 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> linearEquationSolver; - + std::shared_ptr> nondeterministicLinearEquationSolver; }; - } // namespace prctl } // namespace modelchecker } // namespace storm diff --git a/src/solver/AbstractNondeterministicLinearEquationSolver.h b/src/solver/AbstractNondeterministicLinearEquationSolver.h index f0036a397..4ab08cf90 100644 --- a/src/solver/AbstractNondeterministicLinearEquationSolver.h +++ b/src/solver/AbstractNondeterministicLinearEquationSolver.h @@ -86,8 +86,10 @@ namespace storm { // auto startTime = std::chrono::high_resolution_clock::now(); // std::chrono::nanoseconds totalTime(0); + bool multiplyResultMemoryProvided = true; if (multiplyResult == nullptr) { multiplyResult = new std::vector(A.getRowCount()); + multiplyResultMemoryProvided = false; } std::vector* currentX = &x; bool xMemoryProvided = true; @@ -134,6 +136,10 @@ namespace storm { } else if (!xMemoryProvided) { delete newX; } + + if (!multiplyResultMemoryProvided) { + delete multiplyResult; + } // // Check if the solver converged and issue a warning otherwise. // if (converged) { diff --git a/src/solver/GmmxxNondeterministicLinearEquationSolver.h b/src/solver/GmmxxNondeterministicLinearEquationSolver.h index bef5c838f..205e724f1 100644 --- a/src/solver/GmmxxNondeterministicLinearEquationSolver.h +++ b/src/solver/GmmxxNondeterministicLinearEquationSolver.h @@ -52,8 +52,10 @@ namespace storm { gmm::csr_matrix* gmmxxMatrix = storm::adapters::GmmxxAdapter::toGmmxxSparseMatrix(A); // Set up the environment for the power method. + bool multiplyResultMemoryProvided = true; if (multiplyResult == nullptr) { multiplyResult = new std::vector(A.getRowCount()); + multiplyResultMemoryProvided = false; } std::vector* currentX = &x; @@ -101,7 +103,11 @@ namespace storm { delete newX; } delete gmmxxMatrix; - + + if (!multiplyResultMemoryProvided) { + delete multiplyResult; + } + // Check if the solver converged and issue a warning otherwise. if (converged) { LOG4CPLUS_INFO(logger, "Iterative solver converged after " << iterations << " iterations."); diff --git a/src/storm.cpp b/src/storm.cpp index 820b2be7f..6aee44050 100644 --- a/src/storm.cpp +++ b/src/storm.cpp @@ -231,18 +231,7 @@ storm::modelchecker::prctl::AbstractModelChecker* createPrctlModelChecke */ storm::modelchecker::prctl::AbstractModelChecker* createPrctlModelChecker(storm::models::Mdp& mdp) { // Create the appropriate model checker. - storm::settings::Settings* s = storm::settings::Settings::getInstance(); - std::string const chosenMatrixLibrary = s->getOptionByLongName("matrixLibrary").getArgument(0).getValueAsString(); - if (chosenMatrixLibrary == "gmm++") { - return new storm::modelchecker::prctl::SparseMdpPrctlModelChecker(mdp, new storm::solver::GmmxxNondeterministicLinearEquationSolver()); - } else if (chosenMatrixLibrary == "native") { - return new storm::modelchecker::prctl::SparseMdpPrctlModelChecker(mdp, new storm::solver::AbstractNondeterministicLinearEquationSolver()); - } - - // The control flow should never reach this point, as there is a default setting for matrixlib. - std::string message = "No matrix library suitable for MDP model checking has been set."; - throw storm::exceptions::InvalidSettingsException() << message; - return nullptr; + return new storm::modelchecker::prctl::SparseMdpPrctlModelChecker(mdp); } /*! @@ -473,10 +462,10 @@ int main(const int argc, const char* argv[]) { LOG4CPLUS_INFO(logger, "Model is a Markov automaton."); std::shared_ptr> markovAutomaton = parser.getModel>(); markovAutomaton->close(); - storm::modelchecker::csl::SparseMarkovAutomatonCslModelChecker mc(*markovAutomaton); -// std::cout << mc.checkExpectedTime(true, markovAutomaton->getLabeledStates("goal")) << std::endl; -// std::cout << mc.checkLongRunAverage(true, markovAutomaton->getLabeledStates("goal")) << std::endl; -// std::cout << mc.checkTimeBoundedEventually(true, markovAutomaton->getLabeledStates("goal"), 0, 1) << std::endl; + storm::modelchecker::csl::SparseMarkovAutomatonCslModelChecker mc(*markovAutomaton, std::shared_ptr>(new storm::solver::AbstractNondeterministicLinearEquationSolver())); + std::cout << mc.checkExpectedTime(true, markovAutomaton->getLabeledStates("goal")) << std::endl; + std::cout << mc.checkLongRunAverage(true, markovAutomaton->getLabeledStates("goal")) << std::endl; + std::cout << mc.checkTimeBoundedEventually(true, markovAutomaton->getLabeledStates("goal"), 0, 1) << std::endl; std::cout << mc.checkTimeBoundedEventually(true, markovAutomaton->getLabeledStates("goal"), 1, 2) << std::endl; break; } diff --git a/src/utility/solver.cpp b/src/utility/solver.cpp new file mode 100644 index 000000000..16578aaf8 --- /dev/null +++ b/src/utility/solver.cpp @@ -0,0 +1,25 @@ +#include "src/utility/solver.h" + +namespace storm { + namespace utility { + namespace solver { + std::shared_ptr getLpSolver(std::string const& name) { + return std::shared_ptr(new storm::solver::GurobiLpSolver(name)); + } + + template + std::shared_ptr> getNondeterministicLinearEquationSolver() { + std::string const& matrixLibrary = storm::settings::Settings::getInstance()->getOptionByLongName("matrixLibrary").getArgument(0).getValueAsString(); + if (matrixLibrary == "gmm++") { + return std::shared_ptr>(new storm::solver::GmmxxNondeterministicLinearEquationSolver()); + } else if (matrixLibrary == "native") { + return std::shared_ptr>(new storm::solver::AbstractNondeterministicLinearEquationSolver()); + } + + throw storm::exceptions::InvalidSettingsException() << "No suitable nondeterministic linear equation solver selected."; + } + + template std::shared_ptr> getNondeterministicLinearEquationSolver(); + } + } +} \ No newline at end of file diff --git a/src/utility/solver.h b/src/utility/solver.h index fdf838174..165ba9c00 100644 --- a/src/utility/solver.h +++ b/src/utility/solver.h @@ -5,19 +5,16 @@ #include "src/solver/GmmxxNondeterministicLinearEquationSolver.h" #include "src/solver/GurobiLpSolver.h" +#include "src/exceptions/InvalidSettingsException.h" + namespace storm { namespace utility { namespace solver { - std::unique_ptr getLpSolver(std::string const& name) { - return std::unique_ptr(new storm::solver::GurobiLpSolver(name)); - } + std::shared_ptr getLpSolver(std::string const& name); template - std::unique_ptr> getNondeterministicLinearEquationSolver() { - return std::unique_ptr>(new storm::solver::AbstractNondeterministicLinearEquationSolver()); -// return std::unique_ptr>(new storm::solver::GmmxxNondeterministicLinearEquationSolver()); - } + std::shared_ptr> getNondeterministicLinearEquationSolver(); } } } diff --git a/src/utility/vector.h b/src/utility/vector.h index 0ee5033d2..c9bf3b35d 100644 --- a/src/utility/vector.h +++ b/src/utility/vector.h @@ -1,10 +1,3 @@ -/* - * vector.h - * - * Created on: 06.12.2012 - * Author: Christian Dehnert - */ - #ifndef STORM_UTILITY_VECTOR_H_ #define STORM_UTILITY_VECTOR_H_ @@ -14,295 +7,318 @@ #include #include -#include "log4cplus/logger.h" -#include "log4cplus/loggingmacros.h" - -extern log4cplus::Logger logger; - namespace storm { - -namespace utility { - -namespace vector { - -/*! - * Sets the provided values at the provided positions in the given vector. - * - * @param vector The vector in which the values are to be set. - * @param positions The positions at which the values are to be set. - * @param values The values that are to be set. - */ -template -void setVectorValues(std::vector& vector, storm::storage::BitVector const& positions, std::vector const& values) { - uint_fast64_t oldPosition = 0; - for (auto position : positions) { - vector[position] = values[oldPosition++]; - } -} - -/*! - * Sets the provided value at the provided positions in the given vector. - * - * @param vector The vector in which the value is to be set. - * @param positions The positions at which the value is to be set. - * @param value The value that is to be set. - */ -template -void setVectorValues(std::vector& vector, storm::storage::BitVector const& positions, T value) { - for (auto position : positions) { - vector[position] = value; - } -} - -/*! - * Sets the provided value at the provided positions in the given vector. - * - * @param vector The vector in which the value is to be set. - * @param positions The positions at which the value is to be set. - * @param value The value that is to be set. - */ -template -void setVectorValues(Eigen::Matrix& eigenVector, storm::storage::BitVector const& positions, T value) { - for (auto position : positions) { - eigenVector(position, 0) = value; - } -} - -/*! - * Selects the elements from a vector at the specified positions and writes them consecutively into another vector. - * @param vector The vector into which the selected elements are to be written. - * @param positions The positions at which to select the elements from the values vector. - * @param values The vector from which to select the elements. - */ -template -void selectVectorValues(std::vector& vector, storm::storage::BitVector const& positions, std::vector const& values) { - uint_fast64_t oldPosition = 0; - for (auto position : positions) { - vector[oldPosition++] = values[position]; - } -} - -/*! - * Selects groups of elements from a vector at the specified positions and writes them consecutively into another vector. - * - * @param vector The vector into which the selected elements are to be written. - * @param positions The positions of the groups of elements that are to be selected. - * @param rowGrouping A vector that specifies the begin and end of each group of elements in the values vector. - * @param values The vector from which to select groups of elements. - */ -template -void selectVectorValues(std::vector& vector, storm::storage::BitVector const& positions, std::vector const& rowGrouping, std::vector const& values) { - uint_fast64_t oldPosition = 0; - for (auto position : positions) { - for (uint_fast64_t i = rowGrouping[position]; i < rowGrouping[position + 1]; ++i) { - vector[oldPosition++] = values[i]; - } - } -} - -/*! - * Selects one element out of each row group and writes it to the target vector. - * - * @param vector The target vector to which the values are written. - * @param rowGroupToRowIndexMapping A mapping from row group indices to an offset that specifies which of the values to - * take from the row group. - * @param rowGrouping A vector that specifies the begin and end of each group of elements in the values vector. - * @param values The vector from which to select the values. - */ -template -void selectVectorValues(std::vector& vector, std::vector const& rowGroupToRowIndexMapping, std::vector const& rowGrouping, std::vector const& values) { - uint_fast64_t oldPosition = 0; - for (uint_fast64_t i = 0; i < vector.size(); ++i) { - vector[i] = values[rowGrouping[i] + rowGroupToRowIndexMapping[i]]; - } -} - -/*! - * Selects values from a vector at the specified positions and writes them into another vector as often as given by - * the size of the corresponding group of elements. - * - * @param vector The vector into which the selected elements are written. - * @param positions The positions at which to select the values. - * @param rowGrouping A vector that specifies the begin and end of each group of elements in the values vector. This - * implicitly defines the number of times any element is written to the output vector. - */ -template -void selectVectorValuesRepeatedly(std::vector& vector, storm::storage::BitVector const& positions, std::vector const& rowGrouping, std::vector const& values) { - uint_fast64_t oldPosition = 0; - for (auto position : positions) { - for (uint_fast64_t i = rowGrouping[position]; i < rowGrouping[position + 1]; ++i) { - vector[oldPosition++] = values[position]; - } - } -} - -/*! - * Subtracts the given vector from the constant one-vector and writes the result to the input vector. - * - * @param vector The vector that is to be subtracted from the constant one-vector. - */ -template -void subtractFromConstantOneVector(std::vector& vector) { - for (auto& element : vector) { - element = storm::utility::constGetOne() - element; - } -} - -/*! - * Adds the two given vectors and writes the result into the first operand. - * - * @param target The first summand and target vector. - * @param summand The second summand. - */ -template -void addVectorsInPlace(std::vector& target, std::vector const& summand) { - if (target.size() != summand.size()) { - LOG4CPLUS_ERROR(logger, "Lengths of vectors do not match, which makes operation impossible."); - throw storm::exceptions::InvalidArgumentException() << "Length of vectors do not match, which makes operation impossible."; - } - - std::transform(target.begin(), target.end(), summand.begin(), target.begin(), std::plus()); -} - -/*! - * Reduces the given source vector by selecting an element according to the given filter out of each row group. - * - * @param source The source vector which is to be reduced. - * @param target The target vector into which a single element from each row group is written. - * @param rowGrouping A vector that specifies the begin and end of each group of elements in the values vector. - * @param filter A function that compares two elements v1 and v2 according to some filter criterion. This function must - * return true iff v1 is supposed to be taken instead of v2. - * @param choices If non-null, this vector is used to store the choices made during the selection. - */ -template -void reduceVector(std::vector const& source, std::vector& target, std::vector const& rowGrouping, std::function filter, std::vector* choices = nullptr) { - uint_fast64_t currentSourceRow = 0; - uint_fast64_t currentTargetRow = -1; - uint_fast64_t currentLocalRow = 0; - for (auto it = source.cbegin(), ite = source.cend(); it != ite; ++it, ++currentSourceRow, ++currentLocalRow) { - // Check whether we have considered all source rows for the current target row. - if (rowGrouping[currentTargetRow + 1] <= currentSourceRow || currentSourceRow == 0) { - currentLocalRow = 0; - ++currentTargetRow; - target[currentTargetRow] = source[currentSourceRow]; - if (choices != nullptr) { - (*choices)[currentTargetRow] = 0; + namespace utility { + namespace vector { + + /*! + * Sets the provided values at the provided positions in the given vector. + * + * @param vector The vector in which the values are to be set. + * @param positions The positions at which the values are to be set. + * @param values The values that are to be set. + */ + template + void setVectorValues(std::vector& vector, storm::storage::BitVector const& positions, std::vector const& values) { + uint_fast64_t oldPosition = 0; + for (auto position : positions) { + vector[position] = values[oldPosition++]; + } } - continue; - } - // We have to upate the value, so only overwrite the current value if the value passes the filter. - if (filter(*it, target[currentTargetRow])) { - target[currentTargetRow] = *it; - if (choices != nullptr) { - (*choices)[currentTargetRow] = currentLocalRow; + /*! + * Sets the provided value at the provided positions in the given vector. + * + * @param vector The vector in which the value is to be set. + * @param positions The positions at which the value is to be set. + * @param value The value that is to be set. + */ + template + void setVectorValues(std::vector& vector, storm::storage::BitVector const& positions, T value) { + for (auto position : positions) { + vector[position] = value; + } } - } - } -} - -/*! - * Reduces the given source vector by selecting the smallest element out of each row group. - * - * @param source The source vector which is to be reduced. - * @param target The target vector into which a single element from each row group is written. - * @param rowGrouping A vector that specifies the begin and end of each group of elements in the source vector. - * @param choices If non-null, this vector is used to store the choices made during the selection. - */ -template -void reduceVectorMin(std::vector const& source, std::vector& target, std::vector const& rowGrouping, std::vector* choices = nullptr) { - reduceVector(source, target, rowGrouping, std::less(), choices); -} - -/*! - * Reduces the given source vector by selecting the largest element out of each row group. - * - * @param source The source vector which is to be reduced. - * @param target The target vector into which a single element from each row group is written. - * @param rowGrouping A vector that specifies the begin and end of each group of elements in the source vector. - * @param choices If non-null, this vector is used to store the choices made during the selection. - */ -template -void reduceVectorMax(std::vector const& source, std::vector& target, std::vector const& rowGrouping, std::vector* choices = nullptr) { - reduceVector(source, target, rowGrouping, std::greater(), choices); -} - -/*! - * Compares the given elements and determines whether they are equal modulo the given precision. The provided flag - * additionaly specifies whether the error is computed in relative or absolute terms. - * - * @param val1 The first value to compare. - * @param val2 The second value to compare. - * @param precision The precision up to which the elements are compared. - * @param relativeError If set, the error is computed relative to the second value. - * @return True iff the elements are considered equal. - */ -template -bool equalModuloPrecision(T const& val1, T const& val2, T precision, bool relativeError = true) { - if (relativeError) { - if (std::abs(val1 - val2)/val2 > precision) return false; - } else { - if (std::abs(val1 - val2) > precision) return false; - } - return true; -} - -/*! - * Compares the two vectors and determines whether they are equal modulo the provided precision. Depending on whether the - * flag is set, the difference between the vectors is computed relative to the value or in absolute terms. - * - * @param vectorLeft The first vector of the comparison. - * @param vectorRight The second vector of the comparison. - * @param precision The precision up to which the vectors are to be checked for equality. - * @param relativeError If set, the difference between the vectors is computed relative to the value or in absolute terms. - */ -template -bool equalModuloPrecision(std::vector const& vectorLeft, std::vector const& vectorRight, T precision, bool relativeError) { - if (vectorLeft.size() != vectorRight.size()) { - LOG4CPLUS_ERROR(logger, "Lengths of vectors do not match, which makes comparison impossible."); - throw storm::exceptions::InvalidArgumentException() << "Lengths of vectors do not match, which makes comparison impossible."; - } - - for (uint_fast64_t i = 0; i < vectorLeft.size(); ++i) { - if (!equalModuloPrecision(vectorLeft[i], vectorRight[i], precision, relativeError)) { - return false; - } - } - - return true; -} - -/*! - * Compares the two vectors at the specified positions and determines whether they are equal modulo the provided - * precision. Depending on whether the flag is set, the difference between the vectors is computed relative to the value - * or in absolute terms. - * - * @param vectorLeft The first vector of the comparison. - * @param vectorRight The second vector of the comparison. - * @param precision The precision up to which the vectors are to be checked for equality. - * @param positions A vector representing a set of positions at which the vectors are compared. - * @param relativeError If set, the difference between the vectors is computed relative to the value or in absolute terms. - */ -template -bool equalModuloPrecision(std::vector const& vectorLeft, std::vector const& vectorRight, std::vector const& positions, T precision, bool relativeError) { - if (vectorLeft.size() != vectorRight.size()) { - LOG4CPLUS_ERROR(logger, "Lengths of vectors do not match, which makes comparison impossible."); - throw storm::exceptions::InvalidArgumentException() << "Lengths of vectors do not match, which makes comparison impossible."; - } - - for (uint_fast64_t position : positions) { - if (!equalModuloPrecision(vectorLeft[position], vectorRight[position], precision, relativeError)) { - return false; - } - } - - return true; -} - -} // namespace vector - -} // namespace utility - + + /*! + * Sets the provided value at the provided positions in the given vector. + * + * @param vector The vector in which the value is to be set. + * @param positions The positions at which the value is to be set. + * @param value The value that is to be set. + */ + template + void setVectorValues(Eigen::Matrix& eigenVector, storm::storage::BitVector const& positions, T value) { + for (auto position : positions) { + eigenVector(position, 0) = value; + } + } + + /*! + * Selects the elements from a vector at the specified positions and writes them consecutively into another vector. + * @param vector The vector into which the selected elements are to be written. + * @param positions The positions at which to select the elements from the values vector. + * @param values The vector from which to select the elements. + */ + template + void selectVectorValues(std::vector& vector, storm::storage::BitVector const& positions, std::vector const& values) { + uint_fast64_t oldPosition = 0; + for (auto position : positions) { + vector[oldPosition++] = values[position]; + } + } + + /*! + * Selects groups of elements from a vector at the specified positions and writes them consecutively into another vector. + * + * @param vector The vector into which the selected elements are to be written. + * @param positions The positions of the groups of elements that are to be selected. + * @param rowGrouping A vector that specifies the begin and end of each group of elements in the values vector. + * @param values The vector from which to select groups of elements. + */ + template + void selectVectorValues(std::vector& vector, storm::storage::BitVector const& positions, std::vector const& rowGrouping, std::vector const& values) { + uint_fast64_t oldPosition = 0; + for (auto position : positions) { + for (uint_fast64_t i = rowGrouping[position]; i < rowGrouping[position + 1]; ++i) { + vector[oldPosition++] = values[i]; + } + } + } + + /*! + * Selects one element out of each row group and writes it to the target vector. + * + * @param vector The target vector to which the values are written. + * @param rowGroupToRowIndexMapping A mapping from row group indices to an offset that specifies which of the values to + * take from the row group. + * @param rowGrouping A vector that specifies the begin and end of each group of elements in the values vector. + * @param values The vector from which to select the values. + */ + template + void selectVectorValues(std::vector& vector, std::vector const& rowGroupToRowIndexMapping, std::vector const& rowGrouping, std::vector const& values) { + uint_fast64_t oldPosition = 0; + for (uint_fast64_t i = 0; i < vector.size(); ++i) { + vector[i] = values[rowGrouping[i] + rowGroupToRowIndexMapping[i]]; + } + } + + /*! + * Selects values from a vector at the specified positions and writes them into another vector as often as given by + * the size of the corresponding group of elements. + * + * @param vector The vector into which the selected elements are written. + * @param positions The positions at which to select the values. + * @param rowGrouping A vector that specifies the begin and end of each group of elements in the values vector. This + * implicitly defines the number of times any element is written to the output vector. + */ + template + void selectVectorValuesRepeatedly(std::vector& vector, storm::storage::BitVector const& positions, std::vector const& rowGrouping, std::vector const& values) { + uint_fast64_t oldPosition = 0; + for (auto position : positions) { + for (uint_fast64_t i = rowGrouping[position]; i < rowGrouping[position + 1]; ++i) { + vector[oldPosition++] = values[position]; + } + } + } + + /*! + * Subtracts the given vector from the constant one-vector and writes the result to the input vector. + * + * @param vector The vector that is to be subtracted from the constant one-vector. + */ + template + void subtractFromConstantOneVector(std::vector& vector) { + for (auto& element : vector) { + element = storm::utility::constGetOne() - element; + } + } + + /*! + * Adds the two given vectors and writes the result into the first operand. + * + * @param target The first summand and target vector. + * @param summand The second summand. + */ + template + void addVectorsInPlace(std::vector& target, std::vector const& summand) { + if (target.size() != summand.size()) { + LOG4CPLUS_ERROR(logger, "Lengths of vectors do not match, which makes operation impossible."); + throw storm::exceptions::InvalidArgumentException() << "Length of vectors do not match, which makes operation impossible."; + } + + std::transform(target.begin(), target.end(), summand.begin(), target.begin(), std::plus()); + } + + /*! + * Reduces the given source vector by selecting an element according to the given filter out of each row group. + * + * @param source The source vector which is to be reduced. + * @param target The target vector into which a single element from each row group is written. + * @param rowGrouping A vector that specifies the begin and end of each group of elements in the values vector. + * @param filter A function that compares two elements v1 and v2 according to some filter criterion. This function must + * return true iff v1 is supposed to be taken instead of v2. + * @param choices If non-null, this vector is used to store the choices made during the selection. + */ + template + void reduceVector(std::vector const& source, std::vector& target, std::vector const& rowGrouping, std::function filter, std::vector* choices = nullptr) { + uint_fast64_t currentSourceRow = 0; + uint_fast64_t currentTargetRow = -1; + uint_fast64_t currentLocalRow = 0; + for (auto it = source.cbegin(), ite = source.cend(); it != ite; ++it, ++currentSourceRow, ++currentLocalRow) { + // Check whether we have considered all source rows for the current target row. + if (rowGrouping[currentTargetRow + 1] <= currentSourceRow || currentSourceRow == 0) { + currentLocalRow = 0; + ++currentTargetRow; + target[currentTargetRow] = source[currentSourceRow]; + if (choices != nullptr) { + (*choices)[currentTargetRow] = 0; + } + continue; + } + + // We have to upate the value, so only overwrite the current value if the value passes the filter. + if (filter(*it, target[currentTargetRow])) { + target[currentTargetRow] = *it; + if (choices != nullptr) { + (*choices)[currentTargetRow] = currentLocalRow; + } + } + } + } + + /*! + * Reduces the given source vector by selecting the smallest element out of each row group. + * + * @param source The source vector which is to be reduced. + * @param target The target vector into which a single element from each row group is written. + * @param rowGrouping A vector that specifies the begin and end of each group of elements in the source vector. + * @param choices If non-null, this vector is used to store the choices made during the selection. + */ + template + void reduceVectorMin(std::vector const& source, std::vector& target, std::vector const& rowGrouping, std::vector* choices = nullptr) { + reduceVector(source, target, rowGrouping, std::less(), choices); + } + + /*! + * Reduces the given source vector by selecting the largest element out of each row group. + * + * @param source The source vector which is to be reduced. + * @param target The target vector into which a single element from each row group is written. + * @param rowGrouping A vector that specifies the begin and end of each group of elements in the source vector. + * @param choices If non-null, this vector is used to store the choices made during the selection. + */ + template + void reduceVectorMax(std::vector const& source, std::vector& target, std::vector const& rowGrouping, std::vector* choices = nullptr) { + reduceVector(source, target, rowGrouping, std::greater(), choices); + } + + /*! + * Compares the given elements and determines whether they are equal modulo the given precision. The provided flag + * additionaly specifies whether the error is computed in relative or absolute terms. + * + * @param val1 The first value to compare. + * @param val2 The second value to compare. + * @param precision The precision up to which the elements are compared. + * @param relativeError If set, the error is computed relative to the second value. + * @return True iff the elements are considered equal. + */ + template + bool equalModuloPrecision(T const& val1, T const& val2, T precision, bool relativeError = true) { + if (relativeError) { + if (std::abs(val1 - val2)/val2 > precision) return false; + } else { + if (std::abs(val1 - val2) > precision) return false; + } + return true; + } + + /*! + * Compares the two vectors and determines whether they are equal modulo the provided precision. Depending on whether the + * flag is set, the difference between the vectors is computed relative to the value or in absolute terms. + * + * @param vectorLeft The first vector of the comparison. + * @param vectorRight The second vector of the comparison. + * @param precision The precision up to which the vectors are to be checked for equality. + * @param relativeError If set, the difference between the vectors is computed relative to the value or in absolute terms. + */ + template + bool equalModuloPrecision(std::vector const& vectorLeft, std::vector const& vectorRight, T precision, bool relativeError) { + if (vectorLeft.size() != vectorRight.size()) { + LOG4CPLUS_ERROR(logger, "Lengths of vectors do not match, which makes comparison impossible."); + throw storm::exceptions::InvalidArgumentException() << "Lengths of vectors do not match, which makes comparison impossible."; + } + + for (uint_fast64_t i = 0; i < vectorLeft.size(); ++i) { + if (!equalModuloPrecision(vectorLeft[i], vectorRight[i], precision, relativeError)) { + return false; + } + } + + return true; + } + + /*! + * Compares the two vectors at the specified positions and determines whether they are equal modulo the provided + * precision. Depending on whether the flag is set, the difference between the vectors is computed relative to the value + * or in absolute terms. + * + * @param vectorLeft The first vector of the comparison. + * @param vectorRight The second vector of the comparison. + * @param precision The precision up to which the vectors are to be checked for equality. + * @param positions A vector representing a set of positions at which the vectors are compared. + * @param relativeError If set, the difference between the vectors is computed relative to the value or in absolute terms. + */ + template + bool equalModuloPrecision(std::vector const& vectorLeft, std::vector const& vectorRight, std::vector const& positions, T precision, bool relativeError) { + if (vectorLeft.size() != vectorRight.size()) { + LOG4CPLUS_ERROR(logger, "Lengths of vectors do not match, which makes comparison impossible."); + throw storm::exceptions::InvalidArgumentException() << "Lengths of vectors do not match, which makes comparison impossible."; + } + + for (uint_fast64_t position : positions) { + if (!equalModuloPrecision(vectorLeft[position], vectorRight[position], precision, relativeError)) { + return false; + } + } + + return true; + } + + /*! + * Takes the given offset vector and applies the given contraint. That is, it produces another offset vector that contains + * the relative offsets of the entries given by the constraint. + * + * @param offsetVector The offset vector to constrain. + * @param constraint The constraint to apply to the offset vector. + * @return An offset vector that contains all selected relative offsets. + */ + template + std::vector getConstrainedOffsetVector(std::vector const& offsetVector, storm::storage::BitVector const& constraint) { + // Reserve the known amount of slots for the resulting vector. + std::vector subVector(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. + subVector[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) { + subVector[currentIndexCount] = currentRowCount + offsetVector[index + 1] - offsetVector[index]; + currentRowCount += offsetVector[index + 1] - offsetVector[index]; + ++currentIndexCount; + } + + // Put a sentinel element at the end. + subVector[constraint.getNumberOfSetBits()] = currentRowCount; + + return subVector; + } + + } // namespace vector + } // namespace utility } // namespace storm #endif /* STORM_UTILITY_VECTOR_H_ */ diff --git a/test/functional/modelchecker/GmmxxMdpPrctlModelCheckerTest.cpp b/test/functional/modelchecker/GmmxxMdpPrctlModelCheckerTest.cpp index 7bd87856f..d7ac097b5 100644 --- a/test/functional/modelchecker/GmmxxMdpPrctlModelCheckerTest.cpp +++ b/test/functional/modelchecker/GmmxxMdpPrctlModelCheckerTest.cpp @@ -17,7 +17,7 @@ TEST(GmmxxMdpPrctlModelCheckerTest, Dice) { ASSERT_EQ(mdp->getNumberOfStates(), 169ull); ASSERT_EQ(mdp->getNumberOfTransitions(), 436ull); - storm::modelchecker::prctl::SparseMdpPrctlModelChecker mc(*mdp, new storm::solver::GmmxxNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker mc(*mdp, std::shared_ptr>(new storm::solver::GmmxxNondeterministicLinearEquationSolver())); storm::property::prctl::Ap* apFormula = new storm::property::prctl::Ap("two"); storm::property::prctl::Eventually* eventuallyFormula = new storm::property::prctl::Eventually(apFormula); @@ -105,7 +105,7 @@ TEST(GmmxxMdpPrctlModelCheckerTest, Dice) { std::shared_ptr> stateRewardMdp = stateRewardParser.getModel>(); - storm::modelchecker::prctl::SparseMdpPrctlModelChecker stateRewardModelChecker(*stateRewardMdp, new storm::solver::GmmxxNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker stateRewardModelChecker(*stateRewardMdp, std::shared_ptr>(new storm::solver::GmmxxNondeterministicLinearEquationSolver())); apFormula = new storm::property::prctl::Ap("done"); reachabilityRewardFormula = new storm::property::prctl::ReachabilityReward(apFormula); @@ -133,7 +133,7 @@ TEST(GmmxxMdpPrctlModelCheckerTest, Dice) { std::shared_ptr> stateAndTransitionRewardMdp = stateAndTransitionRewardParser.getModel>(); - storm::modelchecker::prctl::SparseMdpPrctlModelChecker stateAndTransitionRewardModelChecker(*stateAndTransitionRewardMdp, new storm::solver::GmmxxNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker stateAndTransitionRewardModelChecker(*stateAndTransitionRewardMdp, std::shared_ptr>(new storm::solver::GmmxxNondeterministicLinearEquationSolver())); apFormula = new storm::property::prctl::Ap("done"); reachabilityRewardFormula = new storm::property::prctl::ReachabilityReward(apFormula); @@ -167,7 +167,7 @@ TEST(GmmxxMdpPrctlModelCheckerTest, AsynchronousLeader) { ASSERT_EQ(mdp->getNumberOfStates(), 3172ull); ASSERT_EQ(mdp->getNumberOfTransitions(), 7144ull); - storm::modelchecker::prctl::SparseMdpPrctlModelChecker mc(*mdp, new storm::solver::GmmxxNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker mc(*mdp, std::shared_ptr>(new storm::solver::GmmxxNondeterministicLinearEquationSolver())); storm::property::prctl::Ap* apFormula = new storm::property::prctl::Ap("elected"); storm::property::prctl::Eventually* eventuallyFormula = new storm::property::prctl::Eventually(apFormula); diff --git a/test/functional/modelchecker/SparseMdpPrctlModelCheckerTest.cpp b/test/functional/modelchecker/SparseMdpPrctlModelCheckerTest.cpp index c07c583e8..29b74594d 100644 --- a/test/functional/modelchecker/SparseMdpPrctlModelCheckerTest.cpp +++ b/test/functional/modelchecker/SparseMdpPrctlModelCheckerTest.cpp @@ -16,7 +16,7 @@ TEST(SparseMdpPrctlModelCheckerTest, Dice) { ASSERT_EQ(mdp->getNumberOfStates(), 169ull); ASSERT_EQ(mdp->getNumberOfTransitions(), 436ull); - storm::modelchecker::prctl::SparseMdpPrctlModelChecker mc(*mdp, new storm::solver::AbstractNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker mc(*mdp, std::shared_ptr>(new storm::solver::AbstractNondeterministicLinearEquationSolver())); storm::property::prctl::Ap* apFormula = new storm::property::prctl::Ap("two"); storm::property::prctl::Eventually* eventuallyFormula = new storm::property::prctl::Eventually(apFormula); @@ -104,7 +104,7 @@ TEST(SparseMdpPrctlModelCheckerTest, Dice) { std::shared_ptr> stateRewardMdp = stateRewardParser.getModel>(); - storm::modelchecker::prctl::SparseMdpPrctlModelChecker stateRewardModelChecker(*stateRewardMdp, new storm::solver::AbstractNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker stateRewardModelChecker(*stateRewardMdp, std::shared_ptr>(new storm::solver::AbstractNondeterministicLinearEquationSolver())); apFormula = new storm::property::prctl::Ap("done"); reachabilityRewardFormula = new storm::property::prctl::ReachabilityReward(apFormula); @@ -132,7 +132,7 @@ TEST(SparseMdpPrctlModelCheckerTest, Dice) { std::shared_ptr> stateAndTransitionRewardMdp = stateAndTransitionRewardParser.getModel>(); - storm::modelchecker::prctl::SparseMdpPrctlModelChecker stateAndTransitionRewardModelChecker(*stateAndTransitionRewardMdp, new storm::solver::AbstractNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker stateAndTransitionRewardModelChecker(*stateAndTransitionRewardMdp, std::shared_ptr>(new storm::solver::AbstractNondeterministicLinearEquationSolver())); apFormula = new storm::property::prctl::Ap("done"); reachabilityRewardFormula = new storm::property::prctl::ReachabilityReward(apFormula); @@ -166,7 +166,7 @@ TEST(SparseMdpPrctlModelCheckerTest, AsynchronousLeader) { ASSERT_EQ(mdp->getNumberOfStates(), 3172ull); ASSERT_EQ(mdp->getNumberOfTransitions(), 7144ull); - storm::modelchecker::prctl::SparseMdpPrctlModelChecker mc(*mdp, new storm::solver::AbstractNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker mc(*mdp, std::shared_ptr>(new storm::solver::AbstractNondeterministicLinearEquationSolver())); storm::property::prctl::Ap* apFormula = new storm::property::prctl::Ap("elected"); storm::property::prctl::Eventually* eventuallyFormula = new storm::property::prctl::Eventually(apFormula); diff --git a/test/performance/modelchecker/GmmxxMdpPrctModelCheckerTest.cpp b/test/performance/modelchecker/GmmxxMdpPrctModelCheckerTest.cpp index 8d575caab..da34f7fec 100644 --- a/test/performance/modelchecker/GmmxxMdpPrctModelCheckerTest.cpp +++ b/test/performance/modelchecker/GmmxxMdpPrctModelCheckerTest.cpp @@ -17,7 +17,7 @@ TEST(GmmxxMdpPrctlModelCheckerTest, AsynchronousLeader) { ASSERT_EQ(mdp->getNumberOfStates(), 2095783ull); ASSERT_EQ(mdp->getNumberOfTransitions(), 7714385ull); - storm::modelchecker::prctl::SparseMdpPrctlModelChecker mc(*mdp, new storm::solver::GmmxxNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker mc(*mdp, std::shared_ptr>(new storm::solver::GmmxxNondeterministicLinearEquationSolver())); storm::property::prctl::Ap* apFormula = new storm::property::prctl::Ap("elected"); storm::property::prctl::Eventually* eventuallyFormula = new storm::property::prctl::Eventually(apFormula); @@ -104,7 +104,7 @@ TEST(GmmxxMdpPrctlModelCheckerTest, Consensus) { ASSERT_EQ(mdp->getNumberOfStates(), 63616ull); ASSERT_EQ(mdp->getNumberOfTransitions(), 213472ull); - storm::modelchecker::prctl::SparseMdpPrctlModelChecker mc(*mdp, new storm::solver::GmmxxNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker mc(*mdp, std::shared_ptr>(new storm::solver::GmmxxNondeterministicLinearEquationSolver())); storm::property::prctl::Ap* apFormula = new storm::property::prctl::Ap("finished"); storm::property::prctl::Eventually* eventuallyFormula = new storm::property::prctl::Eventually(apFormula); diff --git a/test/performance/modelchecker/SparseMdpPrctlModelCheckerTest.cpp b/test/performance/modelchecker/SparseMdpPrctlModelCheckerTest.cpp index 2a24197a1..a9e489fd1 100644 --- a/test/performance/modelchecker/SparseMdpPrctlModelCheckerTest.cpp +++ b/test/performance/modelchecker/SparseMdpPrctlModelCheckerTest.cpp @@ -17,7 +17,7 @@ TEST(SparseMdpPrctlModelCheckerTest, AsynchronousLeader) { ASSERT_EQ(mdp->getNumberOfStates(), 2095783ull); ASSERT_EQ(mdp->getNumberOfTransitions(), 7714385ull); - storm::modelchecker::prctl::SparseMdpPrctlModelChecker mc(*mdp, new storm::solver::AbstractNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker mc(*mdp, std::shared_ptr>(new storm::solver::AbstractNondeterministicLinearEquationSolver())); storm::property::prctl::Ap* apFormula = new storm::property::prctl::Ap("elected"); storm::property::prctl::Eventually* eventuallyFormula = new storm::property::prctl::Eventually(apFormula); @@ -106,7 +106,7 @@ TEST(SparseMdpPrctlModelCheckerTest, Consensus) { ASSERT_EQ(mdp->getNumberOfStates(), 63616ull); ASSERT_EQ(mdp->getNumberOfTransitions(), 213472ull); - storm::modelchecker::prctl::SparseMdpPrctlModelChecker mc(*mdp, new storm::solver::AbstractNondeterministicLinearEquationSolver()); + storm::modelchecker::prctl::SparseMdpPrctlModelChecker mc(*mdp, std::shared_ptr>(new storm::solver::AbstractNondeterministicLinearEquationSolver())); storm::property::prctl::Ap* apFormula = new storm::property::prctl::Ap("finished"); storm::property::prctl::Eventually* eventuallyFormula = new storm::property::prctl::Eventually(apFormula); From b3601782a968e01a12a15538dd8045727022cf82 Mon Sep 17 00:00:00 2001 From: dehnert Date: Mon, 2 Dec 2013 17:06:09 +0100 Subject: [PATCH 4/9] Added Lp Solver class for glpk and added it as an option in CMakeLists.txt. Former-commit-id: e5c5215a293fbecaf49442e5ca80c7905d5f7c12 --- CMakeLists.txt | 20 ++ .../SparseMarkovAutomatonCslModelChecker.h | 21 ++- ...ractNondeterministicLinearEquationSolver.h | 3 - src/solver/GlpkLpSolver.cpp | 172 ++++++++++++++++++ src/solver/GlpkLpSolver.h | 113 ++++++++++++ src/solver/GurobiLpSolver.cpp | 1 - src/storm.cpp | 4 +- src/utility/StormOptions.cpp | 13 +- src/utility/solver.cpp | 13 +- src/utility/solver.h | 3 +- storm-config.h.in | 3 + 11 files changed, 343 insertions(+), 23 deletions(-) create mode 100644 src/solver/GlpkLpSolver.cpp create mode 100644 src/solver/GlpkLpSolver.h diff --git a/CMakeLists.txt b/CMakeLists.txt index 89615eb3a..31d82172c 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -32,6 +32,7 @@ option(USE_INTELTBB "Sets whether the Intel TBB Extensions should be used." OFF) option(STORM_USE_COTIRE "Sets whether Cotire should be used (for building precompiled headers)." OFF) option(LINK_LIBCXXABI "Sets whether libc++abi should be linked." OFF) option(USE_LIBCXX "Sets whether the standard library is libc++." OFF) +option(ENABLE_GLPK "Sets whether StoRM is built with support for glpk." OFF) set(GUROBI_ROOT "" CACHE STRING "The root directory of Gurobi (if available).") set(Z3_ROOT "" CACHE STRING "The root directory of Z3 (if available).") set(ADDITIONAL_INCLUDE_DIRS "" CACHE STRING "Additional directories added to the include directories.") @@ -184,6 +185,13 @@ else() set(STORM_CPP_GUROBI_DEF "undef") endif() +# glpk defines +if (ENABLE_GLPK) + set(STORM_CPP_GLPK_DEF "define") +else() + set(STORM_CPP_GLPK_DEF "undef") +endif() + # Z3 Defines if (ENABLE_Z3) set(STORM_CPP_Z3_DEF "define") @@ -359,6 +367,18 @@ if (ENABLE_GUROBI) target_link_libraries(storm-performance-tests "gurobi56") endif(ENABLE_GUROBI) +############################################################# +## +## glpk (optional) +## +############################################################# +if (ENABLE_GLPK) + message (STATUS "StoRM - Linking with glpk") + target_link_libraries(storm "glpk") + target_link_libraries(storm-functional-tests "glpk") + target_link_libraries(storm-performance-tests "glpk") +endif(ENABLE_GLPK) + ############################################################# ## ## Z3 (optional) diff --git a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h index 123503cac..c040b92ed 100644 --- a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h +++ b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h @@ -274,9 +274,9 @@ namespace storm { } // Compute the LRA value for the current MEC. - lraValuesForEndComponents.push_back(this->computeLraForMaximalEndComponent(min, transitionMatrix, nondeterministicChoiceIndices, this->getModel().getMarkovianStates(), this->getModel().getExitRates(), goalStates, mec)); + lraValuesForEndComponents.push_back(this->computeLraForMaximalEndComponent(min, transitionMatrix, nondeterministicChoiceIndices, this->getModel().getMarkovianStates(), this->getModel().getExitRates(), goalStates, mec, currentMecIndex)); } - + // For fast transition rewriting, we build some auxiliary data structures. storm::storage::BitVector statesNotContainedInAnyMec = ~statesInMecs; uint_fast64_t firstAuxiliaryStateIndex = statesNotContainedInAnyMec.getNumberOfSetBits(); @@ -388,23 +388,23 @@ namespace storm { // Finalize the matrix and solve the corresponding system of equations. sspMatrix.finalize(); std::vector x(numberOfStatesNotInMecs + mecDecomposition.size()); - std::shared_ptr> nondeterministiclinearEquationSolver = storm::utility::solver::getNondeterministicLinearEquationSolver(); - nondeterministiclinearEquationSolver->solveEquationSystem(min, sspMatrix, x, b, sspNondeterministicChoiceIndices); + nondeterministicLinearEquationSolver->solveEquationSystem(min, sspMatrix, x, b, sspNondeterministicChoiceIndices); // Prepare result vector. std::vector result(this->getModel().getNumberOfStates()); + std::cout << "res " << result << std::endl; + // Set the values for states not contained in MECs. - uint_fast64_t stateIndex = 0; - for (auto state : statesNotContainedInAnyMec) { - result[state] = x[stateIndex]; - ++stateIndex; - } + std::cout << x << std::endl; + storm::utility::vector::setVectorValues(result, statesNotContainedInAnyMec, x); + std::cout << "res " << result << std::endl; // Set the values for all states in MECs. for (auto state : statesInMecs) { result[state] = lraValuesForEndComponents[stateToMecIndexMap[state]]; } + std::cout << "res " << result << std::endl; return result; } @@ -428,9 +428,10 @@ namespace storm { * @param exitRates A vector with exit rates for all states. Exit rates of probabilistic states are assumed to be zero. * @param goalStates A bit vector indicating which states are to be considered as goal states. * @param mec The maximal end component to consider for computing the long-run average. + * @param mecIndex The index of the MEC. * @return The long-run average of being in a goal state for the given MEC. */ - static ValueType computeLraForMaximalEndComponent(bool min, storm::storage::SparseMatrix const& transitionMatrix, std::vector const& nondeterministicChoiceIndices, storm::storage::BitVector const& markovianStates, std::vector const& exitRates, storm::storage::BitVector const& goalStates, storm::storage::MaximalEndComponent const& mec) { + static ValueType computeLraForMaximalEndComponent(bool min, storm::storage::SparseMatrix const& transitionMatrix, std::vector const& nondeterministicChoiceIndices, storm::storage::BitVector const& markovianStates, std::vector const& exitRates, storm::storage::BitVector const& goalStates, storm::storage::MaximalEndComponent const& mec, uint_fast64_t mecIndex = 0) { std::shared_ptr solver = storm::utility::solver::getLpSolver("LRA for MEC"); solver->setModelSense(min ? storm::solver::LpSolver::MAXIMIZE : storm::solver::LpSolver::MINIMIZE); diff --git a/src/solver/AbstractNondeterministicLinearEquationSolver.h b/src/solver/AbstractNondeterministicLinearEquationSolver.h index 4ab08cf90..2929bb695 100644 --- a/src/solver/AbstractNondeterministicLinearEquationSolver.h +++ b/src/solver/AbstractNondeterministicLinearEquationSolver.h @@ -83,9 +83,6 @@ namespace storm { // LOG4CPLUS_INFO(logger, "Starting iterative solver."); // Set up the environment for the power method. -// auto startTime = std::chrono::high_resolution_clock::now(); -// std::chrono::nanoseconds totalTime(0); - bool multiplyResultMemoryProvided = true; if (multiplyResult == nullptr) { multiplyResult = new std::vector(A.getRowCount()); diff --git a/src/solver/GlpkLpSolver.cpp b/src/solver/GlpkLpSolver.cpp new file mode 100644 index 000000000..0d51fb590 --- /dev/null +++ b/src/solver/GlpkLpSolver.cpp @@ -0,0 +1,172 @@ +#include "src/solver/GlpkLpSolver.h" + +#ifdef STORM_HAVE_GLPK + +#include + +#include "src/exceptions/InvalidStateException.h" +#include "src/settings/Settings.h" + +#include "log4cplus/logger.h" +#include "log4cplus/loggingmacros.h" + +extern log4cplus::Logger logger; + +namespace storm { + namespace solver { + GlpkLpSolver::GlpkLpSolver(std::string const& name, ModelSense const& modelSense) : LpSolver(modelSense), lp(nullptr), nextVariableIndex(1), nextConstraintIndex(1), isOptimal(false), rowIndices(), columnIndices(), coefficientValues() { + // Create the LP problem for glpk. + lp = glp_create_prob(); + + // Set its name and model sense. + glp_set_prob_name(lp, name.c_str()); + this->setModelSense(modelSense); + + // Because glpk uses 1-based indexing (wtf!?), we need to put dummy elements into the matrix vectors. + rowIndices.push_back(0); + columnIndices.push_back(0); + coefficientValues.push_back(0); + } + + GlpkLpSolver::GlpkLpSolver(std::string const& name) : GlpkLpSolver(name, MINIMIZE) { + // Intentionally left empty. + } + + GlpkLpSolver::~GlpkLpSolver() { + glp_delete_prob(this->lp); + glp_free_env(); + } + + uint_fast64_t GlpkLpSolver::createContinuousVariable(std::string const& name, VariableType const& variableType, double lowerBound, double upperBound, double objectiveFunctionCoefficient) { + glp_add_cols(this->lp, 1); + glp_set_col_name(this->lp, nextVariableIndex, name.c_str()); + switch (variableType) { + case LpSolver::BOUNDED: + glp_set_col_bnds(lp, nextVariableIndex, GLP_DB, lowerBound, upperBound); + break; + case LpSolver::UNBOUNDED: + glp_set_col_bnds(lp, nextVariableIndex, GLP_FR, 0, 0); + break; + case LpSolver::UPPER_BOUND: + glp_set_col_bnds(lp, nextVariableIndex, GLP_UP, 0, upperBound); + break; + case LpSolver::LOWER_BOUND: + glp_set_col_bnds(lp, nextVariableIndex, GLP_LO, lowerBound, 0); + break; + } + glp_set_col_kind(this->lp, nextVariableIndex, GLP_CV); + glp_set_obj_coef(this->lp, nextVariableIndex, objectiveFunctionCoefficient); + ++nextVariableIndex; + + this->isOptimal = false; + + return nextVariableIndex - 1; + } + + uint_fast64_t GlpkLpSolver::createIntegerVariable(std::string const& name, VariableType const& variableType, double lowerBound, double upperBound, double objectiveFunctionCoefficient) { + uint_fast64_t index = this->createContinuousVariable(name, variableType, lowerBound, upperBound, objectiveFunctionCoefficient); + glp_set_col_kind(this->lp, index, GLP_IV); + return index; + } + + uint_fast64_t GlpkLpSolver::createBinaryVariable(std::string const& name, double objectiveFunctionCoefficient) { + uint_fast64_t index = this->createContinuousVariable(name, UNBOUNDED, 0, 1, objectiveFunctionCoefficient); + glp_set_col_kind(this->lp, index, GLP_BV); + return index; + } + + void GlpkLpSolver::addConstraint(std::string const& name, std::vector const& variables, std::vector const& coefficients, BoundType const& boundType, double rightHandSideValue) { + if (variables.size() != coefficients.size()) { + LOG4CPLUS_ERROR(logger, "Sizes of variable indices vector and coefficients vector do not match."); + throw storm::exceptions::InvalidStateException() << "Sizes of variable indices vector and coefficients vector do not match."; + } + + // Add the row that will represent this constraint. + glp_add_rows(this->lp, 1); + glp_set_row_name(this->lp, nextConstraintIndex, name.c_str()); + + // Determine the type of the constraint and add it properly. + bool isUpperBound = boundType == LESS || boundType == LESS_EQUAL; + bool isStrict = boundType == LESS || boundType == GREATER; + switch (boundType) { + case LESS: + glp_set_row_bnds(this->lp, nextConstraintIndex, GLP_UP, 0, rightHandSideValue - storm::settings::Settings::getInstance()->getOptionByLongName("precision").getArgument(0).getValueAsDouble()); + break; + case LESS_EQUAL: + glp_set_row_bnds(this->lp, nextConstraintIndex, GLP_UP, 0, rightHandSideValue); + break; + case GREATER: + glp_set_row_bnds(this->lp, nextConstraintIndex, GLP_LO, rightHandSideValue + storm::settings::Settings::getInstance()->getOptionByLongName("precision").getArgument(0).getValueAsDouble(), 0); + break; + case GREATER_EQUAL: + glp_set_row_bnds(this->lp, nextConstraintIndex, GLP_LO, rightHandSideValue, 0); + break; + case EQUAL: + glp_set_row_bnds(this->lp, nextConstraintIndex, GLP_FX, rightHandSideValue, rightHandSideValue); + break; + } + + // Record the variables and coefficients in the coefficient matrix. + rowIndices.insert(rowIndices.end(), variables.size(), nextConstraintIndex); + columnIndices.insert(columnIndices.end(), variables.begin(), variables.end()); + coefficientValues.insert(coefficientValues.end(), coefficients.begin(), coefficients.end()); + + ++nextConstraintIndex; + this->isOptimal = false; + } + + void GlpkLpSolver::setModelSense(ModelSense const& newModelSense) { + glp_set_obj_dir(this->lp, newModelSense == MINIMIZE ? GLP_MIN : GLP_MAX); + this->isOptimal = false; + } + + void GlpkLpSolver::optimize() const { + glp_load_matrix(this->lp, rowIndices.size() - 1, rowIndices.data(), columnIndices.data(), coefficientValues.data()); + int error = glp_simplex(this->lp, nullptr); + + if (error != 0) { + LOG4CPLUS_ERROR(logger, "Unable to optimize glpk model (" << error << ")."); + throw storm::exceptions::InvalidStateException() << "Unable to optimize glpk model (" << error << ")."; + } + + this->isOptimal = true; + } + + int_fast64_t GlpkLpSolver::getIntegerValue(uint_fast64_t variableIndex) const { + double value = glp_get_col_prim(this->lp, static_cast(variableIndex)); + + if (std::abs(value - static_cast(value)) <= storm::settings::Settings::getInstance()->getOptionByLongName("precision").getArgument(0).getValueAsDouble()) { + // Nothing to do in this case. + } else if (std::abs(value) > storm::settings::Settings::getInstance()->getOptionByLongName("precision").getArgument(0).getValueAsDouble()) { + LOG4CPLUS_ERROR(logger, "Illegal value for integer variable in glpk solution (" << value << ")."); + throw storm::exceptions::InvalidStateException() << "Illegal value for integer variable in glpk solution (" << value << ")."; + } + + return static_cast(value); + } + + bool GlpkLpSolver::getBinaryValue(uint_fast64_t variableIndex) const { + double value = glp_get_col_prim(this->lp, static_cast(variableIndex)); + + if (std::abs(value - 1) <= storm::settings::Settings::getInstance()->getOptionByLongName("precision").getArgument(0).getValueAsDouble()) { + // Nothing to do in this case. + } else if (std::abs(value) > storm::settings::Settings::getInstance()->getOptionByLongName("precision").getArgument(0).getValueAsDouble()) { + LOG4CPLUS_ERROR(logger, "Illegal value for binary variable in Gurobi solution (" << value << ")."); + throw storm::exceptions::InvalidStateException() << "Illegal value for binary variable in Gurobi solution (" << value << ")."; + } + + return static_cast(value); + } + + double GlpkLpSolver::getContinuousValue(uint_fast64_t variableIndex) const { + return glp_get_col_prim(this->lp, static_cast(variableIndex)); + } + + void GlpkLpSolver::writeModelToFile(std::string const& filename) const { + glp_load_matrix(this->lp, rowIndices.size() - 1, rowIndices.data(), columnIndices.data(), coefficientValues.data()); + glp_write_lp(this->lp, 0, filename.c_str()); + } + } +} + +#endif \ No newline at end of file diff --git a/src/solver/GlpkLpSolver.h b/src/solver/GlpkLpSolver.h new file mode 100644 index 000000000..5576b1111 --- /dev/null +++ b/src/solver/GlpkLpSolver.h @@ -0,0 +1,113 @@ +#ifndef STORM_SOLVER_GLPKLPSOLVER_H_ +#define STORM_SOLVER_GLPKLPSOLVER_H_ + +#include "src/solver/LpSolver.h" +#include "src/exceptions/NotImplementedException.h" + +// To detect whether the usage of glpk is possible, this include is neccessary. +#include "storm-config.h" + +#ifdef STORM_HAVE_GLPK +#include +#endif + +namespace storm { + namespace solver { +#ifdef STORM_HAVE_GLPK + class GlpkLpSolver : public LpSolver { + public: + GlpkLpSolver(std::string const& name, ModelSense const& modelSense); + GlpkLpSolver(std::string const& name); + virtual ~GlpkLpSolver(); + + virtual uint_fast64_t createContinuousVariable(std::string const& name, VariableType const& variableType, double lowerBound, double upperBound, double objectiveFunctionCoefficient) override; + virtual uint_fast64_t createIntegerVariable(std::string const& name, VariableType const& variableType, double lowerBound, double upperBound, double objectiveFunctionCoefficient) override; + virtual uint_fast64_t createBinaryVariable(std::string const& name, double objectiveFunctionCoefficient) override; + + virtual void addConstraint(std::string const& name, std::vector const& variables, std::vector const& coefficients, BoundType const& boundType, double rightHandSideValue) override; + + virtual void setModelSense(ModelSense const& newModelSense) override; + virtual void optimize() const override; + + virtual int_fast64_t getIntegerValue(uint_fast64_t variableIndex) const override; + virtual bool getBinaryValue(uint_fast64_t variableIndex) const override; + virtual double getContinuousValue(uint_fast64_t variableIndex) const override; + + virtual void writeModelToFile(std::string const& filename) const override; + + private: + // The glpk LP problem. + glp_prob* lp; + + // A counter that keeps track of the next free variable index. + uint_fast64_t nextVariableIndex; + + // A counter that keeps track of the next free constraint index. + uint_fast64_t nextConstraintIndex; + + // A flag that stores whether the model was optimized properly. + mutable bool isOptimal; + + // The arrays that store the coefficient matrix of the problem. + std::vector rowIndices; + std::vector columnIndices; + std::vector coefficientValues; + }; +#else + // If glpk is not available, we provide a stub implementation that emits an error if any of its methods is called. + class GlpkLpSolver : public LpSolver { + public: + + GlpkLpSolver(std::string const& name) : LpSolver(MINIMIZE) { + throw storm::exceptions::NotImplementedException() << "This version of StoRM was compiled without support for glpk. Yet, a method was called that requires this support. Please choose a version of support with glpk support."; + } + + virtual GlpkLpSolver() { + throw storm::exceptions::NotImplementedException() << "This version of StoRM was compiled without support for glpk. Yet, a method was called that requires this support. Please choose a version of support with glpk support."; + } + + virtual uint_fast64_t createContinuousVariable(std::string const& name, VariableType const& variableType, double lowerBound, double upperBound, double objectiveFunctionCoefficient) override { + throw storm::exceptions::NotImplementedException() << "This version of StoRM was compiled without support for glpk. Yet, a method was called that requires this support. Please choose a version of support with glpk support."; + } + + virtual uint_fast64_t createIntegerVariable(std::string const& name, VariableType const& variableType, double lowerBound, double upperBound, double objectiveFunctionCoefficient) override { + throw storm::exceptions::NotImplementedException() << "This version of StoRM was compiled without support for glpk. Yet, a method was called that requires this support. Please choose a version of support with glpk support."; + } + + virtual uint_fast64_t createBinaryVariable(std::string const& name, double objectiveFunctionCoefficient) override { + throw storm::exceptions::NotImplementedException() << "This version of StoRM was compiled without support for glpk. Yet, a method was called that requires this support. Please choose a version of support with glpk support."; + } + + virtual void addConstraint(std::string const& name, std::vector const& variables, std::vector const& coefficients, BoundType const& boundType, double rightHandSideValue) override { + throw storm::exceptions::NotImplementedException() << "This version of StoRM was compiled without support for glpk. Yet, a method was called that requires this support. Please choose a version of support with glpk support."; + } + + virtual void setModelSense(ModelSense const& newModelSense) { + throw storm::exceptions::NotImplementedException() << "This version of StoRM was compiled without support for glpk. Yet, a method was called that requires this support. Please choose a version of support with glpk support."; + } + + virtual void optimize() const override { + throw storm::exceptions::NotImplementedException() << "This version of StoRM was compiled without support for glpk. Yet, a method was called that requires this support. Please choose a version of support with glpk support."; + } + + virtual int_fast64_t getIntegerValue(uint_fast64_t variableIndex) const override { + throw storm::exceptions::NotImplementedException() << "This version of StoRM was compiled without support for glpk. Yet, a method was called that requires this support. Please choose a version of support with glpk support."; + } + + virtual bool getBinaryValue(uint_fast64_t variableIndex) const override { + throw storm::exceptions::NotImplementedException() << "This version of StoRM was compiled without support for glpk. Yet, a method was called that requires this support. Please choose a version of support with glpk support."; + } + + virtual double getContinuousValue(uint_fast64_t variableIndex) const override { + throw storm::exceptions::NotImplementedException() << "This version of StoRM was compiled without support for glpk. Yet, a method was called that requires this support. Please choose a version of support with glpk support."; + } + + virtual void writeModelToFile(std::string const& filename) const override { + throw storm::exceptions::NotImplementedException() << "This version of StoRM was compiled without support for glpk. Yet, a method was called that requires this support. Please choose a version of support with glpk support."; + } + }; +#endif + } +} + +#endif /* STORM_SOLVER_GLPKLPSOLVER_H_ */ diff --git a/src/solver/GurobiLpSolver.cpp b/src/solver/GurobiLpSolver.cpp index 6ef9b130d..5b143fe9c 100644 --- a/src/solver/GurobiLpSolver.cpp +++ b/src/solver/GurobiLpSolver.cpp @@ -233,7 +233,6 @@ namespace storm { throw storm::exceptions::InvalidStateException() << "Unable to get Gurobi solution (" << GRBgeterrormsg(env) << ")."; } - bool returnValue = false; if (std::abs(value - 1) <= storm::settings::Settings::getInstance()->getOptionByLongName("precision").getArgument(0).getValueAsDouble()) { // Nothing to do in this case. } else if (std::abs(value) > storm::settings::Settings::getInstance()->getOptionByLongName("precision").getArgument(0).getValueAsDouble()) { diff --git a/src/storm.cpp b/src/storm.cpp index 6aee44050..0babe7612 100644 --- a/src/storm.cpp +++ b/src/storm.cpp @@ -465,8 +465,8 @@ int main(const int argc, const char* argv[]) { storm::modelchecker::csl::SparseMarkovAutomatonCslModelChecker mc(*markovAutomaton, std::shared_ptr>(new storm::solver::AbstractNondeterministicLinearEquationSolver())); std::cout << mc.checkExpectedTime(true, markovAutomaton->getLabeledStates("goal")) << std::endl; std::cout << mc.checkLongRunAverage(true, markovAutomaton->getLabeledStates("goal")) << std::endl; - std::cout << mc.checkTimeBoundedEventually(true, markovAutomaton->getLabeledStates("goal"), 0, 1) << std::endl; - std::cout << mc.checkTimeBoundedEventually(true, markovAutomaton->getLabeledStates("goal"), 1, 2) << std::endl; + // std::cout << mc.checkTimeBoundedEventually(true, markovAutomaton->getLabeledStates("goal"), 0, 1) << std::endl; + // std::cout << mc.checkTimeBoundedEventually(true, markovAutomaton->getLabeledStates("goal"), 1, 2) << std::endl; break; } case storm::models::Unknown: diff --git a/src/utility/StormOptions.cpp b/src/utility/StormOptions.cpp index b8985c9c9..39e57a6b9 100644 --- a/src/utility/StormOptions.cpp +++ b/src/utility/StormOptions.cpp @@ -34,10 +34,15 @@ bool storm::utility::StormOptions::optionsRegistered = storm::settings::Settings settings->addOption(storm::settings::OptionBuilder("StoRM Main", "fixDeadlocks", "", "If the model contains deadlock states, setting this option will insert self-loops for these states.").build()); - std::vector matrixLibrarys; - matrixLibrarys.push_back("gmm++"); - matrixLibrarys.push_back("native"); - settings->addOption(storm::settings::OptionBuilder("StoRM Main", "matrixLibrary", "m", "Sets which matrix library is preferred for numerical operations.").addArgument(storm::settings::ArgumentBuilder::createStringArgument("matrixLibraryName", "The name of a matrix library. Valid values are gmm++ and native.").addValidationFunctionString(storm::settings::ArgumentValidators::stringInListValidator(matrixLibrarys)).setDefaultValueString("gmm++").build()).build()); + std::vector matrixLibraries; + matrixLibraries.push_back("gmm++"); + matrixLibraries.push_back("native"); + settings->addOption(storm::settings::OptionBuilder("StoRM Main", "matrixLibrary", "m", "Sets which matrix library is preferred for numerical operations.").addArgument(storm::settings::ArgumentBuilder::createStringArgument("matrixLibraryName", "The name of a matrix library. Valid values are gmm++ and native.").addValidationFunctionString(storm::settings::ArgumentValidators::stringInListValidator(matrixLibraries)).setDefaultValueString("gmm++").build()).build()); + std::vector lpSolvers; + lpSolvers.push_back("gurobi"); + lpSolvers.push_back("glpk"); + settings->addOption(storm::settings::OptionBuilder("StoRM Main", "lpSolver", "", "Sets which LP solver is preferred.").addArgument(storm::settings::ArgumentBuilder::createStringArgument("LP solver name", "The name of an available LP solver. Valid values are gurobi and glpk.").addValidationFunctionString(storm::settings::ArgumentValidators::stringInListValidator(lpSolvers)).setDefaultValueString("glpk").build()).build()); + return true; }); diff --git a/src/utility/solver.cpp b/src/utility/solver.cpp index 16578aaf8..b6036ae58 100644 --- a/src/utility/solver.cpp +++ b/src/utility/solver.cpp @@ -1,10 +1,21 @@ #include "src/utility/solver.h" +#include "src/solver/GmmxxNondeterministicLinearEquationSolver.h" +#include "src/solver/GurobiLpSolver.h" +#include "src/solver/GlpkLpSolver.h" + namespace storm { namespace utility { namespace solver { std::shared_ptr getLpSolver(std::string const& name) { - return std::shared_ptr(new storm::solver::GurobiLpSolver(name)); + std::string const& lpSolver = storm::settings::Settings::getInstance()->getOptionByLongName("lpSolver").getArgument(0).getValueAsString(); + if (lpSolver == "gurobi") { + return std::shared_ptr(new storm::solver::GurobiLpSolver(name)); + } else if (lpSolver == "glpk") { + return std::shared_ptr(new storm::solver::GlpkLpSolver(name)); + } + + throw storm::exceptions::InvalidSettingsException() << "No suitable LP-solver selected."; } template diff --git a/src/utility/solver.h b/src/utility/solver.h index 165ba9c00..0122a49f3 100644 --- a/src/utility/solver.h +++ b/src/utility/solver.h @@ -2,8 +2,7 @@ #define STORM_UTILITY_SOLVER_H_ #include "src/solver/AbstractNondeterministicLinearEquationSolver.h" -#include "src/solver/GmmxxNondeterministicLinearEquationSolver.h" -#include "src/solver/GurobiLpSolver.h" +#include "src/solver/LpSolver.h" #include "src/exceptions/InvalidSettingsException.h" diff --git a/storm-config.h.in b/storm-config.h.in index 08bf0ca38..d44ae122c 100644 --- a/storm-config.h.in +++ b/storm-config.h.in @@ -19,6 +19,9 @@ // Whether Gurobi is available and to be used (define/undef) #@STORM_CPP_GUROBI_DEF@ STORM_HAVE_GUROBI +// Whether GLPK is available and to be used (define/undef) +#@STORM_CPP_GLPK_DEF@ STORM_HAVE_GLPK + // Whether Z3 is available and to be used (define/undef) #@STORM_CPP_Z3_DEF@ STORM_HAVE_Z3 From fde78ad7597057ba2c82ea59f2aa7c43a9efb3a5 Mon Sep 17 00:00:00 2001 From: dehnert Date: Mon, 2 Dec 2013 17:25:48 +0100 Subject: [PATCH 5/9] Bugfix for matrix creation in LRA computation. Former-commit-id: cb4c9cb7286048847fb3d7ebd1cae24c9cd61228 --- .../csl/SparseMarkovAutomatonCslModelChecker.h | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h index c040b92ed..8077086bb 100644 --- a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h +++ b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h @@ -312,7 +312,7 @@ namespace storm { for (auto element : row) { if (statesNotContainedInAnyMec.get(element.column())) { // If the target state is not contained in an MEC, we can copy over the entry. - sspMatrix.insertNextValue(currentChoice, statesNotInMecsBeforeIndex[element.column()], element.value()); + sspMatrix.insertNextValue(currentChoice, statesNotInMecsBeforeIndex[element.column()], element.value(), true); } else { // If the target state is contained in MEC i, we need to add the probability to the corresponding field in the vector // so that we are able to write the cumulative probability to the MEC into the matrix. @@ -323,7 +323,7 @@ namespace storm { // Now insert all (cumulative) probability values that target an MEC. for (uint_fast64_t mecIndex = 0; mecIndex < auxiliaryStateToProbabilityMap.size(); ++mecIndex) { if (auxiliaryStateToProbabilityMap[mecIndex] != 0) { - sspMatrix.insertNextValue(currentChoice, firstAuxiliaryStateIndex + mecIndex, auxiliaryStateToProbabilityMap[mecIndex]); + sspMatrix.insertNextValue(currentChoice, firstAuxiliaryStateIndex + mecIndex, auxiliaryStateToProbabilityMap[mecIndex], true); } } } @@ -349,7 +349,7 @@ namespace storm { for (auto element : row) { if (statesNotContainedInAnyMec.get(element.column())) { // If the target state is not contained in an MEC, we can copy over the entry. - sspMatrix.insertNextValue(currentChoice, statesNotInMecsBeforeIndex[element.column()], element.value()); + sspMatrix.insertNextValue(currentChoice, statesNotInMecsBeforeIndex[element.column()], element.value(), true); } else { // If the target state is contained in MEC i, we need to add the probability to the corresponding field in the vector // so that we are able to write the cumulative probability to the MEC into the matrix. @@ -366,7 +366,7 @@ namespace storm { b.back() += auxiliaryStateToProbabilityMap[mecIndex] * lraValuesForEndComponents[mecIndex]; } else { // Otherwise, we add a transition to the auxiliary state that is associated with the target MEC. - sspMatrix.insertNextValue(currentChoice, firstAuxiliaryStateIndex + targetMecIndex, auxiliaryStateToProbabilityMap[targetMecIndex]); + sspMatrix.insertNextValue(currentChoice, firstAuxiliaryStateIndex + targetMecIndex, auxiliaryStateToProbabilityMap[targetMecIndex], true); } } } From ece4085a61d39821fcd94e80f16dd6a29a0663fe Mon Sep 17 00:00:00 2001 From: dehnert Date: Mon, 2 Dec 2013 17:28:18 +0100 Subject: [PATCH 6/9] Another bugfix for matrix creation during LRA computation. Former-commit-id: c3325b891308dffd50e9c5bc474b9bebeafe2710 --- src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h index 8077086bb..378e8803a 100644 --- a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h +++ b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h @@ -377,7 +377,7 @@ namespace storm { } // For each auxiliary state, there is the option to achieve the reward value of the LRA associated with the MEC. - sspMatrix.insertEmptyRow(); + sspMatrix.insertEmptyRow(true); ++currentChoice; b.push_back(lraValuesForEndComponents[mecIndex]); } From 3dab26463d931aca0b3eb335f76f2e2704e1a917 Mon Sep 17 00:00:00 2001 From: dehnert Date: Mon, 2 Dec 2013 17:48:14 +0100 Subject: [PATCH 7/9] Introduced precision for digitization-based techniques as a new parameter. Former-commit-id: e9c57f821b5176cef3952c812292e6c5e256dc93 --- .../csl/SparseMarkovAutomatonCslModelChecker.cpp | 8 ++++++++ .../csl/SparseMarkovAutomatonCslModelChecker.h | 4 +--- src/storm.cpp | 10 ++++++---- 3 files changed, 15 insertions(+), 7 deletions(-) create mode 100644 src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.cpp diff --git a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.cpp b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.cpp new file mode 100644 index 000000000..82590e31e --- /dev/null +++ b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.cpp @@ -0,0 +1,8 @@ +#include "src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h" + +bool SparseMarkovAutomatonCslModelCheckerOptionsRegistered = storm::settings::Settings::registerNewModule([] (storm::settings::Settings* instance) -> bool { + + instance->addOption(storm::settings::OptionBuilder("GmmxxLinearEquationSolver", "digiprecision", "", "Precision used for iterative solving of linear equation systems").addArgument(storm::settings::ArgumentBuilder::createDoubleArgument("precision value", "Precision").setDefaultValueDouble(1e-4).addValidationFunctionDouble(storm::settings::ArgumentValidators::doubleRangeValidatorExcluding(0.0, 1.0)).build()).build()); + + return true; +}); \ No newline at end of file diff --git a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h index 378e8803a..80943c8e0 100644 --- a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h +++ b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h @@ -179,7 +179,7 @@ namespace storm { // (1) Compute the accuracy we need to achieve the required error bound. ValueType maxExitRate = this->getModel().getMaximalExitRate(); - ValueType delta = (2 * storm::settings::Settings::getInstance()->getOptionByLongName("precision").getArgument(0).getValueAsDouble()) / (upperBound * maxExitRate * maxExitRate); + ValueType delta = (2 * storm::settings::Settings::getInstance()->getOptionByLongName("digiprecision").getArgument(0).getValueAsDouble()) / (upperBound * maxExitRate * maxExitRate); // (2) Compute the number of steps we need to make for the interval. uint_fast64_t numberOfSteps = static_cast(std::ceil((upperBound - lowerBound) / delta)); @@ -203,10 +203,8 @@ namespace storm { // Create the starting value vectors for the next value iteration based on the results of the previous one. storm::utility::vector::setVectorValues(vAllProbabilistic, ~markovianStates % goalStates, storm::utility::constGetOne()); storm::utility::vector::setVectorValues(vAllProbabilistic, ~markovianStates % ~goalStates, vProbabilistic); - std::cout << vAllProbabilistic << std::endl; storm::utility::vector::setVectorValues(vAllMarkovian, markovianStates % goalStates, storm::utility::constGetOne()); storm::utility::vector::setVectorValues(vAllMarkovian, markovianStates % ~goalStates, vMarkovian); - std::cout << vAllMarkovian << std::endl; // Compute the number of steps to reach the target interval. numberOfSteps = static_cast(std::ceil(lowerBound / delta)); diff --git a/src/storm.cpp b/src/storm.cpp index 0babe7612..0ffd18d92 100644 --- a/src/storm.cpp +++ b/src/storm.cpp @@ -463,10 +463,12 @@ int main(const int argc, const char* argv[]) { std::shared_ptr> markovAutomaton = parser.getModel>(); markovAutomaton->close(); storm::modelchecker::csl::SparseMarkovAutomatonCslModelChecker mc(*markovAutomaton, std::shared_ptr>(new storm::solver::AbstractNondeterministicLinearEquationSolver())); - std::cout << mc.checkExpectedTime(true, markovAutomaton->getLabeledStates("goal")) << std::endl; - std::cout << mc.checkLongRunAverage(true, markovAutomaton->getLabeledStates("goal")) << std::endl; - // std::cout << mc.checkTimeBoundedEventually(true, markovAutomaton->getLabeledStates("goal"), 0, 1) << std::endl; - // std::cout << mc.checkTimeBoundedEventually(true, markovAutomaton->getLabeledStates("goal"), 1, 2) << std::endl; +// std::cout << mc.checkExpectedTime(true, markovAutomaton->getLabeledStates("goal")) << std::endl; +// std::cout << mc.checkExpectedTime(false, markovAutomaton->getLabeledStates("goal")) << std::endl; +// std::cout << mc.checkLongRunAverage(true, markovAutomaton->getLabeledStates("goal")) << std::endl; +// std::cout << mc.checkLongRunAverage(false, markovAutomaton->getLabeledStates("goal")) << std::endl; +// std::cout << mc.checkTimeBoundedEventually(true, markovAutomaton->getLabeledStates("goal"), 0, 1) << std::endl; + std::cout << mc.checkTimeBoundedEventually(true, markovAutomaton->getLabeledStates("goal"), 1, 2) << std::endl; break; } case storm::models::Unknown: From 344e1b6dd30afb2d2fc086a749f70986825bf96c Mon Sep 17 00:00:00 2001 From: dehnert Date: Tue, 3 Dec 2013 17:30:02 +0100 Subject: [PATCH 8/9] Enabled checking of some untimed properties on Markov automata. Former-commit-id: e71aa66c62ef8da141715facca79d0895b52e820 --- .../MILPMinimalLabelSetGenerator.h | 5 +- .../PathBasedSubsystemGenerator.h | 2 +- .../SMTMinimalCommandSetGenerator.h | 6 +- .../SparseMarkovAutomatonCslModelChecker.h | 21 ++-- .../prctl/SparseMdpPrctlModelChecker.h | 67 ++++++------ src/models/AtomicPropositionsLabeling.h | 2 +- src/models/Dtmc.h | 4 +- src/storage/BitVector.h | 4 +- src/utility/graph.h | 102 ++++++++++-------- 9 files changed, 116 insertions(+), 97 deletions(-) diff --git a/src/counterexamples/MILPMinimalLabelSetGenerator.h b/src/counterexamples/MILPMinimalLabelSetGenerator.h index 5023c342e..9b3c2b3e2 100644 --- a/src/counterexamples/MILPMinimalLabelSetGenerator.h +++ b/src/counterexamples/MILPMinimalLabelSetGenerator.h @@ -90,10 +90,9 @@ namespace storm { */ static struct StateInformation determineRelevantAndProblematicStates(storm::models::Mdp const& labeledMdp, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates) { StateInformation result; - storm::storage::SparseMatrix backwardTransitions = labeledMdp.getBackwardTransitions(); - result.relevantStates = storm::utility::graph::performProbGreater0E(labeledMdp, backwardTransitions, phiStates, psiStates); + result.relevantStates = storm::utility::graph::performProbGreater0E(labeledMdp.getTransitionMatrix(), labeledMdp.getNondeterministicChoiceIndices(), labeledMdp.getBackwardTransitions(), phiStates, psiStates); result.relevantStates &= ~psiStates; - result.problematicStates = storm::utility::graph::performProbGreater0E(labeledMdp, backwardTransitions, phiStates, psiStates); + result.problematicStates = storm::utility::graph::performProbGreater0E(labeledMdp.getTransitionMatrix(), labeledMdp.getNondeterministicChoiceIndices(), labeledMdp.getBackwardTransitions(), phiStates, psiStates); result.problematicStates.complement(); result.problematicStates &= result.relevantStates; LOG4CPLUS_DEBUG(logger, "Found " << phiStates.getNumberOfSetBits() << " filter states."); diff --git a/src/counterexamples/PathBasedSubsystemGenerator.h b/src/counterexamples/PathBasedSubsystemGenerator.h index 18355e1c4..976da3120 100644 --- a/src/counterexamples/PathBasedSubsystemGenerator.h +++ b/src/counterexamples/PathBasedSubsystemGenerator.h @@ -570,7 +570,7 @@ public: storm::property::prctl::Until const* until = dynamic_cast const*>(pathFormulaPtr); if(eventually != nullptr) { targetStates = eventually->getChild().check(modelCheck); - allowedStates = storm::storage::BitVector(targetStates.getSize(), true); + allowedStates = storm::storage::BitVector(targetStates.size(), true); } else if(globally != nullptr){ //eventually reaching a state without property visiting only states with property diff --git a/src/counterexamples/SMTMinimalCommandSetGenerator.h b/src/counterexamples/SMTMinimalCommandSetGenerator.h index a4041a070..bd8d45499 100644 --- a/src/counterexamples/SMTMinimalCommandSetGenerator.h +++ b/src/counterexamples/SMTMinimalCommandSetGenerator.h @@ -103,7 +103,7 @@ namespace storm { // Compute all relevant states, i.e. states for which there exists a scheduler that has a non-zero // probabilitiy of satisfying phi until psi. storm::storage::SparseMatrix backwardTransitions = labeledMdp.getBackwardTransitions(); - relevancyInformation.relevantStates = storm::utility::graph::performProbGreater0E(labeledMdp, backwardTransitions, phiStates, psiStates); + relevancyInformation.relevantStates = storm::utility::graph::performProbGreater0E(labeledMdp.getTransitionMatrix(), labeledMdp.getNondeterministicChoiceIndices(), backwardTransitions, phiStates, psiStates); relevancyInformation.relevantStates &= ~psiStates; LOG4CPLUS_DEBUG(logger, "Found " << relevancyInformation.relevantStates.getNumberOfSetBits() << " relevant states."); @@ -1427,7 +1427,7 @@ namespace storm { } storm::storage::BitVector unreachableRelevantStates = ~reachableStates & relevancyInformation.relevantStates; - storm::storage::BitVector statesThatCanReachTargetStates = storm::utility::graph::performProbGreater0E(subMdp, subMdp.getBackwardTransitions(), phiStates, psiStates); + storm::storage::BitVector statesThatCanReachTargetStates = storm::utility::graph::performProbGreater0E(subMdp.getTransitionMatrix(), subMdp.getNondeterministicChoiceIndices(), subMdp.getBackwardTransitions(), phiStates, psiStates); std::vector> guaranteedLabelSets = storm::utility::counterexamples::getGuaranteedLabelSets(originalMdp, statesThatCanReachTargetStates, relevancyInformation.relevantLabels); LOG4CPLUS_DEBUG(logger, "Found " << reachableLabels.size() << " reachable labels and " << reachableStates.getNumberOfSetBits() << " reachable states."); @@ -1547,7 +1547,7 @@ namespace storm { LOG4CPLUS_DEBUG(logger, "Successfully determined reachable state space."); storm::storage::BitVector unreachableRelevantStates = ~reachableStates & relevancyInformation.relevantStates; - storm::storage::BitVector statesThatCanReachTargetStates = storm::utility::graph::performProbGreater0E(subMdp, subMdp.getBackwardTransitions(), phiStates, psiStates); + storm::storage::BitVector statesThatCanReachTargetStates = storm::utility::graph::performProbGreater0E(subMdp.getTransitionMatrix(), subMdp.getNondeterministicChoiceIndices(), subMdp.getBackwardTransitions(), phiStates, psiStates); std::vector> guaranteedLabelSets = storm::utility::counterexamples::getGuaranteedLabelSets(originalMdp, statesThatCanReachTargetStates, relevancyInformation.relevantLabels); // Search for states for which we could enable another option and possibly improve the reachability probability. diff --git a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h index 80943c8e0..f050a2286 100644 --- a/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h +++ b/src/modelchecker/csl/SparseMarkovAutomatonCslModelChecker.h @@ -4,6 +4,7 @@ #include #include "src/modelchecker/csl/AbstractModelChecker.h" +#include "src/modelchecker/prctl/SparseMdpPrctlModelChecker.h" #include "src/models/MarkovAutomaton.h" #include "src/storage/BitVector.h" #include "src/storage/MaximalEndComponentDecomposition.h" @@ -20,7 +21,6 @@ namespace storm { template class SparseMarkovAutomatonCslModelChecker : public AbstractModelChecker { public: - explicit SparseMarkovAutomatonCslModelChecker(storm::models::MarkovAutomaton const& model, std::shared_ptr> nondeterministicLinearEquationSolver = storm::utility::solver::getNondeterministicLinearEquationSolver()) : AbstractModelChecker(model), minimumOperatorStack(), nondeterministicLinearEquationSolver(nondeterministicLinearEquationSolver) { // Intentionally left empty. } @@ -34,7 +34,13 @@ namespace storm { } std::vector checkUntil(storm::property::csl::Until const& formula, bool qualitative) const { - throw storm::exceptions::NotImplementedException() << "Model checking Until formulas on Markov automata is not yet implemented."; + storm::storage::BitVector leftStates = formula.getLeft().check(*this); + storm::storage::BitVector rightStates = formula.getRight().check(*this); + return computeUnboundedUntilProbabilities(minimumOperatorStack.top(), leftStates, rightStates, qualitative).first; + } + + std::pair, storm::storage::TotalScheduler> computeUnboundedUntilProbabilities(bool min, storm::storage::BitVector const& leftStates, storm::storage::BitVector const& rightStates, bool qualitative) const { + return storm::modelchecker::prctl::SparseMdpPrctlModelChecker::computeUnboundedUntilProbabilities(min, this->getModel().getTransitionMatrix(), this->getModel().getNondeterministicChoiceIndices(), this->getModel().getBackwardTransitions(), this->getModel().getInitialStates(), leftStates, rightStates, nondeterministicLinearEquationSolver, qualitative); } std::vector checkTimeBoundedUntil(storm::property::csl::TimeBoundedUntil const& formula, bool qualitative) const { @@ -42,7 +48,8 @@ namespace storm { } std::vector checkTimeBoundedEventually(storm::property::csl::TimeBoundedEventually const& formula, bool qualitative) const { - throw storm::exceptions::NotImplementedException() << "Model checking time-bounded Until formulas on Markov automata is not yet implemented."; + storm::storage::BitVector goalStates = formula.getChild().check(*this); + return this->checkTimeBoundedEventually(this->minimumOperatorStack.top(), goalStates, formula.getLowerBound(), formula.getUpperBound()); } std::vector checkGlobally(storm::property::csl::Globally const& formula, bool qualitative) const { @@ -50,7 +57,8 @@ namespace storm { } std::vector checkEventually(storm::property::csl::Eventually const& formula, bool qualitative) const { - throw storm::exceptions::NotImplementedException() << "Model checking Eventually formulas on Markov automata is not yet implemented."; + storm::storage::BitVector subFormulaStates = formula.getChild().check(*this); + return computeUnboundedUntilProbabilities(minimumOperatorStack.top(), storm::storage::BitVector(this->getModel().getNumberOfStates(), true), subFormulaStates, qualitative).first; } std::vector checkNext(storm::property::csl::Next const& formula, bool qualitative) const { @@ -391,18 +399,13 @@ namespace storm { // Prepare result vector. std::vector result(this->getModel().getNumberOfStates()); - std::cout << "res " << result << std::endl; - // Set the values for states not contained in MECs. - std::cout << x << std::endl; storm::utility::vector::setVectorValues(result, statesNotContainedInAnyMec, x); - std::cout << "res " << result << std::endl; // Set the values for all states in MECs. for (auto state : statesInMecs) { result[state] = lraValuesForEndComponents[stateToMecIndexMap[state]]; } - std::cout << "res " << result << std::endl; return result; } diff --git a/src/modelchecker/prctl/SparseMdpPrctlModelChecker.h b/src/modelchecker/prctl/SparseMdpPrctlModelChecker.h index b51bb5565..008ea3606 100644 --- a/src/modelchecker/prctl/SparseMdpPrctlModelChecker.h +++ b/src/modelchecker/prctl/SparseMdpPrctlModelChecker.h @@ -96,9 +96,9 @@ namespace storm { // Determine the states that have 0 probability of reaching the target states. storm::storage::BitVector statesWithProbabilityGreater0; if (this->minimumOperatorStack.top()) { - statesWithProbabilityGreater0 = storm::utility::graph::performProbGreater0A(this->getModel(), this->getModel().getBackwardTransitions(), phiStates, psiStates, true, stepBound); + statesWithProbabilityGreater0 = storm::utility::graph::performProbGreater0A(this->getModel().getTransitionMatrix(), this->getModel().getNondeterministicChoiceIndices(), this->getModel().getBackwardTransitions(), phiStates, psiStates, true, stepBound); } else { - statesWithProbabilityGreater0 = storm::utility::graph::performProbGreater0E(this->getModel(), this->getModel().getBackwardTransitions(), phiStates, psiStates, true, stepBound); + statesWithProbabilityGreater0 = storm::utility::graph::performProbGreater0E(this->getModel().getTransitionMatrix(), this->getModel().getNondeterministicChoiceIndices(), this->getModel().getBackwardTransitions(), phiStates, psiStates, true, stepBound); } // Check if we already know the result (i.e. probability 0) for all initial states and @@ -278,29 +278,30 @@ namespace storm { * @return The probabilities for the satisfying phi until psi for each state of the model. If the * qualitative flag is set, exact probabilities might not be computed. */ - std::pair, storm::storage::TotalScheduler> checkUntil(bool minimize, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates, bool qualitative) const { + static std::pair, storm::storage::TotalScheduler> computeUnboundedUntilProbabilities(bool minimize, storm::storage::SparseMatrix const& transitionMatrix, std::vector nondeterministicChoiceIndices, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& initialStates, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates, std::shared_ptr> nondeterministicLinearEquationSolver, bool qualitative) { + size_t numberOfStates = phiStates.size(); + // We need to identify the states which have to be taken out of the matrix, i.e. // all states that have probability 0 and 1 of satisfying the until-formula. std::pair statesWithProbability01; if (minimize) { - statesWithProbability01 = storm::utility::graph::performProb01Min(this->getModel(), phiStates, psiStates); + statesWithProbability01 = storm::utility::graph::performProb01Min(transitionMatrix, nondeterministicChoiceIndices, backwardTransitions, phiStates, psiStates); } else { - statesWithProbability01 = storm::utility::graph::performProb01Max(this->getModel(), phiStates, psiStates); + statesWithProbability01 = storm::utility::graph::performProb01Max(transitionMatrix, nondeterministicChoiceIndices, backwardTransitions, phiStates, psiStates); } storm::storage::BitVector statesWithProbability0 = std::move(statesWithProbability01.first); storm::storage::BitVector statesWithProbability1 = std::move(statesWithProbability01.second); - storm::storage::BitVector maybeStates = ~(statesWithProbability0 | statesWithProbability1); LOG4CPLUS_INFO(logger, "Found " << statesWithProbability0.getNumberOfSetBits() << " 'no' states."); LOG4CPLUS_INFO(logger, "Found " << statesWithProbability1.getNumberOfSetBits() << " 'yes' states."); LOG4CPLUS_INFO(logger, "Found " << maybeStates.getNumberOfSetBits() << " 'maybe' states."); // Create resulting vector. - std::vector result(this->getModel().getNumberOfStates()); + std::vector result(numberOfStates); // Check whether we need to compute exact probabilities for some states. - if (this->getModel().getInitialStates().isDisjointFrom(maybeStates) || qualitative) { + if (initialStates.isDisjointFrom(maybeStates) || qualitative) { if (qualitative) { LOG4CPLUS_INFO(logger, "The formula was checked qualitatively. No exact probabilities were computed."); } else { @@ -315,20 +316,20 @@ namespace storm { // First, we can eliminate the rows and columns from the original transition probability matrix for states // whose probabilities are already known. - storm::storage::SparseMatrix submatrix = this->getModel().getTransitionMatrix().getSubmatrix(maybeStates, this->getModel().getNondeterministicChoiceIndices()); + storm::storage::SparseMatrix submatrix = transitionMatrix.getSubmatrix(maybeStates, nondeterministicChoiceIndices); // Get the "new" nondeterministic choice indices for the submatrix. - std::vector subNondeterministicChoiceIndices = storm::utility::vector::getConstrainedOffsetVector(this->getModel().getNondeterministicChoiceIndices(), maybeStates); + std::vector subNondeterministicChoiceIndices = storm::utility::vector::getConstrainedOffsetVector(nondeterministicChoiceIndices, maybeStates); // Prepare the right-hand side of the equation system. For entry i this corresponds to // the accumulated probability of going from state i to some 'yes' state. - std::vector b = this->getModel().getTransitionMatrix().getConstrainedRowSumVector(maybeStates, this->getModel().getNondeterministicChoiceIndices(), statesWithProbability1, submatrix.getRowCount()); + std::vector b = transitionMatrix.getConstrainedRowSumVector(maybeStates, nondeterministicChoiceIndices, statesWithProbability1, submatrix.getRowCount()); // Create vector for results for maybe states. std::vector x(maybeStates.getNumberOfSetBits()); - + // Solve the corresponding system of equations. - this->nondeterministicLinearEquationSolver->solveEquationSystem(minimize, submatrix, x, b, subNondeterministicChoiceIndices); + nondeterministicLinearEquationSolver->solveEquationSystem(minimize, submatrix, x, b, subNondeterministicChoiceIndices); // Set values of resulting vector according to result. storm::utility::vector::setVectorValues(result, maybeStates, x); @@ -339,11 +340,15 @@ namespace storm { storm::utility::vector::setVectorValues(result, statesWithProbability1, storm::utility::constGetOne()); // Finally, compute a scheduler that achieves the extramal value. - storm::storage::TotalScheduler scheduler = this->computeExtremalScheduler(minimize, false, result); - + storm::storage::TotalScheduler scheduler = computeExtremalScheduler(minimize, transitionMatrix, nondeterministicChoiceIndices, result); + return std::make_pair(result, scheduler); } + std::pair, storm::storage::TotalScheduler> checkUntil(bool minimize, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates, bool qualitative) const { + return computeUnboundedUntilProbabilities(minimize, this->getModel().getTransitionMatrix(), this->getModel().getNondeterministicChoiceIndices(), this->getModel().getBackwardTransitions(), this->getModel().getInitialStates(), phiStates, psiStates, this->nondeterministicLinearEquationSolver, qualitative); + } + /*! * Checks the given formula that is an instantaneous reward formula. * @@ -453,9 +458,9 @@ namespace storm { storm::storage::BitVector infinityStates; storm::storage::BitVector trueStates(this->getModel().getNumberOfStates(), true); if (minimize) { - infinityStates = std::move(storm::utility::graph::performProb1A(this->getModel(), this->getModel().getBackwardTransitions(), trueStates, targetStates)); + infinityStates = std::move(storm::utility::graph::performProb1A(this->getModel().getTransitionMatrix(), this->getModel().getNondeterministicChoiceIndices(), this->getModel().getBackwardTransitions(), trueStates, targetStates)); } else { - infinityStates = std::move(storm::utility::graph::performProb1E(this->getModel(), this->getModel().getBackwardTransitions(), trueStates, targetStates)); + infinityStates = std::move(storm::utility::graph::performProb1E(this->getModel().getTransitionMatrix(), this->getModel().getNondeterministicChoiceIndices(), this->getModel().getBackwardTransitions(), trueStates, targetStates)); } infinityStates.complement(); storm::storage::BitVector maybeStates = ~targetStates & ~infinityStates; @@ -526,8 +531,8 @@ namespace storm { storm::utility::vector::setVectorValues(result, infinityStates, storm::utility::constGetInfinity()); // Finally, compute a scheduler that achieves the extramal value. - storm::storage::TotalScheduler scheduler = this->computeExtremalScheduler(this->minimumOperatorStack.top(), false, result); - + storm::storage::TotalScheduler scheduler = computeExtremalScheduler(this->minimumOperatorStack.top(), this->getModel().getTransitionMatrix(), this->getModel().getNondeterministicChoiceIndices(), result, this->getModel().hasStateRewards() ? &this->getModel().getStateRewardVector() : nullptr, this->getModel().hasTransitionRewards() ? &this->getModel().getTransitionRewardMatrix() : nullptr); + return std::make_pair(result, scheduler); } @@ -540,33 +545,33 @@ namespace storm { * @param takenChoices The output vector that is to store the taken choices. * @param nondeterministicChoiceIndices The assignment of states to their nondeterministic choices in the matrix. */ - storm::storage::TotalScheduler computeExtremalScheduler(bool minimize, bool addRewards, std::vector const& result) const { - std::vector temporaryResult(this->getModel().getNondeterministicChoiceIndices().size() - 1); + static storm::storage::TotalScheduler computeExtremalScheduler(bool minimize, storm::storage::SparseMatrix const& transitionMatrix, std::vector const& nondeterministicChoiceIndices, std::vector const& result, std::vector const* stateRewardVector = nullptr, storm::storage::SparseMatrix const* transitionRewardMatrix = nullptr) { + std::vector temporaryResult(nondeterministicChoiceIndices.size() - 1); std::vector nondeterministicResult(result); storm::solver::GmmxxLinearEquationSolver solver; - solver.performMatrixVectorMultiplication(this->getModel().getTransitionMatrix(), nondeterministicResult, nullptr, 1); - if (addRewards) { + solver.performMatrixVectorMultiplication(transitionMatrix, nondeterministicResult, nullptr, 1); + if (stateRewardVector != nullptr || transitionRewardMatrix != nullptr) { std::vector totalRewardVector; - if (this->getModel().hasTransitionRewards()) { - std::vector totalRewardVector = this->getModel().getTransitionMatrix().getPointwiseProductRowSumVector(this->getModel().getTransitionRewardMatrix()); - if (this->getModel().hasStateRewards()) { + if (transitionRewardMatrix != nullptr) { + totalRewardVector = transitionMatrix.getPointwiseProductRowSumVector(*transitionRewardMatrix); + if (stateRewardVector != nullptr) { std::vector stateRewards(totalRewardVector.size()); - storm::utility::vector::selectVectorValuesRepeatedly(stateRewards, storm::storage::BitVector(this->getModel().getStateRewardVector().size(), true), this->getModel().getNondeterministicChoiceIndices(), this->getModel().getStateRewardVector()); + storm::utility::vector::selectVectorValuesRepeatedly(stateRewards, storm::storage::BitVector(stateRewardVector->size(), true), nondeterministicChoiceIndices, *stateRewardVector); storm::utility::vector::addVectorsInPlace(totalRewardVector, stateRewards); } } else { totalRewardVector.resize(nondeterministicResult.size()); - storm::utility::vector::selectVectorValuesRepeatedly(totalRewardVector, storm::storage::BitVector(this->getModel().getStateRewardVector().size(), true), this->getModel().getNondeterministicChoiceIndices(), this->getModel().getStateRewardVector()); + storm::utility::vector::selectVectorValuesRepeatedly(totalRewardVector, storm::storage::BitVector(stateRewardVector->size(), true), nondeterministicChoiceIndices, *stateRewardVector); } storm::utility::vector::addVectorsInPlace(nondeterministicResult, totalRewardVector); } - std::vector choices(this->getModel().getNumberOfStates()); + std::vector choices(result.size()); if (minimize) { - storm::utility::vector::reduceVectorMin(nondeterministicResult, temporaryResult, this->getModel().getNondeterministicChoiceIndices(), &choices); + storm::utility::vector::reduceVectorMin(nondeterministicResult, temporaryResult, nondeterministicChoiceIndices, &choices); } else { - storm::utility::vector::reduceVectorMax(nondeterministicResult, temporaryResult, this->getModel().getNondeterministicChoiceIndices(), &choices); + storm::utility::vector::reduceVectorMax(nondeterministicResult, temporaryResult, nondeterministicChoiceIndices, &choices); } return storm::storage::TotalScheduler(choices); diff --git a/src/models/AtomicPropositionsLabeling.h b/src/models/AtomicPropositionsLabeling.h index 486997829..2de73bd1d 100644 --- a/src/models/AtomicPropositionsLabeling.h +++ b/src/models/AtomicPropositionsLabeling.h @@ -289,7 +289,7 @@ public: */ void addState() { for(uint_fast64_t i = 0; i < apsCurrent; i++) { - singleLabelings[i].resize(singleLabelings[i].getSize() + 1); + singleLabelings[i].resize(singleLabelings[i].size() + 1); } stateCount++; } diff --git a/src/models/Dtmc.h b/src/models/Dtmc.h index b573f848d..563eee78e 100644 --- a/src/models/Dtmc.h +++ b/src/models/Dtmc.h @@ -147,13 +147,13 @@ public: } // Does the vector have the right size? - if(subSysStates.getSize() != this->getNumberOfStates()) { + if(subSysStates.size() != this->getNumberOfStates()) { LOG4CPLUS_INFO(logger, "BitVector has wrong size. Resizing it..."); subSysStates.resize(this->getNumberOfStates()); } // Test if it is a proper subsystem of this Dtmc, i.e. if there is at least one state to be left out. - if(subSysStates.getNumberOfSetBits() == subSysStates.getSize()) { + if(subSysStates.getNumberOfSetBits() == subSysStates.size()) { LOG4CPLUS_INFO(logger, "All states are kept. This is no proper subsystem."); return storm::models::Dtmc(*this); } diff --git a/src/storage/BitVector.h b/src/storage/BitVector.h index ea6c0a3b4..4b3cc1efb 100644 --- a/src/storage/BitVector.h +++ b/src/storage/BitVector.h @@ -675,8 +675,8 @@ public: /*! * Retrieves the number of bits this bit vector can store. */ - uint_fast64_t getSize() const { - return bitCount; + size_t size() const { + return static_cast(bitCount); } /*! diff --git a/src/utility/graph.h b/src/utility/graph.h index c9c58a86a..4a7a6f067 100644 --- a/src/utility/graph.h +++ b/src/utility/graph.h @@ -177,9 +177,11 @@ namespace storm { * @return A bit vector that represents all states with probability 0. */ template - storm::storage::BitVector performProbGreater0E(storm::models::AbstractNondeterministicModel const& model, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates, bool useStepBound = false, uint_fast64_t maximalSteps = 0) { + storm::storage::BitVector performProbGreater0E(storm::storage::SparseMatrix const& transitionMatrix, std::vector const& nondeterministicChoiceIndices, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates, bool useStepBound = false, uint_fast64_t maximalSteps = 0) { + size_t numberOfStates = phiStates.size(); + // Prepare resulting bit vector. - storm::storage::BitVector statesWithProbabilityGreater0(model.getNumberOfStates()); + storm::storage::BitVector statesWithProbabilityGreater0(numberOfStates); // Add all psi states as the already satisfy the condition. statesWithProbabilityGreater0 |= psiStates; @@ -191,9 +193,9 @@ namespace storm { std::vector stepStack; std::vector remainingSteps; if (useStepBound) { - stepStack.reserve(model.getNumberOfStates()); + stepStack.reserve(numberOfStates); stepStack.insert(stepStack.begin(), psiStates.getNumberOfSetBits(), maximalSteps); - remainingSteps.resize(model.getNumberOfStates()); + remainingSteps.resize(numberOfStates); for (auto state : psiStates) { remainingSteps[state] = maximalSteps; } @@ -230,6 +232,13 @@ namespace storm { return statesWithProbabilityGreater0; } + template + storm::storage::BitVector performProb0A(storm::storage::SparseMatrix const& transitionMatrix, std::vector const& nondeterministicChoiceIndices, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates) { + storm::storage::BitVector statesWithProbability0 = performProbGreater0E(transitionMatrix, nondeterministicChoiceIndices, backwardTransitions, phiStates, psiStates); + statesWithProbability0.complement(); + return statesWithProbability0; + } + /*! * Computes the sets of states that have probability 0 of satisfying phi until psi under all * possible resolutions of non-determinism in a non-deterministic model. Stated differently, @@ -246,11 +255,9 @@ namespace storm { */ template storm::storage::BitVector performProb0A(storm::models::AbstractNondeterministicModel const& model, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates) { - storm::storage::BitVector statesWithProbability0 = performProbGreater0E(model, backwardTransitions, phiStates, psiStates); - statesWithProbability0.complement(); - return statesWithProbability0; + return performProb0A(model.getTransitionMatrix(), model.getNondeterministicChoiceIndices(), backwardTransitions, phiStates, psiStates); } - + /*! * Computes the sets of states that have probability 1 of satisfying phi until psi under at least * one possible resolution of non-determinism in a non-deterministic model. Stated differently, @@ -264,15 +271,13 @@ namespace storm { * @return A bit vector that represents all states with probability 1. */ template - storm::storage::BitVector performProb1E(storm::models::AbstractNondeterministicModel const& model, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates) { - // Get some temporaries for convenience. - storm::storage::SparseMatrix const& transitionMatrix = model.getTransitionMatrix(); - std::vector const& nondeterministicChoiceIndices = model.getNondeterministicChoiceIndices(); + storm::storage::BitVector performProb1E(storm::storage::SparseMatrix const& transitionMatrix, std::vector const& nondeterministicChoiceIndices, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates) { + size_t numberOfStates = phiStates.size(); // Initialize the environment for the iterative algorithm. - storm::storage::BitVector currentStates(model.getNumberOfStates(), true); + storm::storage::BitVector currentStates(numberOfStates, true); std::vector stack; - stack.reserve(model.getNumberOfStates()); + stack.reserve(numberOfStates); // Perform the loop as long as the set of states gets larger. bool done = false; @@ -323,6 +328,15 @@ namespace storm { return currentStates; } + template + std::pair performProb01Max(storm::storage::SparseMatrix const& transitionMatrix, std::vector const& nondeterministicChoiceIndices, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates) { + std::pair result; + + result.first = performProb0A(transitionMatrix, nondeterministicChoiceIndices, backwardTransitions, phiStates, psiStates); + result.second = performProb1E(transitionMatrix, nondeterministicChoiceIndices, backwardTransitions, phiStates, psiStates); + return result; + } + /*! * Computes the sets of states that have probability 0 or 1, respectively, of satisfying phi * until psi in a non-deterministic model in which all non-deterministic choices are resolved @@ -335,14 +349,7 @@ namespace storm { */ template std::pair performProb01Max(storm::models::AbstractNondeterministicModel const& model, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates) { - std::pair result; - - // Get the backwards transition relation from the model to ease the search. - storm::storage::SparseMatrix backwardTransitions = model.getBackwardTransitions(); - - result.first = performProb0A(model, backwardTransitions, phiStates, psiStates); - result.second = performProb1E(model, backwardTransitions, phiStates, psiStates); - return result; + return performProb01Max(model.getTransitionMatrix(), model.getNondeterministicChoiceIndices(), model.getBackwardTransitions(), phiStates, psiStates); } /*! @@ -360,13 +367,11 @@ namespace storm { * @return A bit vector that represents all states with probability 0. */ template - storm::storage::BitVector performProbGreater0A(storm::models::AbstractNondeterministicModel const& model, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates, bool useStepBound = false, uint_fast64_t maximalSteps = 0) { - // Prepare resulting bit vector. - storm::storage::BitVector statesWithProbabilityGreater0(model.getNumberOfStates()); + storm::storage::BitVector performProbGreater0A(storm::storage::SparseMatrix const& transitionMatrix, std::vector const& nondeterministicChoiceIndices, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates, bool useStepBound = false, uint_fast64_t maximalSteps = 0) { + size_t numberOfStates = phiStates.size(); - // Get some temporaries for convenience. - storm::storage::SparseMatrix const& transitionMatrix = model.getTransitionMatrix(); - std::vector const& nondeterministicChoiceIndices = model.getNondeterministicChoiceIndices(); + // Prepare resulting bit vector. + storm::storage::BitVector statesWithProbabilityGreater0(numberOfStates); // Add all psi states as the already satisfy the condition. statesWithProbabilityGreater0 |= psiStates; @@ -378,9 +383,9 @@ namespace storm { std::vector stepStack; std::vector remainingSteps; if (useStepBound) { - stepStack.reserve(model.getNumberOfStates()); + stepStack.reserve(numberOfStates); stepStack.insert(stepStack.begin(), psiStates.getNumberOfSetBits(), maximalSteps); - remainingSteps.resize(model.getNumberOfStates()); + remainingSteps.resize(numberOfStates); for (auto state : psiStates) { remainingSteps[state] = maximalSteps; } @@ -452,7 +457,14 @@ namespace storm { */ template storm::storage::BitVector performProb0E(storm::models::AbstractNondeterministicModel const& model, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates) { - storm::storage::BitVector statesWithProbability0 = performProbGreater0A(model, backwardTransitions, phiStates, psiStates); + storm::storage::BitVector statesWithProbability0 = performProbGreater0A(model.getTransitionMatrix(), model.getNondeterministicChoiceIndices(), backwardTransitions, phiStates, psiStates); + statesWithProbability0.complement(); + return statesWithProbability0; + } + + template + storm::storage::BitVector performProb0E(storm::storage::SparseMatrix const& transitionMatrix, std::vector const& nondeterministicChoiceIndices, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates) { + storm::storage::BitVector statesWithProbability0 = performProbGreater0A(transitionMatrix, nondeterministicChoiceIndices, backwardTransitions, phiStates, psiStates); statesWithProbability0.complement(); return statesWithProbability0; } @@ -470,15 +482,13 @@ namespace storm { * @return A bit vector that represents all states with probability 0. */ template - storm::storage::BitVector performProb1A(storm::models::AbstractNondeterministicModel const& model, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates) { - // Get some temporaries for convenience. - storm::storage::SparseMatrix const& transitionMatrix = model.getTransitionMatrix(); - std::vector const& nondeterministicChoiceIndices = model.getNondeterministicChoiceIndices(); + storm::storage::BitVector performProb1A( storm::storage::SparseMatrix const& transitionMatrix, std::vector const& nondeterministicChoiceIndices, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates) { + size_t numberOfStates = phiStates.size(); // Initialize the environment for the iterative algorithm. - storm::storage::BitVector currentStates(model.getNumberOfStates(), true); + storm::storage::BitVector currentStates(numberOfStates, true); std::vector stack; - stack.reserve(model.getNumberOfStates()); + stack.reserve(numberOfStates); // Perform the loop as long as the set of states gets smaller. bool done = false; @@ -528,6 +538,15 @@ namespace storm { return currentStates; } + template + std::pair performProb01Min(storm::storage::SparseMatrix const& transitionMatrix, std::vector const& nondeterministicChoiceIndices, storm::storage::SparseMatrix const& backwardTransitions, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates) { + std::pair result; + + result.first = performProb0E(transitionMatrix, nondeterministicChoiceIndices, backwardTransitions, phiStates, psiStates); + result.second = performProb1A(transitionMatrix, nondeterministicChoiceIndices, backwardTransitions, phiStates, psiStates); + return result; + } + /*! * Computes the sets of states that have probability 0 or 1, respectively, of satisfying phi * until psi in a non-deterministic model in which all non-deterministic choices are resolved @@ -540,14 +559,7 @@ namespace storm { */ template std::pair performProb01Min(storm::models::AbstractNondeterministicModel const& model, storm::storage::BitVector const& phiStates, storm::storage::BitVector const& psiStates) { - std::pair result; - - // Get the backwards transition relation from the model to ease the search. - storm::storage::SparseMatrix backwardTransitions = model.getBackwardTransitions(); - - result.first = performProb0E(model, backwardTransitions, phiStates, psiStates); - result.second = performProb1A(model, backwardTransitions, phiStates, psiStates); - return result; + return performProb01Min(model.getTransitionMatrix(), model.getNondeterministicChoiceIndices(), model.getBackwardTransitions(), phiStates, psiStates); } /*! From c336fd7ff8283a60cbcbd4b892cedb64fe9dc3da Mon Sep 17 00:00:00 2001 From: dehnert Date: Tue, 3 Dec 2013 22:29:44 +0100 Subject: [PATCH 9/9] Minor fixes for implementation of GlpkLpSolver if glpk is unavailable. Former-commit-id: 778f93a33c955025b1cc001bc3bd6581fcfc9bb0 --- src/solver/GlpkLpSolver.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/solver/GlpkLpSolver.h b/src/solver/GlpkLpSolver.h index 5576b1111..3741c70e3 100644 --- a/src/solver/GlpkLpSolver.h +++ b/src/solver/GlpkLpSolver.h @@ -62,7 +62,7 @@ namespace storm { throw storm::exceptions::NotImplementedException() << "This version of StoRM was compiled without support for glpk. Yet, a method was called that requires this support. Please choose a version of support with glpk support."; } - virtual GlpkLpSolver() { + virtual ~GlpkLpSolver() { throw storm::exceptions::NotImplementedException() << "This version of StoRM was compiled without support for glpk. Yet, a method was called that requires this support. Please choose a version of support with glpk support."; }