diff --git a/src/builder/DdPrismModelBuilder.cpp b/src/builder/DdPrismModelBuilder.cpp index 1c1d599a5..4f6ab3871 100644 --- a/src/builder/DdPrismModelBuilder.cpp +++ b/src/builder/DdPrismModelBuilder.cpp @@ -190,18 +190,18 @@ namespace storm { }; template - DdPrismModelBuilder::Options::Options() : buildAllRewardModels(true), rewardModelsToBuild(), constantDefinitions(), buildAllLabels(true), labelsToBuild(), expressionLabels(), terminalStates() { + DdPrismModelBuilder::Options::Options() : buildAllRewardModels(true), rewardModelsToBuild(), constantDefinitions(), buildAllLabels(true), labelsToBuild(), expressionLabels(), terminalStates(), negatedTerminalStates() { // Intentionally left empty. } template - DdPrismModelBuilder::Options::Options(storm::logic::Formula const& formula) : buildAllRewardModels(false), rewardModelsToBuild(), constantDefinitions(), buildAllLabels(false), labelsToBuild(std::set()), expressionLabels(std::vector()), terminalStates() { + DdPrismModelBuilder::Options::Options(storm::logic::Formula const& formula) : buildAllRewardModels(false), rewardModelsToBuild(), constantDefinitions(), buildAllLabels(false), labelsToBuild(std::set()), expressionLabels(std::vector()), terminalStates(), negatedTerminalStates() { this->preserveFormula(formula); this->setTerminalStatesFromFormula(formula); } template - DdPrismModelBuilder::Options::Options(std::vector> const& formulas) : buildAllRewardModels(false), rewardModelsToBuild(), constantDefinitions(), buildAllLabels(false), labelsToBuild(), expressionLabels(), terminalStates() { + DdPrismModelBuilder::Options::Options(std::vector> const& formulas) : buildAllRewardModels(false), rewardModelsToBuild(), constantDefinitions(), buildAllLabels(false), labelsToBuild(), expressionLabels(), terminalStates(), negatedTerminalStates() { if (formulas.empty()) { this->buildAllRewardModels = true; this->buildAllLabels = true; @@ -221,6 +221,9 @@ namespace storm { if (terminalStates) { terminalStates.reset(); } + if (negatedTerminalStates) { + negatedTerminalStates.reset(); + } // If we are not required to build all reward models, we determine the reward models we need to build. if (!buildAllRewardModels) { @@ -261,10 +264,16 @@ namespace storm { this->setTerminalStatesFromFormula(sub); } } else if (formula.isUntilFormula()) { - storm::logic::Formula const& right = formula.asUntilFormula().getLeftSubformula(); + storm::logic::Formula const& right = formula.asUntilFormula().getRightSubformula(); if (right.isAtomicExpressionFormula() || right.isAtomicLabelFormula()) { this->setTerminalStatesFromFormula(right); } + storm::logic::Formula const& left = formula.asUntilFormula().getLeftSubformula(); + if (left.isAtomicExpressionFormula()) { + negatedTerminalStates = left.asAtomicExpressionFormula().getExpression(); + } else if (left.isAtomicLabelFormula()) { + negatedTerminalStates = left.asAtomicLabelFormula().getLabel(); + } } else if (formula.isProbabilityOperatorFormula()) { storm::logic::Formula const& sub = formula.asProbabilityOperatorFormula().getSubformula(); if (sub.isEventuallyFormula() || sub.isUntilFormula()) { @@ -1023,16 +1032,33 @@ namespace storm { storm::dd::Add stateActionDd = system.stateActionDd; // If we were asked to treat some states as terminal states, we cut away their transitions now. - if (options.terminalStates) { - storm::expressions::Expression terminalExpression; - if (options.terminalStates.get().type() == typeid(storm::expressions::Expression)) { - terminalExpression = boost::get(options.terminalStates.get()); - } else { - std::string const& labelName = boost::get(options.terminalStates.get()); - terminalExpression = preparedProgram.getLabelExpression(labelName); + if (options.terminalStates || options.negatedTerminalStates) { + storm::dd::Add terminalStatesAdd = generationInfo.manager->getAddZero(); + if (options.terminalStates) { + storm::expressions::Expression terminalExpression; + if (options.terminalStates.get().type() == typeid(storm::expressions::Expression)) { + terminalExpression = boost::get(options.terminalStates.get()); + } else { + std::string const& labelName = boost::get(options.terminalStates.get()); + terminalExpression = preparedProgram.getLabelExpression(labelName); + } + + STORM_LOG_TRACE("Making the states satisfying " << terminalExpression << " terminal."); + terminalStatesAdd = generationInfo.rowExpressionAdapter->translateExpression(terminalExpression); } - STORM_LOG_TRACE("Making the states satisfying " << terminalExpression << " terminal."); - storm::dd::Add terminalStatesAdd = generationInfo.rowExpressionAdapter->translateExpression(terminalExpression); + if (options.negatedTerminalStates) { + storm::expressions::Expression nonTerminalExpression; + if (options.negatedTerminalStates.get().type() == typeid(storm::expressions::Expression)) { + nonTerminalExpression = boost::get(options.negatedTerminalStates.get()); + } else { + std::string const& labelName = boost::get(options.terminalStates.get()); + nonTerminalExpression = preparedProgram.getLabelExpression(labelName); + } + + STORM_LOG_TRACE("Making the states *not* satisfying " << nonTerminalExpression << " terminal."); + terminalStatesAdd |= !generationInfo.rowExpressionAdapter->translateExpression(nonTerminalExpression); + } + transitionMatrix *= !terminalStatesAdd; } diff --git a/src/builder/DdPrismModelBuilder.h b/src/builder/DdPrismModelBuilder.h index 20f6e76f0..4669432a8 100644 --- a/src/builder/DdPrismModelBuilder.h +++ b/src/builder/DdPrismModelBuilder.h @@ -99,9 +99,13 @@ namespace storm { // An optional set of expressions for which labels need to be built. boost::optional> expressionLabels; - // An optional expression or label that characterizes the terminal states of the model. If this is set, - // the outgoing transitions of these states are replaced with a self-loop. + // An optional expression or label that (a subset of) characterizes the terminal states of the model. + // If this is set, the outgoing transitions of these states are replaced with a self-loop. boost::optional> terminalStates; + + // An optional expression or label whose negation characterizes (a subset of) the terminal states of the + // model. If this is set, the outgoing transitions of these states are replaced with a self-loop. + boost::optional> negatedTerminalStates; }; /*! diff --git a/src/builder/ExplicitPrismModelBuilder.cpp b/src/builder/ExplicitPrismModelBuilder.cpp index 7fa0043e9..9a9da5295 100644 --- a/src/builder/ExplicitPrismModelBuilder.cpp +++ b/src/builder/ExplicitPrismModelBuilder.cpp @@ -106,17 +106,17 @@ namespace storm { } template - ExplicitPrismModelBuilder::Options::Options() : buildCommandLabels(false), buildAllRewardModels(true), rewardModelsToBuild(), constantDefinitions(), buildAllLabels(true), labelsToBuild(), expressionLabels(), terminalStates() { + ExplicitPrismModelBuilder::Options::Options() : buildCommandLabels(false), buildAllRewardModels(true), rewardModelsToBuild(), constantDefinitions(), buildAllLabels(true), labelsToBuild(), expressionLabels(), terminalStates(), negatedTerminalStates() { // Intentionally left empty. } template - ExplicitPrismModelBuilder::Options::Options(storm::logic::Formula const& formula) : buildCommandLabels(false), buildAllRewardModels(false), rewardModelsToBuild(), constantDefinitions(), buildAllLabels(false), labelsToBuild(std::set()), expressionLabels(std::vector()), terminalStates() { + ExplicitPrismModelBuilder::Options::Options(storm::logic::Formula const& formula) : buildCommandLabels(false), buildAllRewardModels(false), rewardModelsToBuild(), constantDefinitions(), buildAllLabels(false), labelsToBuild(std::set()), expressionLabels(std::vector()), terminalStates(), negatedTerminalStates() { this->preserveFormula(formula); } template - ExplicitPrismModelBuilder::Options::Options(std::vector> const& formulas) : buildCommandLabels(false), buildAllRewardModels(false), rewardModelsToBuild(), constantDefinitions(), buildAllLabels(false), labelsToBuild(), expressionLabels(), terminalStates() { + ExplicitPrismModelBuilder::Options::Options(std::vector> const& formulas) : buildCommandLabels(false), buildAllRewardModels(false), rewardModelsToBuild(), constantDefinitions(), buildAllLabels(false), labelsToBuild(), expressionLabels(), terminalStates(), negatedTerminalStates() { if (formulas.empty()) { this->buildAllRewardModels = true; this->buildAllLabels = true; @@ -142,10 +142,16 @@ namespace storm { this->setTerminalStatesFromFormula(sub); } } else if (formula.isUntilFormula()) { - storm::logic::Formula const& right = formula.asUntilFormula().getLeftSubformula(); + storm::logic::Formula const& right = formula.asUntilFormula().getRightSubformula(); if (right.isAtomicExpressionFormula() || right.isAtomicLabelFormula()) { this->setTerminalStatesFromFormula(right); } + storm::logic::Formula const& left = formula.asUntilFormula().getLeftSubformula(); + if (left.isAtomicExpressionFormula()) { + negatedTerminalStates = left.asAtomicExpressionFormula().getExpression(); + } else if (left.isAtomicLabelFormula()) { + negatedTerminalStates = left.asAtomicLabelFormula().getLabel(); + } } else if (formula.isProbabilityOperatorFormula()) { storm::logic::Formula const& sub = formula.asProbabilityOperatorFormula().getSubformula(); if (sub.isEventuallyFormula() || sub.isUntilFormula()) { @@ -160,6 +166,9 @@ namespace storm { if (terminalStates) { terminalStates.reset(); } + if (negatedTerminalStates) { + negatedTerminalStates.reset(); + } // If we are not required to build all reward models, we determine the reward models we need to build. if (!buildAllRewardModels) { @@ -934,7 +943,8 @@ namespace storm { rewardModelBuilders.emplace_back(deterministicModel, rewardModel.get().hasStateRewards(), rewardModel.get().hasStateActionRewards(), rewardModel.get().hasTransitionRewards()); } - // If we were asked to treat some states as terminal states, we cut away their transitions now. + // If we were asked to treat some states as terminal states, we determine an expression characterizing the + // terminal states of the model that we pass on to the matrix building routine. boost::optional terminalExpression; if (options.terminalStates) { if (options.terminalStates.get().type() == typeid(storm::expressions::Expression)) { @@ -944,9 +954,27 @@ namespace storm { terminalExpression = program.getLabelExpression(labelName); } } + if (options.negatedTerminalStates) { + if (options.negatedTerminalStates.get().type() == typeid(storm::expressions::Expression)) { + if (terminalExpression) { + terminalExpression = terminalExpression.get() || !boost::get(options.negatedTerminalStates.get()); + } else { + terminalExpression = !boost::get(options.negatedTerminalStates.get()); + } + } else { + std::string const& labelName = boost::get(options.negatedTerminalStates.get()); + if (terminalExpression) { + terminalExpression = terminalExpression.get() || !program.getLabelExpression(labelName); + } else { + terminalExpression = !program.getLabelExpression(labelName); + } + } + } + if (terminalExpression) { + STORM_LOG_TRACE("Making the states satisfying " << terminalExpression.get() << " terminal."); + } modelComponents.choiceLabeling = buildMatrices(program, variableInformation, selectedRewardModels, stateInformation, options.buildCommandLabels, deterministicModel, discreteTimeModel, transitionMatrixBuilder, rewardModelBuilders, terminalExpression); - modelComponents.transitionMatrix = transitionMatrixBuilder.build(); // Now finalize all reward models. diff --git a/src/builder/ExplicitPrismModelBuilder.h b/src/builder/ExplicitPrismModelBuilder.h index 313b1652c..1356394d8 100644 --- a/src/builder/ExplicitPrismModelBuilder.h +++ b/src/builder/ExplicitPrismModelBuilder.h @@ -193,9 +193,13 @@ namespace storm { // An optional set of expressions for which labels need to be built. boost::optional> expressionLabels; - // An optional expression or label that characterizes the terminal states of the model. If this is set, - // the outgoing transitions of these states are replaced with a self-loop. + // An optional expression or label that characterizes (a subset of) the terminal states of the model. If + // this is set, the outgoing transitions of these states are replaced with a self-loop. boost::optional> terminalStates; + + // An optional expression or label whose negation characterizes (a subset of) the terminal states of the + // model. If this is set, the outgoing transitions of these states are replaced with a self-loop. + boost::optional> negatedTerminalStates; }; /*! diff --git a/src/modelchecker/results/CheckResult.h b/src/modelchecker/results/CheckResult.h index 93ff1399f..1de7a33c5 100644 --- a/src/modelchecker/results/CheckResult.h +++ b/src/modelchecker/results/CheckResult.h @@ -21,6 +21,8 @@ namespace storm { // The base class of all check results. class CheckResult { public: + virtual ~CheckResult() = default; + /*! * Filters the current result wrt. to the filter, i.e. only keeps the entries that are selected by the filter. * This means that the filter must be a qualitative result of proper type (symbolic/explicit). diff --git a/src/modelchecker/results/ExplicitQualitativeCheckResult.h b/src/modelchecker/results/ExplicitQualitativeCheckResult.h index abce327b4..665675cfd 100644 --- a/src/modelchecker/results/ExplicitQualitativeCheckResult.h +++ b/src/modelchecker/results/ExplicitQualitativeCheckResult.h @@ -18,6 +18,7 @@ namespace storm { typedef std::map map_type; ExplicitQualitativeCheckResult(); + virtual ~ExplicitQualitativeCheckResult() = default; ExplicitQualitativeCheckResult(map_type const& map); ExplicitQualitativeCheckResult(map_type&& map); ExplicitQualitativeCheckResult(storm::storage::sparse::state_type state, bool value); diff --git a/src/modelchecker/results/ExplicitQuantitativeCheckResult.h b/src/modelchecker/results/ExplicitQuantitativeCheckResult.h index 07166d258..542abe059 100644 --- a/src/modelchecker/results/ExplicitQuantitativeCheckResult.h +++ b/src/modelchecker/results/ExplicitQuantitativeCheckResult.h @@ -30,6 +30,7 @@ namespace storm { ExplicitQuantitativeCheckResult(ExplicitQuantitativeCheckResult&& other) = default; ExplicitQuantitativeCheckResult& operator=(ExplicitQuantitativeCheckResult&& other) = default; #endif + virtual ~ExplicitQuantitativeCheckResult() = default; ValueType& operator[](storm::storage::sparse::state_type state); ValueType const& operator[](storm::storage::sparse::state_type state) const; diff --git a/src/modelchecker/results/QualitativeCheckResult.h b/src/modelchecker/results/QualitativeCheckResult.h index 5904d2b6e..608e2c0d1 100644 --- a/src/modelchecker/results/QualitativeCheckResult.h +++ b/src/modelchecker/results/QualitativeCheckResult.h @@ -7,6 +7,7 @@ namespace storm { namespace modelchecker { class QualitativeCheckResult : public CheckResult { public: + virtual ~QualitativeCheckResult() = default; virtual QualitativeCheckResult& operator&=(QualitativeCheckResult const& other); virtual QualitativeCheckResult& operator|=(QualitativeCheckResult const& other); virtual void complement(); diff --git a/src/modelchecker/results/QuantitativeCheckResult.h b/src/modelchecker/results/QuantitativeCheckResult.h index 2dc1afc39..425e7b6ef 100644 --- a/src/modelchecker/results/QuantitativeCheckResult.h +++ b/src/modelchecker/results/QuantitativeCheckResult.h @@ -7,6 +7,9 @@ namespace storm { namespace modelchecker { class QuantitativeCheckResult : public CheckResult { public: + + virtual ~QuantitativeCheckResult() = default; + virtual std::unique_ptr compareAgainstBound(storm::logic::ComparisonType comparisonType, double bound) const; virtual bool isQuantitative() const override;