#include "storm/models/symbolic/Model.h" #include #include "storm/exceptions/IllegalArgumentException.h" #include "storm/exceptions/InvalidOperationException.h" #include "storm/adapters/AddExpressionAdapter.h" #include "storm/storage/dd/DdManager.h" #include "storm/storage/dd/Add.h" #include "storm/storage/dd/Bdd.h" #include "storm/models/symbolic/StandardRewardModel.h" #include "storm-config.h" #include "storm/adapters/CarlAdapter.h" namespace storm { namespace models { namespace symbolic { template Model::Model(storm::models::ModelType const& modelType, std::shared_ptr> manager, storm::dd::Bdd reachableStates, storm::dd::Bdd initialStates, storm::dd::Bdd deadlockStates, storm::dd::Add transitionMatrix, std::set const& rowVariables, std::shared_ptr> rowExpressionAdapter, std::set const& columnVariables, std::shared_ptr> columnExpressionAdapter, std::vector> const& rowColumnMetaVariablePairs, std::map labelToExpressionMap, std::unordered_map const& rewardModels) : ModelBase(modelType), manager(manager), reachableStates(reachableStates), initialStates(initialStates), deadlockStates(deadlockStates), transitionMatrix(transitionMatrix), rowVariables(rowVariables), rowExpressionAdapter(rowExpressionAdapter), columnVariables(columnVariables), columnExpressionAdapter(columnExpressionAdapter), rowColumnMetaVariablePairs(rowColumnMetaVariablePairs), labelToExpressionMap(labelToExpressionMap), rewardModels(rewardModels) { // Intentionally left empty. } template uint_fast64_t Model::getNumberOfStates() const { return reachableStates.getNonZeroCount(); } template uint_fast64_t Model::getNumberOfTransitions() const { return transitionMatrix.getNonZeroCount(); } template storm::dd::DdManager const& Model::getManager() const { return *manager; } template storm::dd::DdManager& Model::getManager() { return *manager; } template std::shared_ptr> const& Model::getManagerAsSharedPointer() const { return manager; } template storm::dd::Bdd const& Model::getReachableStates() const { return reachableStates; } template storm::dd::Bdd const& Model::getInitialStates() const { return initialStates; } template storm::dd::Bdd const& Model::getDeadlockStates() const { return deadlockStates; } template storm::dd::Bdd Model::getStates(std::string const& label) const { STORM_LOG_THROW(labelToExpressionMap.find(label) != labelToExpressionMap.end(), storm::exceptions::IllegalArgumentException, "The label " << label << " is invalid for the labeling of the model."); return this->getStates(labelToExpressionMap.at(label)); } template storm::dd::Bdd Model::getStates(storm::expressions::Expression const& expression) const { if (expression.isTrue()) { return this->getReachableStates(); } else if (expression.isFalse()) { return manager->getBddZero(); } STORM_LOG_THROW(rowExpressionAdapter != nullptr, storm::exceptions::InvalidOperationException, "Cannot create BDD for expression without expression adapter."); return rowExpressionAdapter->translateExpression(expression).toBdd() && this->reachableStates; } template bool Model::hasLabel(std::string const& label) const { auto labelIt = labelToExpressionMap.find(label); if (labelIt != labelToExpressionMap.end()) { return true; } else { return label == "init" || label == "deadlock"; } } template storm::dd::Add const& Model::getTransitionMatrix() const { return transitionMatrix; } template storm::dd::Add& Model::getTransitionMatrix() { return transitionMatrix; } template storm::dd::Bdd Model::getQualitativeTransitionMatrix() const { return this->getTransitionMatrix().notZero(); } template std::set const& Model::getRowVariables() const { return rowVariables; } template std::set const& Model::getColumnVariables() const { return columnVariables; } template std::vector> const& Model::getRowColumnMetaVariablePairs() const { return rowColumnMetaVariablePairs; } template void Model::setTransitionMatrix(storm::dd::Add const& transitionMatrix) { this->transitionMatrix = transitionMatrix; } template std::map const& Model::getLabelToExpressionMap() const { return labelToExpressionMap; } template storm::dd::Add Model::getRowColumnIdentity() const { storm::dd::Add result = this->getManager().template getAddOne(); for (auto const& pair : this->getRowColumnMetaVariablePairs()) { result *= this->getManager().template getIdentity(pair.first).equals(this->getManager().template getIdentity(pair.second)).template toAdd(); result *= this->getManager().getRange(pair.first).template toAdd() * this->getManager().getRange(pair.second).template toAdd(); } return result; } template bool Model::hasRewardModel(std::string const& rewardModelName) const { return this->rewardModels.find(rewardModelName) != this->rewardModels.end(); } template typename Model::RewardModelType const& Model::getRewardModel(std::string const& rewardModelName) const { auto it = this->rewardModels.find(rewardModelName); if (it == this->rewardModels.end()) { if (rewardModelName.empty()) { if (this->hasUniqueRewardModel()) { return this->getUniqueRewardModel(); } else { STORM_LOG_THROW(false, storm::exceptions::IllegalArgumentException, "Unable to refer to default reward model, because there is no default model or it is not unique."); } } else { STORM_LOG_THROW(false, storm::exceptions::IllegalArgumentException, "The requested reward model '" << rewardModelName << "' does not exist."); } } return it->second; } template typename Model::RewardModelType const& Model::getUniqueRewardModel() const { STORM_LOG_THROW(this->hasUniqueRewardModel(), storm::exceptions::InvalidOperationException, "Cannot retrieve unique reward model, because there is no unique one."); return this->rewardModels.cbegin()->second; } template typename Model::RewardModelType& Model::getUniqueRewardModel() { STORM_LOG_THROW(this->hasUniqueRewardModel(), storm::exceptions::InvalidOperationException, "Cannot retrieve unique reward model, because there is no unique one."); return this->rewardModels.begin()->second; } template bool Model::hasUniqueRewardModel() const { return this->rewardModels.size() == 1; } template bool Model::hasRewardModel() const { return !this->rewardModels.empty(); } template std::unordered_map::RewardModelType> const& Model::getRewardModels() const { return this->rewardModels; } template void Model::printModelInformationToStream(std::ostream& out) const { this->printModelInformationHeaderToStream(out); this->printModelInformationFooterToStream(out); } template std::vector Model::getLabels() const { std::vector labels; for(auto const& entry : labelToExpressionMap) { labels.push_back(entry.first); } return labels; } template void Model::printModelInformationHeaderToStream(std::ostream& out) const { out << "-------------------------------------------------------------- " << std::endl; out << "Model type: \t" << this->getType() << " (symbolic)" << std::endl; out << "States: \t" << this->getNumberOfStates() << " (" << reachableStates.getNodeCount() << " nodes)" << std::endl; out << "Transitions: \t" << this->getNumberOfTransitions() << " (" << transitionMatrix.getNodeCount() << " nodes)" << std::endl; } template void Model::printModelInformationFooterToStream(std::ostream& out) const { this->printRewardModelsInformationToStream(out); this->printDdVariableInformationToStream(out); out << std::endl; out << "Labels: \t" << this->labelToExpressionMap.size() << std::endl; for (auto const& label : labelToExpressionMap) { out << " * " << label.first << std::endl; } out << "-------------------------------------------------------------- " << std::endl; } template void Model::printRewardModelsInformationToStream(std::ostream& out) const { if (this->rewardModels.size()) { std::vector rewardModelNames; std::for_each(this->rewardModels.cbegin(), this->rewardModels.cend(), [&rewardModelNames] (typename std::pair const& nameRewardModelPair) { if (nameRewardModelPair.first.empty()) { rewardModelNames.push_back("(default)"); } else { rewardModelNames.push_back(nameRewardModelPair.first); } }); out << "Reward Models: " << boost::join(rewardModelNames, ", ") << std::endl; } else { out << "Reward Models: none" << std::endl; } } template void Model::printDdVariableInformationToStream(std::ostream& out) const { uint_fast64_t rowVariableCount = 0; for (auto const& metaVariable : this->rowVariables) { rowVariableCount += this->getManager().getMetaVariable(metaVariable).getNumberOfDdVariables(); } uint_fast64_t columnVariableCount = 0; for (auto const& metaVariable : this->columnVariables) { columnVariableCount += this->getManager().getMetaVariable(metaVariable).getNumberOfDdVariables(); } out << "Variables: \t" << "rows: " << this->rowVariables.size() << " meta variables (" << rowVariableCount << " DD variables)" << ", columns: " << this->columnVariables.size() << " meta variables (" << columnVariableCount << " DD variables)"; } template bool Model::isSymbolicModel() const { return true; } // Explicitly instantiate the template class. template class Model; template class Model; #ifdef STORM_HAVE_CARL template class Model; #endif } // namespace symbolic } // namespace models } // namespace storm