Browse Source
Created StateEliminator with specialized subclasses
Created StateEliminator with specialized subclasses
Former-commit-id: 991e3fcfcd
tempestpy_adaptions
Mavo
9 years ago
10 changed files with 626 additions and 310 deletions
-
326src/modelchecker/reachability/SparseDtmcEliminationModelChecker.cpp
-
41src/modelchecker/reachability/SparseDtmcEliminationModelChecker.h
-
53src/solver/stateelimination/ConditionalEliminator.cpp
-
53src/solver/stateelimination/ConditionalEliminator.h
-
47src/solver/stateelimination/LongRunAverageEliminator.cpp
-
36src/solver/stateelimination/LongRunAverageEliminator.h
-
45src/solver/stateelimination/PrioritizedEliminator.cpp
-
34src/solver/stateelimination/PrioritizedEliminator.h
-
259src/solver/stateelimination/StateEliminator.cpp
-
42src/solver/stateelimination/StateEliminator.h
@ -0,0 +1,53 @@ |
|||
#include "src/solver/stateelimination/ConditionalEliminator.h"
|
|||
|
|||
namespace storm { |
|||
namespace solver { |
|||
namespace stateelimination { |
|||
|
|||
template<typename SparseModelType> |
|||
ConditionalEliminator<SparseModelType>::ConditionalEliminator(storm::storage::FlexibleSparseMatrix<ValueType>& transitionMatrix, storm::storage::FlexibleSparseMatrix<ValueType>& backwardTransitions, std::vector<ValueType>& oneStepProbabilities, storm::storage::BitVector& phiStates, storm::storage::BitVector& psiStates) : StateEliminator<SparseModelType>(transitionMatrix, backwardTransitions), oneStepProbabilities(oneStepProbabilities), phiStates(phiStates), psiStates(psiStates), specificState(NONE) { |
|||
} |
|||
|
|||
template<typename SparseModelType> |
|||
void ConditionalEliminator<SparseModelType>::updateValue(storm::storage::sparse::state_type const& state, ValueType const& loopProbability) { |
|||
oneStepProbabilities[state] = storm::utility::simplify(loopProbability * oneStepProbabilities[state]); |
|||
} |
|||
|
|||
template<typename SparseModelType> |
|||
void ConditionalEliminator<SparseModelType>::updatePredecessor(storm::storage::sparse::state_type const& predecessor, ValueType const& probability, storm::storage::sparse::state_type const& state) { |
|||
oneStepProbabilities[predecessor] = storm::utility::simplify(oneStepProbabilities[predecessor] * storm::utility::simplify(probability * oneStepProbabilities[state])); |
|||
} |
|||
|
|||
template<typename SparseModelType> |
|||
void ConditionalEliminator<SparseModelType>::updatePriority(storm::storage::sparse::state_type const& state) { |
|||
// Do nothing
|
|||
} |
|||
|
|||
template<typename SparseModelType> |
|||
bool ConditionalEliminator<SparseModelType>::filterPredecessor(storm::storage::sparse::state_type const& state) { |
|||
// TODO find better solution than flag
|
|||
switch (specificState) { |
|||
case PHI: |
|||
return phiStates.get(state); |
|||
case PSI: |
|||
return psiStates.get(state); |
|||
case NONE: |
|||
assert(false); |
|||
} |
|||
} |
|||
|
|||
template<typename SparseModelType> |
|||
bool ConditionalEliminator<SparseModelType>::isFilterPredecessor() const { |
|||
return true; |
|||
} |
|||
|
|||
|
|||
template class ConditionalEliminator<storm::models::sparse::Dtmc<double>>; |
|||
|
|||
#ifdef STORM_HAVE_CARL
|
|||
template class ConditionalEliminator<storm::models::sparse::Dtmc<storm::RationalFunction>>; |
|||
#endif
|
|||
|
|||
} // namespace stateelimination
|
|||
} // namespace storage
|
|||
} // namespace storm
|
@ -0,0 +1,53 @@ |
|||
#ifndef STORM_SOLVER_STATEELIMINATION_CONDITIONALELIMINATOR_H_ |
|||
#define STORM_SOLVER_STATEELIMINATION_CONDITIONALELIMINATOR_H_ |
|||
|
|||
#include "src/solver/stateelimination/StateEliminator.h" |
|||
|
|||
#include "src/storage/BitVector.h" |
|||
|
|||
namespace storm { |
|||
namespace solver { |
|||
namespace stateelimination { |
|||
|
|||
template<typename SparseModelType> |
|||
class ConditionalEliminator : public StateEliminator<SparseModelType> { |
|||
|
|||
typedef typename SparseModelType::ValueType ValueType; |
|||
|
|||
enum SpecificState { NONE, PHI, PSI}; |
|||
|
|||
public: |
|||
ConditionalEliminator(storm::storage::FlexibleSparseMatrix<ValueType>& transitionMatrix, storm::storage::FlexibleSparseMatrix<ValueType>& backwardTransitions, std::vector<ValueType>& oneStepProbabilities, storm::storage::BitVector& phiStates, storm::storage::BitVector& psiStates); |
|||
|
|||
// Instantiaton of Virtual methods |
|||
void updateValue(storm::storage::sparse::state_type const& state, ValueType const& loopProbability) override; |
|||
void updatePredecessor(storm::storage::sparse::state_type const& predecessor, ValueType const& probability, storm::storage::sparse::state_type const& state) override; |
|||
void updatePriority(storm::storage::sparse::state_type const& state) override; |
|||
bool filterPredecessor(storm::storage::sparse::state_type const& state) override; |
|||
bool isFilterPredecessor() const override; |
|||
|
|||
void setStatePsi() { |
|||
specificState = PSI; |
|||
} |
|||
|
|||
void setStatePhi() { |
|||
specificState = PHI; |
|||
} |
|||
|
|||
void clearState() { |
|||
specificState = NONE; |
|||
} |
|||
|
|||
private: |
|||
std::vector<ValueType>& oneStepProbabilities; |
|||
storm::storage::BitVector& phiStates; |
|||
storm::storage::BitVector& psiStates; |
|||
SpecificState specificState; |
|||
|
|||
}; |
|||
|
|||
} // namespace stateelimination |
|||
} // namespace storage |
|||
} // namespace storm |
|||
|
|||
#endif // STORM_SOLVER_STATEELIMINATION_CONDITIONALELIMINATOR_H_ |
@ -0,0 +1,47 @@ |
|||
#include "src/solver/stateelimination/LongRunAverageEliminator.h"
|
|||
|
|||
namespace storm { |
|||
namespace solver { |
|||
namespace stateelimination { |
|||
|
|||
template<typename SparseModelType> |
|||
LongRunAverageEliminator<SparseModelType>::LongRunAverageEliminator(storm::storage::FlexibleSparseMatrix<ValueType>& transitionMatrix, storm::storage::FlexibleSparseMatrix<ValueType>& backwardTransitions, storm::modelchecker::StatePriorityQueue<ValueType>& priorityQueue, std::vector<ValueType>& stateValues, std::vector<ValueType>& averageTimeInStates) : StateEliminator<SparseModelType>(transitionMatrix, backwardTransitions), priorityQueue(priorityQueue), stateValues(stateValues), averageTimeInStates(averageTimeInStates) { |
|||
} |
|||
|
|||
template<typename SparseModelType> |
|||
void LongRunAverageEliminator<SparseModelType>::updateValue(storm::storage::sparse::state_type const& state, ValueType const& loopProbability) { |
|||
stateValues[state] = storm::utility::simplify(loopProbability * stateValues[state]); |
|||
averageTimeInStates[state] = storm::utility::simplify(loopProbability * averageTimeInStates[state]); |
|||
} |
|||
|
|||
template<typename SparseModelType> |
|||
void LongRunAverageEliminator<SparseModelType>::updatePredecessor(storm::storage::sparse::state_type const& predecessor, ValueType const& probability, storm::storage::sparse::state_type const& state) { |
|||
stateValues[predecessor] = storm::utility::simplify(stateValues[predecessor] + storm::utility::simplify(probability * stateValues[state])); |
|||
averageTimeInStates[predecessor] = storm::utility::simplify(averageTimeInStates[predecessor] + storm::utility::simplify(probability * averageTimeInStates[state])); |
|||
} |
|||
|
|||
template<typename SparseModelType> |
|||
void LongRunAverageEliminator<SparseModelType>::updatePriority(storm::storage::sparse::state_type const& state) { |
|||
priorityQueue.update(state, StateEliminator<SparseModelType>::transitionMatrix, StateEliminator<SparseModelType>::backwardTransitions, stateValues); |
|||
} |
|||
|
|||
template<typename SparseModelType> |
|||
bool LongRunAverageEliminator<SparseModelType>::filterPredecessor(storm::storage::sparse::state_type const& state) { |
|||
assert(false); |
|||
} |
|||
|
|||
template<typename SparseModelType> |
|||
bool LongRunAverageEliminator<SparseModelType>::isFilterPredecessor() const { |
|||
return false; |
|||
} |
|||
|
|||
|
|||
template class LongRunAverageEliminator<storm::models::sparse::Dtmc<double>>; |
|||
|
|||
#ifdef STORM_HAVE_CARL
|
|||
template class LongRunAverageEliminator<storm::models::sparse::Dtmc<storm::RationalFunction>>; |
|||
#endif
|
|||
|
|||
} // namespace stateelimination
|
|||
} // namespace storage
|
|||
} // namespace storm
|
@ -0,0 +1,36 @@ |
|||
#ifndef STORM_SOLVER_STATEELIMINATION_LONGRUNAVERAGEELIMINATOR_H_ |
|||
#define STORM_SOLVER_STATEELIMINATION_LONGRUNAVERAGEELIMINATOR_H_ |
|||
|
|||
#include "src/solver/stateelimination/StateEliminator.h" |
|||
|
|||
namespace storm { |
|||
namespace solver { |
|||
namespace stateelimination { |
|||
|
|||
template<typename SparseModelType> |
|||
class LongRunAverageEliminator : public StateEliminator<SparseModelType> { |
|||
|
|||
typedef typename SparseModelType::ValueType ValueType; |
|||
|
|||
public: |
|||
LongRunAverageEliminator(storm::storage::FlexibleSparseMatrix<ValueType>& transitionMatrix, storm::storage::FlexibleSparseMatrix<ValueType>& backwardTransitions, storm::modelchecker::StatePriorityQueue<ValueType>& priorityQueue, std::vector<ValueType>& stateValues, std::vector<ValueType>& averageTimeInStates); |
|||
|
|||
// Instantiaton of Virtual methods |
|||
void updateValue(storm::storage::sparse::state_type const& state, ValueType const& loopProbability) override; |
|||
void updatePredecessor(storm::storage::sparse::state_type const& predecessor, ValueType const& probability, storm::storage::sparse::state_type const& state) override; |
|||
void updatePriority(storm::storage::sparse::state_type const& state) override; |
|||
bool filterPredecessor(storm::storage::sparse::state_type const& state) override; |
|||
bool isFilterPredecessor() const override; |
|||
|
|||
private: |
|||
|
|||
storm::modelchecker::StatePriorityQueue<ValueType>& priorityQueue; |
|||
std::vector<ValueType>& stateValues; |
|||
std::vector<ValueType>& averageTimeInStates; |
|||
}; |
|||
|
|||
} // namespace stateelimination |
|||
} // namespace storage |
|||
} // namespace storm |
|||
|
|||
#endif // STORM_SOLVER_STATEELIMINATION_LONGRUNAVERAGEELIMINATOR_H_ |
@ -0,0 +1,45 @@ |
|||
#include "src/solver/stateelimination/PrioritizedEliminator.h"
|
|||
|
|||
namespace storm { |
|||
namespace solver { |
|||
namespace stateelimination { |
|||
|
|||
template<typename SparseModelType> |
|||
PrioritizedEliminator<SparseModelType>::PrioritizedEliminator(storm::storage::FlexibleSparseMatrix<ValueType>& transitionMatrix, storm::storage::FlexibleSparseMatrix<ValueType>& backwardTransitions, storm::modelchecker::StatePriorityQueue<ValueType> priorityQueue, std::vector<ValueType>& stateValues) : StateEliminator<SparseModelType>(transitionMatrix, backwardTransitions), priorityQueue(priorityQueue), stateValues(stateValues) { |
|||
} |
|||
|
|||
template<typename SparseModelType> |
|||
void PrioritizedEliminator<SparseModelType>::updateValue(storm::storage::sparse::state_type const& state, ValueType const& loopProbability) { |
|||
stateValues[state] = storm::utility::simplify(loopProbability * stateValues[state]); |
|||
} |
|||
|
|||
template<typename SparseModelType> |
|||
void PrioritizedEliminator<SparseModelType>::updatePredecessor(storm::storage::sparse::state_type const& predecessor, ValueType const& probability, storm::storage::sparse::state_type const& state) { |
|||
stateValues[predecessor] = storm::utility::simplify(stateValues[predecessor] + storm::utility::simplify(probability * stateValues[state])); |
|||
} |
|||
|
|||
template<typename SparseModelType> |
|||
void PrioritizedEliminator<SparseModelType>::updatePriority(storm::storage::sparse::state_type const& state) { |
|||
priorityQueue.update(state, StateEliminator<SparseModelType>::transitionMatrix, StateEliminator<SparseModelType>::backwardTransitions, stateValues); |
|||
} |
|||
|
|||
template<typename SparseModelType> |
|||
bool PrioritizedEliminator<SparseModelType>::filterPredecessor(storm::storage::sparse::state_type const& state) { |
|||
assert(false); |
|||
} |
|||
|
|||
template<typename SparseModelType> |
|||
bool PrioritizedEliminator<SparseModelType>::isFilterPredecessor() const { |
|||
return false; |
|||
} |
|||
|
|||
|
|||
template class PrioritizedEliminator<storm::models::sparse::Dtmc<double>>; |
|||
|
|||
#ifdef STORM_HAVE_CARL
|
|||
template class PrioritizedEliminator<storm::models::sparse::Dtmc<storm::RationalFunction>>; |
|||
#endif
|
|||
|
|||
} // namespace stateelimination
|
|||
} // namespace storage
|
|||
} // namespace storm
|
@ -0,0 +1,34 @@ |
|||
#ifndef STORM_SOLVER_STATEELIMINATION_PRIORITIZEDELIMINATOR_H_ |
|||
#define STORM_SOLVER_STATEELIMINATION_PRIORITIZEDELIMINATOR_H_ |
|||
|
|||
#include "src/solver/stateelimination/StateEliminator.h" |
|||
|
|||
namespace storm { |
|||
namespace solver { |
|||
namespace stateelimination { |
|||
|
|||
template<typename SparseModelType> |
|||
class PrioritizedEliminator : public StateEliminator<SparseModelType> { |
|||
|
|||
typedef typename SparseModelType::ValueType ValueType; |
|||
|
|||
public: |
|||
PrioritizedEliminator(storm::storage::FlexibleSparseMatrix<ValueType>& transitionMatrix, storm::storage::FlexibleSparseMatrix<ValueType>& backwardTransitions, storm::modelchecker::StatePriorityQueue<ValueType> priorityQueue, std::vector<ValueType>& stateValues); |
|||
|
|||
// Instantiaton of Virtual methods |
|||
void updateValue(storm::storage::sparse::state_type const& state, ValueType const& loopProbability) override; |
|||
void updatePredecessor(storm::storage::sparse::state_type const& predecessor, ValueType const& probability, storm::storage::sparse::state_type const& state) override; |
|||
void updatePriority(storm::storage::sparse::state_type const& state) override; |
|||
bool filterPredecessor(storm::storage::sparse::state_type const& state) override; |
|||
bool isFilterPredecessor() const override; |
|||
|
|||
private: |
|||
storm::modelchecker::StatePriorityQueue<ValueType>& priorityQueue; |
|||
std::vector<ValueType>& stateValues; |
|||
}; |
|||
|
|||
} // namespace stateelimination |
|||
} // namespace storage |
|||
} // namespace storm |
|||
|
|||
#endif // STORM_SOLVER_STATEELIMINATION_PRIORITIZEDELIMINATOR_H_ |
@ -0,0 +1,259 @@ |
|||
#include "src/solver/stateelimination/StateEliminator.h"
|
|||
|
|||
#include "src/storage/BitVector.h"
|
|||
#include "src/adapters/CarlAdapter.h"
|
|||
#include "src/utility/constants.h"
|
|||
#include "src/utility/macros.h"
|
|||
#include "src/exceptions/InvalidStateException.h"
|
|||
|
|||
namespace storm { |
|||
namespace solver { |
|||
namespace stateelimination { |
|||
|
|||
template<typename SparseModelType> |
|||
StateEliminator<SparseModelType>::StateEliminator(storm::storage::FlexibleSparseMatrix<ValueType>& transitionMatrix, storm::storage::FlexibleSparseMatrix<ValueType>& backwardTransitions) : transitionMatrix(transitionMatrix), backwardTransitions(backwardTransitions) { |
|||
} |
|||
|
|||
template<typename SparseModelType> |
|||
void StateEliminator<SparseModelType>::eliminateState(storm::storage::sparse::state_type state, bool removeForwardTransitions, storm::storage::BitVector predecessorConstraint) { |
|||
|
|||
STORM_LOG_TRACE("Eliminating state " << state << "."); |
|||
|
|||
// Start by finding loop probability.
|
|||
bool hasSelfLoop = false; |
|||
ValueType loopProbability = storm::utility::zero<ValueType>(); |
|||
FlexibleRowType& currentStateSuccessors = transitionMatrix.getRow(state); |
|||
for (auto entryIt = currentStateSuccessors.begin(), entryIte = currentStateSuccessors.end(); entryIt != entryIte; ++entryIt) { |
|||
if (entryIt->getColumn() >= state) { |
|||
if (entryIt->getColumn() == state) { |
|||
loopProbability = entryIt->getValue(); |
|||
hasSelfLoop = true; |
|||
|
|||
// If we do not clear the forward transitions completely, we need to remove the self-loop,
|
|||
// because we scale all the other outgoing transitions with it anyway.
|
|||
if (!removeForwardTransitions) { |
|||
currentStateSuccessors.erase(entryIt); |
|||
} |
|||
} |
|||
break; |
|||
} |
|||
} |
|||
|
|||
// Scale all entries in this row with (1 / (1 - loopProbability)) only in case there was a self-loop.
|
|||
STORM_LOG_TRACE((hasSelfLoop ? "State has self-loop." : "State does not have a self-loop.")); |
|||
if (hasSelfLoop) { |
|||
STORM_LOG_ASSERT(loopProbability != storm::utility::one<ValueType>(), "Must not eliminate state with probability 1 self-loop."); |
|||
loopProbability = storm::utility::simplify(storm::utility::one<ValueType>() / (storm::utility::one<ValueType>() - loopProbability)); |
|||
for (auto& entry : transitionMatrix.getRow(state)) { |
|||
// Only scale the non-diagonal entries.
|
|||
if (entry.getColumn() != state) { |
|||
entry.setValue(storm::utility::simplify(entry.getValue() * loopProbability)); |
|||
} |
|||
} |
|||
updateValue(state, loopProbability); |
|||
} |
|||
|
|||
// Now connect the predecessors of the state being eliminated with its successors.
|
|||
FlexibleRowType& currentStatePredecessors = backwardTransitions.getRow(state); |
|||
|
|||
// In case we have a constrained elimination, we need to keep track of the new predecessors.
|
|||
FlexibleRowType newCurrentStatePredecessors; |
|||
|
|||
std::vector<FlexibleRowType> newBackwardProbabilities(currentStateSuccessors.size()); |
|||
for (auto& backwardProbabilities : newBackwardProbabilities) { |
|||
backwardProbabilities.reserve(currentStatePredecessors.size()); |
|||
} |
|||
|
|||
// Now go through the predecessors and eliminate the ones (satisfying the constraint if given).
|
|||
for (auto const& predecessorEntry : currentStatePredecessors) { |
|||
uint_fast64_t predecessor = predecessorEntry.getColumn(); |
|||
STORM_LOG_TRACE("Found predecessor " << predecessor << "."); |
|||
|
|||
// Skip the state itself as one of its predecessors.
|
|||
if (predecessor == state) { |
|||
assert(hasSelfLoop); |
|||
continue; |
|||
} |
|||
|
|||
// Skip the state if the elimination is constrained, but the predecessor is not in the constraint.
|
|||
if (isFilterPredecessor() && !filterPredecessor(predecessor)) { |
|||
newCurrentStatePredecessors.emplace_back(predecessorEntry); |
|||
STORM_LOG_TRACE("Not eliminating predecessor " << predecessor << ", because it does not fit the filter."); |
|||
continue; |
|||
} |
|||
STORM_LOG_TRACE("Eliminating predecessor " << predecessor << "."); |
|||
|
|||
// First, find the probability with which the predecessor can move to the current state, because
|
|||
// the forward probabilities of the state to be eliminated need to be scaled with this factor.
|
|||
FlexibleRowType& predecessorForwardTransitions = transitionMatrix.getRow(predecessor); |
|||
FlexibleRowIterator multiplyElement = std::find_if(predecessorForwardTransitions.begin(), predecessorForwardTransitions.end(), [&](storm::storage::MatrixEntry<typename storm::storage::FlexibleSparseMatrix<ValueType>::index_type, typename storm::storage::FlexibleSparseMatrix<ValueType>::value_type> const& a) { return a.getColumn() == state; }); |
|||
|
|||
// Make sure we have found the probability and set it to zero.
|
|||
STORM_LOG_THROW(multiplyElement != predecessorForwardTransitions.end(), storm::exceptions::InvalidStateException, "No probability for successor found."); |
|||
ValueType multiplyFactor = multiplyElement->getValue(); |
|||
multiplyElement->setValue(storm::utility::zero<ValueType>()); |
|||
|
|||
// At this point, we need to update the (forward) transitions of the predecessor.
|
|||
FlexibleRowIterator first1 = predecessorForwardTransitions.begin(); |
|||
FlexibleRowIterator last1 = predecessorForwardTransitions.end(); |
|||
FlexibleRowIterator first2 = currentStateSuccessors.begin(); |
|||
FlexibleRowIterator last2 = currentStateSuccessors.end(); |
|||
|
|||
FlexibleRowType newSuccessors; |
|||
newSuccessors.reserve((last1 - first1) + (last2 - first2)); |
|||
std::insert_iterator<FlexibleRowType> result(newSuccessors, newSuccessors.end()); |
|||
|
|||
uint_fast64_t successorOffsetInNewBackwardTransitions = 0; |
|||
// Now we merge the two successor lists. (Code taken from std::set_union and modified to suit our needs).
|
|||
for (; first1 != last1; ++result) { |
|||
// Skip the transitions to the state that is currently being eliminated.
|
|||
if (first1->getColumn() == state || (first2 != last2 && first2->getColumn() == state)) { |
|||
if (first1->getColumn() == state) { |
|||
++first1; |
|||
} |
|||
if (first2 != last2 && first2->getColumn() == state) { |
|||
++first2; |
|||
} |
|||
continue; |
|||
} |
|||
|
|||
if (first2 == last2) { |
|||
std::copy_if(first1, last1, result, [&] (storm::storage::MatrixEntry<typename storm::storage::FlexibleSparseMatrix<ValueType>::index_type, typename storm::storage::FlexibleSparseMatrix<ValueType>::value_type> const& a) { return a.getColumn() != state; } ); |
|||
break; |
|||
} |
|||
if (first2->getColumn() < first1->getColumn()) { |
|||
auto successorEntry = storm::utility::simplify(std::move(*first2 * multiplyFactor)); |
|||
*result = successorEntry; |
|||
newBackwardProbabilities[successorOffsetInNewBackwardTransitions].emplace_back(predecessor, successorEntry.getValue()); |
|||
// std::cout << "(1) adding " << first2->getColumn() << " -> " << newBackwardProbabilities[successorOffsetInNewBackwardTransitions].back() << "[" << successorOffsetInNewBackwardTransitions << "]" << std::endl;
|
|||
++first2; |
|||
++successorOffsetInNewBackwardTransitions; |
|||
} else if (first1->getColumn() < first2->getColumn()) { |
|||
*result = *first1; |
|||
++first1; |
|||
} else { |
|||
auto probability = storm::utility::simplify(first1->getValue() + storm::utility::simplify(multiplyFactor * first2->getValue())); |
|||
*result = storm::storage::MatrixEntry<typename storm::storage::FlexibleSparseMatrix<ValueType>::index_type, typename storm::storage::FlexibleSparseMatrix<ValueType>::value_type>(first1->getColumn(), probability); |
|||
newBackwardProbabilities[successorOffsetInNewBackwardTransitions].emplace_back(predecessor, probability); |
|||
// std::cout << "(2) adding " << first2->getColumn() << " -> " << newBackwardProbabilities[successorOffsetInNewBackwardTransitions].back() << "[" << successorOffsetInNewBackwardTransitions << "]" << std::endl;
|
|||
++first1; |
|||
++first2; |
|||
++successorOffsetInNewBackwardTransitions; |
|||
} |
|||
} |
|||
for (; first2 != last2; ++first2) { |
|||
if (first2->getColumn() != state) { |
|||
auto stateProbability = storm::utility::simplify(std::move(*first2 * multiplyFactor)); |
|||
*result = stateProbability; |
|||
newBackwardProbabilities[successorOffsetInNewBackwardTransitions].emplace_back(predecessor, stateProbability.getValue()); |
|||
// std::cout << "(3) adding " << first2->getColumn() << " -> " << newBackwardProbabilities[successorOffsetInNewBackwardTransitions].back() << "[" << successorOffsetInNewBackwardTransitions << "]" << std::endl;
|
|||
++successorOffsetInNewBackwardTransitions; |
|||
} |
|||
} |
|||
|
|||
// Now move the new transitions in place.
|
|||
predecessorForwardTransitions = std::move(newSuccessors); |
|||
STORM_LOG_TRACE("Fixed new next-state probabilities of predecessor state " << predecessor << "."); |
|||
|
|||
updatePredecessor(predecessor, multiplyFactor, state); |
|||
|
|||
STORM_LOG_TRACE("Updating priority of predecessor."); |
|||
updatePriority(predecessor); |
|||
} |
|||
|
|||
// Finally, we need to add the predecessor to the set of predecessors of every successor.
|
|||
uint_fast64_t successorOffsetInNewBackwardTransitions = 0; |
|||
for (auto const& successorEntry : currentStateSuccessors) { |
|||
if (successorEntry.getColumn() == state) { |
|||
continue; |
|||
} |
|||
|
|||
FlexibleRowType& successorBackwardTransitions = backwardTransitions.getRow(successorEntry.getColumn()); |
|||
// std::cout << "old backward trans of " << successorEntry.getColumn() << std::endl;
|
|||
// for (auto const& trans : successorBackwardTransitions) {
|
|||
// std::cout << trans << std::endl;
|
|||
// }
|
|||
|
|||
// Delete the current state as a predecessor of the successor state only if we are going to remove the
|
|||
// current state's forward transitions.
|
|||
if (removeForwardTransitions) { |
|||
FlexibleRowIterator elimIt = std::find_if(successorBackwardTransitions.begin(), successorBackwardTransitions.end(), [&](storm::storage::MatrixEntry<typename storm::storage::FlexibleSparseMatrix<ValueType>::index_type, typename storm::storage::FlexibleSparseMatrix<ValueType>::value_type> const& a) { return a.getColumn() == state; }); |
|||
STORM_LOG_ASSERT(elimIt != successorBackwardTransitions.end(), "Expected a proper backward transition from " << successorEntry.getColumn() << " to " << state << ", but found none."); |
|||
successorBackwardTransitions.erase(elimIt); |
|||
} |
|||
|
|||
FlexibleRowIterator first1 = successorBackwardTransitions.begin(); |
|||
FlexibleRowIterator last1 = successorBackwardTransitions.end(); |
|||
FlexibleRowIterator first2 = newBackwardProbabilities[successorOffsetInNewBackwardTransitions].begin(); |
|||
FlexibleRowIterator last2 = newBackwardProbabilities[successorOffsetInNewBackwardTransitions].end(); |
|||
|
|||
// std::cout << "adding backward trans " << successorEntry.getColumn() << "[" << successorOffsetInNewBackwardTransitions << "]" << std::endl;
|
|||
// for (auto const& trans : newBackwardProbabilities[successorOffsetInNewBackwardTransitions]) {
|
|||
// std::cout << trans << std::endl;
|
|||
// }
|
|||
|
|||
FlexibleRowType newPredecessors; |
|||
newPredecessors.reserve((last1 - first1) + (last2 - first2)); |
|||
std::insert_iterator<FlexibleRowType> result(newPredecessors, newPredecessors.end()); |
|||
|
|||
for (; first1 != last1; ++result) { |
|||
if (first2 == last2) { |
|||
std::copy(first1, last1, result); |
|||
break; |
|||
} |
|||
if (first2->getColumn() < first1->getColumn()) { |
|||
if (first2->getColumn() != state) { |
|||
*result = *first2; |
|||
} |
|||
++first2; |
|||
} else if (first1->getColumn() == first2->getColumn()) { |
|||
if (storm::modelchecker::estimateComplexity(first1->getValue()) > storm::modelchecker::estimateComplexity(first2->getValue())) { |
|||
*result = *first1; |
|||
} else { |
|||
*result = *first2; |
|||
} |
|||
++first1; |
|||
++first2; |
|||
} else { |
|||
*result = *first1; |
|||
++first1; |
|||
} |
|||
} |
|||
if (isFilterPredecessor()) { |
|||
std::copy_if(first2, last2, result, [&] (storm::storage::MatrixEntry<typename storm::storage::FlexibleSparseMatrix<ValueType>::index_type, typename storm::storage::FlexibleSparseMatrix<ValueType>::value_type> const& a) { return a.getColumn() != state && filterPredecessor(a.getColumn()); }); |
|||
} else { |
|||
std::copy_if(first2, last2, result, [&] (storm::storage::MatrixEntry<typename storm::storage::FlexibleSparseMatrix<ValueType>::index_type, typename storm::storage::FlexibleSparseMatrix<ValueType>::value_type> const& a) { return a.getColumn() != state; }); |
|||
} |
|||
// Now move the new predecessors in place.
|
|||
successorBackwardTransitions = std::move(newPredecessors); |
|||
// std::cout << "new backward trans of " << successorEntry.getColumn() << std::endl;
|
|||
// for (auto const& trans : successorBackwardTransitions) {
|
|||
// std::cout << trans << std::endl;
|
|||
// }
|
|||
++successorOffsetInNewBackwardTransitions; |
|||
} |
|||
STORM_LOG_TRACE("Fixed predecessor lists of successor states."); |
|||
|
|||
if (removeForwardTransitions) { |
|||
// Clear the eliminated row to reduce memory consumption.
|
|||
currentStateSuccessors.clear(); |
|||
currentStateSuccessors.shrink_to_fit(); |
|||
} |
|||
if (isFilterPredecessor()) { |
|||
currentStatePredecessors = std::move(newCurrentStatePredecessors); |
|||
} else { |
|||
currentStatePredecessors.clear(); |
|||
currentStatePredecessors.shrink_to_fit(); |
|||
} |
|||
|
|||
} |
|||
|
|||
template class StateEliminator<storm::models::sparse::Dtmc<double>>; |
|||
|
|||
#ifdef STORM_HAVE_CARL
|
|||
template class StateEliminator<storm::models::sparse::Dtmc<storm::RationalFunction>>; |
|||
#endif
|
|||
|
|||
} // namespace stateelimination
|
|||
} // namespace storage
|
|||
} // namespace storm
|
@ -0,0 +1,42 @@ |
|||
#ifndef STORM_SOLVER_STATEELIMINATION_STATEELIMINATOR_H_ |
|||
#define STORM_SOLVER_STATEELIMINATION_STATEELIMINATOR_H_ |
|||
|
|||
#include "src/storage/FlexibleSparseMatrix.h" |
|||
#include "src/storage/SparseMatrix.h" |
|||
#include "src/storage/sparse/StateType.h" |
|||
#include "src/modelchecker/reachability/SparseDtmcEliminationModelChecker.h" |
|||
|
|||
namespace storm { |
|||
namespace solver { |
|||
namespace stateelimination { |
|||
|
|||
template<typename SparseModelType> |
|||
class StateEliminator { |
|||
typedef typename SparseModelType::ValueType ValueType; |
|||
typedef typename storm::storage::FlexibleSparseMatrix<ValueType>::row_type FlexibleRowType; |
|||
typedef typename FlexibleRowType::iterator FlexibleRowIterator; |
|||
|
|||
public: |
|||
StateEliminator(storm::storage::FlexibleSparseMatrix<ValueType>& transitionMatrix, storm::storage::FlexibleSparseMatrix<ValueType>& backwardTransitions); |
|||
|
|||
void eliminateState(storm::storage::sparse::state_type state, bool removeForwardTransitions, storm::storage::BitVector predecessorConstraint = storm::storage::BitVector()); |
|||
|
|||
// Virtual methods for base classes to distinguish between different state elimination approaches |
|||
virtual void updateValue(storm::storage::sparse::state_type const& state, ValueType const& loopProbability) = 0; |
|||
virtual void updatePredecessor(storm::storage::sparse::state_type const& predecessor, ValueType const& probability, storm::storage::sparse::state_type const& state) = 0; |
|||
virtual void updatePriority(storm::storage::sparse::state_type const& state) = 0; |
|||
virtual bool filterPredecessor(storm::storage::sparse::state_type const& state) = 0; |
|||
virtual bool isFilterPredecessor() const = 0; |
|||
|
|||
|
|||
protected: |
|||
storm::storage::FlexibleSparseMatrix<ValueType>& transitionMatrix; |
|||
storm::storage::FlexibleSparseMatrix<ValueType>& backwardTransitions; |
|||
|
|||
}; |
|||
|
|||
} // namespace stateelimination |
|||
} // namespace storage |
|||
} // namespace storm |
|||
|
|||
#endif // STORM_SOLVER_STATEELIMINATION_STATEELIMINATOR_H_ |
Write
Preview
Loading…
Cancel
Save
Reference in new issue