19namespace bisimulation {
21template<storm::dd::DdType DdType,
typename ValueType,
typename ExportValueType>
24 : model(model), quotientFormat(quotientFormat) {
26 STORM_LOG_ERROR(
"Only DD-based partial quotient extraction is currently supported. Switching to DD-based extraction.");
31template<storm::dd::DdType DdType,
typename ValueType,
typename ExportValueType>
34 auto start = std::chrono::high_resolution_clock::now();
35 std::shared_ptr<storm::models::Model<ExportValueType>> result;
38 "Only DD-based partial quotient extraction is currently supported.");
39 result = extractDdQuotient(partition, preservationInformation);
40 auto end = std::chrono::high_resolution_clock::now();
41 STORM_LOG_TRACE(
"Quotient extraction completed in " << std::chrono::duration_cast<std::chrono::milliseconds>(end - start).count() <<
"ms.");
43 STORM_LOG_THROW(result, storm::exceptions::NotSupportedException,
"Quotient could not be extracted.");
48template<storm::dd::DdType DdType,
typename ValueType,
typename ExportValueType>
51 auto modelType = model.getType();
56 "Mismatching partition.");
58 std::set<storm::expressions::Variable> blockVariableSet = {partition.
getBlockVariable()};
60 std::vector<std::pair<storm::expressions::Variable, storm::expressions::Variable>> blockMetaVariablePairs = {
65 auto start = std::chrono::high_resolution_clock::now();
66 partitionAsBdd = partitionAsBdd.
renameVariables(model.getColumnVariables(), model.getRowVariables());
68 storm::dd::Bdd<DdType> initialStates = (model.getInitialStates() && partitionAsBdd).existsAbstract(model.getRowVariables());
70 std::map<std::string, storm::dd::Bdd<DdType>> preservedLabelBdds;
71 for (
auto const& label : preservationInformation.getLabels()) {
72 preservedLabelBdds.emplace(label, (model.getStates(label) && partitionAsBdd).existsAbstract(model.getRowVariables()));
74 for (
auto const& expression : preservationInformation.getExpressions()) {
75 std::stringstream stream;
77 std::string expressionAsString = stream.str();
79 auto it = preservedLabelBdds.find(expressionAsString);
80 if (it != preservedLabelBdds.end()) {
81 STORM_LOG_WARN(
"Duplicate label '" << expressionAsString <<
"', dropping second label definition.");
83 preservedLabelBdds.emplace(stream.str(), (model.getStates(expression) && partitionAsBdd).existsAbstract(model.getRowVariables()));
86 auto end = std::chrono::high_resolution_clock::now();
87 STORM_LOG_TRACE(
"Quotient labels extracted in " << std::chrono::duration_cast<std::chrono::milliseconds>(end - start).
count() <<
"ms.");
89 start = std::chrono::high_resolution_clock::now();
90 std::set<storm::expressions::Variable> blockAndRowVariables;
91 std::set_union(blockVariableSet.begin(), blockVariableSet.end(), model.getRowVariables().begin(), model.getRowVariables().end(),
92 std::inserter(blockAndRowVariables, blockAndRowVariables.end()));
93 std::set<storm::expressions::Variable> blockPrimeAndColumnVariables;
94 std::set_union(blockPrimeVariableSet.begin(), blockPrimeVariableSet.end(), model.getColumnVariables().begin(), model.getColumnVariables().end(),
95 std::inserter(blockPrimeAndColumnVariables, blockPrimeAndColumnVariables.end()));
98 partitionAsAdd.
renameVariables(blockAndRowVariables, blockPrimeAndColumnVariables), model.getColumnVariables());
100 quotientTransitionMatrix = quotientTransitionMatrix * partitionAsAdd;
101 end = std::chrono::high_resolution_clock::now();
104 if (std::is_same<ValueType, storm::RationalNumber>::value) {
105 STORM_LOG_ASSERT(quotientTransitionMatrix.
greater(storm::utility::one<ValueType>()).isZero(),
"Illegal entries in quotient matrix.");
107 STORM_LOG_ASSERT(quotientTransitionMatrix.
greater(storm::utility::one<ValueType>() + storm::utility::convertNumber<ValueType>(1e-6)).isZero(),
108 "Illegal entries in quotient matrix.");
111 STORM_LOG_TRACE(
"Quotient transition matrix extracted in " << std::chrono::duration_cast<std::chrono::milliseconds>(end - start).
count() <<
"ms.");
114 std::set<storm::expressions::Variable> nonSourceVariables;
115 std::set_union(blockPrimeVariableSet.begin(), blockPrimeVariableSet.end(), model.getRowVariables().begin(), model.getRowVariables().end(),
116 std::inserter(nonSourceVariables, nonSourceVariables.begin()));
119 start = std::chrono::high_resolution_clock::now();
120 std::unordered_map<std::string, storm::models::symbolic::StandardRewardModel<DdType, ValueType>> quotientRewardModels;
121 for (
auto const& rewardModelName : preservationInformation.getRewardModelNames()) {
122 auto const& rewardModel = model.getRewardModel(rewardModelName);
124 boost::optional<storm::dd::Add<DdType, ValueType>> quotientStateRewards;
125 if (rewardModel.hasStateRewards()) {
126 quotientStateRewards = rewardModel.getStateRewardVector() * partitionAsAdd;
129 boost::optional<storm::dd::Add<DdType, ValueType>> quotientStateActionRewards;
130 if (rewardModel.hasStateActionRewards()) {
131 quotientStateActionRewards = rewardModel.getStateActionRewardVector() * partitionAsAdd;
135 quotientStateRewards, quotientStateActionRewards, boost::none));
137 end = std::chrono::high_resolution_clock::now();
138 STORM_LOG_TRACE(
"Reward models extracted in " << std::chrono::duration_cast<std::chrono::milliseconds>(end - start).
count() <<
"ms.");
140 std::shared_ptr<storm::models::symbolic::Model<DdType, ValueType>> result;
142 result = std::make_shared<storm::models::symbolic::Mdp<DdType, ValueType>>(
143 model.getManager().asSharedPointer(), reachableStates, initialStates, deadlockStates, quotientTransitionMatrix, blockVariableSet,
144 blockPrimeVariableSet, blockMetaVariablePairs, model.getRowVariables(), preservedLabelBdds, quotientRewardModels);
146 std::set<storm::expressions::Variable> allNondeterminismVariables;
147 std::set_union(model.getRowVariables().begin(), model.getRowVariables().end(), model.getNondeterminismVariables().begin(),
148 model.getNondeterminismVariables().end(), std::inserter(allNondeterminismVariables, allNondeterminismVariables.begin()));
150 result = std::make_shared<storm::models::symbolic::StochasticTwoPlayerGame<DdType, ValueType>>(
151 model.getManager().asSharedPointer(), reachableStates, initialStates, deadlockStates, quotientTransitionMatrix, blockVariableSet,
152 blockPrimeVariableSet, blockMetaVariablePairs, model.getRowVariables(), model.getNondeterminismVariables(), allNondeterminismVariables,
153 preservedLabelBdds, quotientRewardModels);
155 STORM_LOG_THROW(
false, storm::exceptions::NotSupportedException,
"Unsupported quotient type.");
158 return result->template toValueType<ExportValueType>();
160 STORM_LOG_THROW(
false, storm::exceptions::NotSupportedException,
"Cannot extract partial quotient for this model type.");
164template class PartialQuotientExtractor<storm::dd::DdType::CUDD, double>;
165template class PartialQuotientExtractor<storm::dd::DdType::Sylvan, double>;
167#ifdef STORM_HAVE_CARL
168template class PartialQuotientExtractor<storm::dd::DdType::Sylvan, storm::RationalNumber>;
169template class PartialQuotientExtractor<storm::dd::DdType::Sylvan, storm::RationalNumber, double>;
170template class PartialQuotientExtractor<storm::dd::DdType::Sylvan, storm::RationalFunction>;
Bdd< LibraryType > greater(Add< LibraryType, ValueType > const &other) const
Retrieves the function that maps all evaluations to one whose function value in the first ADD are gre...
Add< LibraryType, ValueType > renameVariables(std::set< storm::expressions::Variable > const &from, std::set< storm::expressions::Variable > const &to) const
Renames the given meta variables in the ADD.
Add< LibraryType, ValueType > multiplyMatrix(Add< LibraryType, ValueType > const &otherMatrix, std::set< storm::expressions::Variable > const &summationMetaVariables) const
Multiplies the current ADD (representing a matrix) with the given matrix by summing over the given me...
Bdd< LibraryType > notZero() const
Computes a BDD that represents the function in which all assignments with a function value unequal to...
Bdd< LibraryType > existsAbstract(std::set< storm::expressions::Variable > const &metaVariables) const
Existentially abstracts from the given meta variables.
Bdd< LibraryType > renameVariables(std::set< storm::expressions::Variable > const &from, std::set< storm::expressions::Variable > const &to) const
Renames the given meta variables in the BDD.
storm::expressions::Variable const & getBlockVariable() const
storm::dd::Bdd< DdType > const & asBdd() const
storm::expressions::Variable const & getPrimedBlockVariable() const
storm::dd::Bdd< DdType > getStates() const
storm::dd::Add< DdType, ValueType > const & asAdd() const
uint64_t getNumberOfStates() const
Base class for all symbolic models.
#define STORM_LOG_WARN(message)
#define STORM_LOG_TRACE(message)
#define STORM_LOG_ERROR(message)
#define STORM_LOG_ASSERT(cond, message)
#define STORM_LOG_THROW(cond, exception, message)
std::pair< storm::RationalNumber, storm::RationalNumber > count(std::vector< storm::storage::BitVector > const &origSets, std::vector< storm::storage::BitVector > const &intersects, std::vector< storm::storage::BitVector > const &intersectsInfo, storm::RationalNumber val, bool plus, uint64_t remdepth)