From 2bda04771beb291ea6b81c1f4b70c03e4499f90d Mon Sep 17 00:00:00 2001 From: Jip Spel Date: Wed, 15 Apr 2020 09:19:35 +0200 Subject: [PATCH 01/11] Remove duplicate preprocessing --- src/storm-pars-cli/storm-pars.cpp | 22 ---------------------- 1 file changed, 22 deletions(-) diff --git a/src/storm-pars-cli/storm-pars.cpp b/src/storm-pars-cli/storm-pars.cpp index 4ae3fde89..fbb8c5971 100644 --- a/src/storm-pars-cli/storm-pars.cpp +++ b/src/storm-pars-cli/storm-pars.cpp @@ -582,28 +582,6 @@ namespace storm { STORM_LOG_THROW(model || input.properties.empty(), storm::exceptions::InvalidSettingsException, "No input model."); - - - if (model) { - auto preprocessingResult = storm::pars::preprocessModel(model, input, mpi); - if (preprocessingResult.changed) { - model = preprocessingResult.model; - - if (preprocessingResult.formulas) { - std::vector newProperties; - for (size_t i = 0; i < preprocessingResult.formulas.get().size(); ++i) { - auto formula = preprocessingResult.formulas.get().at(i); - STORM_LOG_ASSERT(i < input.properties.size(), "Index " << i << " greater than number of properties."); - storm::jani::Property property = input.properties.at(i); - newProperties.push_back(storm::jani::Property(property.getName(), formula, property.getUndefinedConstants(), property.getComment())); - } - input.properties = newProperties; - } - - model->printModelInformationToStream(std::cout); - } - } - if (monSettings.isMonotonicityAnalysisSet()) { // Simplify the model storm::utility::Stopwatch simplifyingWatch(true); From c1b4c3270f51a4c8dcd3a6a7ff1eb1dd11e9fe41 Mon Sep 17 00:00:00 2001 From: Matthias Volk Date: Wed, 15 Apr 2020 14:18:46 +0200 Subject: [PATCH 02/11] Fixed initialization order warnings --- src/storm/simulator/DiscreteTimeSparseModelSimulator.cpp | 4 ++-- src/storm/utility/random.cpp | 4 ++-- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/src/storm/simulator/DiscreteTimeSparseModelSimulator.cpp b/src/storm/simulator/DiscreteTimeSparseModelSimulator.cpp index d8fd4a82a..2e9161b8a 100644 --- a/src/storm/simulator/DiscreteTimeSparseModelSimulator.cpp +++ b/src/storm/simulator/DiscreteTimeSparseModelSimulator.cpp @@ -4,7 +4,7 @@ namespace storm { namespace simulator { template - DiscreteTimeSparseModelSimulator::DiscreteTimeSparseModelSimulator(storm::models::sparse::Model const& model) : model(model), currentState(*model.getInitialStates().begin()) { + DiscreteTimeSparseModelSimulator::DiscreteTimeSparseModelSimulator(storm::models::sparse::Model const& model) : currentState(*model.getInitialStates().begin()), model(model) { STORM_LOG_WARN_COND(model.getInitialStates().getNumberOfSetBits()==1, "The model has multiple initial states. This simulator assumes it starts from the initial state with the lowest index."); } @@ -47,4 +47,4 @@ namespace storm { template class DiscreteTimeSparseModelSimulator; } -} \ No newline at end of file +} diff --git a/src/storm/utility/random.cpp b/src/storm/utility/random.cpp index 6f3464cdb..ad0afecb0 100644 --- a/src/storm/utility/random.cpp +++ b/src/storm/utility/random.cpp @@ -10,7 +10,7 @@ namespace storm { } RandomProbabilityGenerator::RandomProbabilityGenerator(uint64_t seed) - : engine(seed), distribution(0.0, 1.0) + : distribution(0.0, 1.0), engine(seed) { } @@ -21,4 +21,4 @@ namespace storm { } -} \ No newline at end of file +} From 325b700c620cd873a5e2c65c9cbe6bd36e6f1e8d Mon Sep 17 00:00:00 2001 From: Matthias Volk Date: Wed, 15 Apr 2020 14:19:49 +0200 Subject: [PATCH 03/11] Explicitly set initialization order for SparseMatrix to avoid nasty segfaults --- src/storm/storage/SparseMatrix.cpp | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/src/storm/storage/SparseMatrix.cpp b/src/storm/storage/SparseMatrix.cpp index 59d158d96..26e1c0dc7 100644 --- a/src/storm/storage/SparseMatrix.cpp +++ b/src/storm/storage/SparseMatrix.cpp @@ -406,7 +406,12 @@ namespace storm { } template - SparseMatrix::SparseMatrix(index_type columnCount, std::vector&& rowIndications, std::vector>&& columnsAndValues, boost::optional>&& rowGroupIndices) : rowCount(rowIndications.size() - 1), columnCount(columnCount), entryCount(columnsAndValues.size()), nonzeroEntryCount(0), columnsAndValues(std::move(columnsAndValues)), rowIndications(std::move(rowIndications)), trivialRowGrouping(!rowGroupIndices), rowGroupIndices(std::move(rowGroupIndices)) { + SparseMatrix::SparseMatrix(index_type columnCount, std::vector&& rowIndications, std::vector>&& columnsAndValues, boost::optional>&& rowGroupIndices) : columnCount(columnCount), nonzeroEntryCount(0), columnsAndValues(std::move(columnsAndValues)), rowIndications(std::move(rowIndications)), rowGroupIndices(std::move(rowGroupIndices)) { + // Initialize some variables here which depend on other variables + // This way we are more robust against different initialization orders + this->rowCount = this->rowIndications.size() - 1; + this->entryCount = this->columnsAndValues.size(); + this->trivialRowGrouping = !this->rowGroupIndices; this->updateNonzeroEntryCount(); } From be7181f9f239912eb8f5990f332f6674dc59158c Mon Sep 17 00:00:00 2001 From: Matthias Volk Date: Wed, 15 Apr 2020 14:20:25 +0200 Subject: [PATCH 04/11] Removed double include --- .../storm/modelchecker/prctl/mdp/MdpPrctlModelCheckerTest.cpp | 2 -- 1 file changed, 2 deletions(-) diff --git a/src/test/storm/modelchecker/prctl/mdp/MdpPrctlModelCheckerTest.cpp b/src/test/storm/modelchecker/prctl/mdp/MdpPrctlModelCheckerTest.cpp index 0df0bea3f..87e670104 100755 --- a/src/test/storm/modelchecker/prctl/mdp/MdpPrctlModelCheckerTest.cpp +++ b/src/test/storm/modelchecker/prctl/mdp/MdpPrctlModelCheckerTest.cpp @@ -1,8 +1,6 @@ #include "test/storm_gtest.h" #include "storm-config.h" -#include "test/storm_gtest.h" - #include "storm/api/builder.h" #include "storm-conv/api/storm-conv.h" #include "storm-parsers/api/model_descriptions.h" From 88c31b36d04cc2e77913cc3dcf1847fb87cab49e Mon Sep 17 00:00:00 2001 From: Tim Quatmann Date: Mon, 20 Apr 2020 08:09:29 +0200 Subject: [PATCH 05/11] Equation system based CTMC LRA solving: For the 'inner' linear equation system solver, also set whether the solver type has been set from default. This avoids potentially using unsound/inexact equation solvers. --- src/storm/modelchecker/csl/helper/SparseCtmcCslHelper.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/storm/modelchecker/csl/helper/SparseCtmcCslHelper.cpp b/src/storm/modelchecker/csl/helper/SparseCtmcCslHelper.cpp index 55cae421b..ab8912c5d 100644 --- a/src/storm/modelchecker/csl/helper/SparseCtmcCslHelper.cpp +++ b/src/storm/modelchecker/csl/helper/SparseCtmcCslHelper.cpp @@ -831,7 +831,7 @@ namespace storm { auto subEnv = env; if (subEnv.solver().getLinearEquationSolverType() == storm::solver::EquationSolverType::Topological) { // Topological solver does not make any sense since the BSCC is connected. - subEnv.solver().setLinearEquationSolverType(subEnv.solver().topological().getUnderlyingEquationSolverType()); + subEnv.solver().setLinearEquationSolverType(subEnv.solver().topological().getUnderlyingEquationSolverType(), subEnv.solver().topological().isUnderlyingEquationSolverTypeSetFromDefault()); } subEnv.solver().setLinearEquationSolverPrecision(env.solver().lra().getPrecision(), env.solver().lra().getRelativeTerminationCriterion()); @@ -926,7 +926,7 @@ namespace storm { auto subEnv = env; if (subEnv.solver().getLinearEquationSolverType() == storm::solver::EquationSolverType::Topological) { // Topological solver does not make any sense since the BSCC is connected. - subEnv.solver().setLinearEquationSolverType(subEnv.solver().topological().getUnderlyingEquationSolverType()); + subEnv.solver().setLinearEquationSolverType(subEnv.solver().topological().getUnderlyingEquationSolverType(), subEnv.solver().topological().isUnderlyingEquationSolverTypeSetFromDefault()); } subEnv.solver().setLinearEquationSolverPrecision(env.solver().lra().getPrecision(), env.solver().lra().getRelativeTerminationCriterion()); From 49dac54e8b80cea8c5209447de1b70ad5b770161 Mon Sep 17 00:00:00 2001 From: Matthias Volk Date: Mon, 20 Apr 2020 19:40:42 +0200 Subject: [PATCH 06/11] Fixed typos --- src/storm-dft-cli/storm-dft.cpp | 1 + src/storm-dft/builder/ExplicitDFTModelBuilder.cpp | 4 ++-- src/storm-dft/storage/dft/DFT.cpp | 2 +- src/storm-dft/storage/dft/elements/BEExponential.h | 3 ++- 4 files changed, 6 insertions(+), 4 deletions(-) diff --git a/src/storm-dft-cli/storm-dft.cpp b/src/storm-dft-cli/storm-dft.cpp index 5d835b4c7..87338ff30 100644 --- a/src/storm-dft-cli/storm-dft.cpp +++ b/src/storm-dft-cli/storm-dft.cpp @@ -87,6 +87,7 @@ void processOptions() { // Apply transformations // TODO transform later before actual analysis dft = storm::api::applyTransformations(*dft, faultTreeSettings.isUniqueFailedBE(), true); + STORM_LOG_DEBUG(dft->getElementsString()); dft->setDynamicBehaviorInfo(); diff --git a/src/storm-dft/builder/ExplicitDFTModelBuilder.cpp b/src/storm-dft/builder/ExplicitDFTModelBuilder.cpp index b5ed7c298..a1bd445d4 100644 --- a/src/storm-dft/builder/ExplicitDFTModelBuilder.cpp +++ b/src/storm-dft/builder/ExplicitDFTModelBuilder.cpp @@ -59,14 +59,14 @@ namespace storm { // Consider all children of the top level gate std::vector isubdft; if (child->nrParents() > 1 || child->hasOutgoingDependencies()) { - STORM_LOG_TRACE("child " << child->name() << "does not allow modularisation."); + STORM_LOG_TRACE("child " << child->name() << " does not allow modularisation."); isubdft.clear(); } else if (dft.isGate(child->id())) { isubdft = dft.getGate(child->id())->independentSubDft(false); } else { STORM_LOG_ASSERT(dft.isBasicElement(child->id()), "Child is no BE."); if(dft.getBasicElement(child->id())->hasIngoingDependencies()) { - STORM_LOG_TRACE("child " << child->name() << "does not allow modularisation."); + STORM_LOG_TRACE("child " << child->name() << " does not allow modularisation."); isubdft.clear(); } else { isubdft = {child->id()}; diff --git a/src/storm-dft/storage/dft/DFT.cpp b/src/storm-dft/storage/dft/DFT.cpp index daa6c33f2..6b79e1c1e 100644 --- a/src/storm-dft/storage/dft/DFT.cpp +++ b/src/storm-dft/storage/dft/DFT.cpp @@ -416,7 +416,7 @@ namespace storm { } else { STORM_LOG_ASSERT(isBasicElement(child->id()), "Child is no BE."); if (getBasicElement(child->id())->hasIngoingDependencies()) { - STORM_LOG_TRACE("child " << child->name() << "does not allow modularisation."); + STORM_LOG_TRACE("child " << child->name() << " does not allow modularisation."); return {*this}; } else { isubdft = {child->id()}; diff --git a/src/storm-dft/storage/dft/elements/BEExponential.h b/src/storm-dft/storage/dft/elements/BEExponential.h index 8347f3179..1ed6221d6 100644 --- a/src/storm-dft/storage/dft/elements/BEExponential.h +++ b/src/storm-dft/storage/dft/elements/BEExponential.h @@ -67,7 +67,8 @@ namespace storm { } bool canFail() const override { - return !storm::utility::isZero(this->activeFailureRate()); + STORM_LOG_ASSERT(!storm::utility::isZero(this->activeFailureRate()), "BEExp should have failure rate > 0"); + return true; } /*! From 7e1f5bf2ac742ac090b7d3e9c7dd00692b644cea Mon Sep 17 00:00:00 2001 From: Matthias Volk Date: Mon, 20 Apr 2020 20:36:06 +0200 Subject: [PATCH 07/11] Fixed handling of constant BE in approximation --- .../builder/ExplicitDFTModelBuilder.cpp | 38 +++++++++---------- 1 file changed, 18 insertions(+), 20 deletions(-) diff --git a/src/storm-dft/builder/ExplicitDFTModelBuilder.cpp b/src/storm-dft/builder/ExplicitDFTModelBuilder.cpp index a1bd445d4..1f5f3c18f 100644 --- a/src/storm-dft/builder/ExplicitDFTModelBuilder.cpp +++ b/src/storm-dft/builder/ExplicitDFTModelBuilder.cpp @@ -745,37 +745,35 @@ namespace storm { storm::storage::BitVector coldBEs(subtree.size(), false); for (size_t i = 0; i < subtree.size(); ++i) { size_t id = subtree[i]; + // Consider only still operational BEs if (state->isOperational(id)) { - // Get BE rate - ValueType rate = state->getBERate(id); - if (storm::utility::isZero(rate)) { - // Get active failure rate for cold BE - auto be = dft.getBasicElement(id); - switch (be->type()) { - case storm::storage::DFTElementType::BE_EXP: - { + auto be = dft.getBasicElement(id); + switch (be->type()) { + case storm::storage::DFTElementType::BE_EXP: + { + // Get BE rate + ValueType rate = state->getBERate(id); + if (storm::utility::isZero(rate)) { + // Get active failure rate for cold BE auto beExp = std::static_pointer_cast const>(be); rate = beExp->activeFailureRate(); STORM_LOG_ASSERT(!storm::utility::isZero(rate), "Failure rate should not be zero."); // Mark BE as cold coldBEs.set(i, true); - break; } - case storm::storage::DFTElementType::BE_CONST: - { - // Ignore BE which cannot fail - continue; - } - default: - STORM_LOG_THROW(false, storm::exceptions::InvalidArgumentException, "BE of type '" << be->type() << "' is not known."); - break; + rates.push_back(rate); + rateSum += rate; + break; } + case storm::storage::DFTElementType::BE_CONST: + // Ignore BE which cannot fail + continue; + default: + STORM_LOG_THROW(false, storm::exceptions::InvalidArgumentException, "BE of type '" << be->type() << "' is not known."); + break; } - rates.push_back(rate); - rateSum += rate; } } - STORM_LOG_ASSERT(rates.size() > 0, "No rates failable"); // Sort rates From 0e97647d9ef7e0661f88c6ba179f4e83098e19c8 Mon Sep 17 00:00:00 2001 From: Sebastian Junges Date: Thu, 23 Apr 2020 15:52:06 -0700 Subject: [PATCH 08/11] canonic POMDPs are now annotated with the state valuations and the choice labellings --- src/storm-pomdp/transformer/MakePOMDPCanonic.cpp | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/storm-pomdp/transformer/MakePOMDPCanonic.cpp b/src/storm-pomdp/transformer/MakePOMDPCanonic.cpp index da03d47ca..9b83f898d 100644 --- a/src/storm-pomdp/transformer/MakePOMDPCanonic.cpp +++ b/src/storm-pomdp/transformer/MakePOMDPCanonic.cpp @@ -136,7 +136,9 @@ namespace storm { newRewardModels, false, boost::none); modelcomponents.observabilityClasses = pomdp.getObservations(); - //modelcomponents.choiceLabeling = pomdp.getChoiceLabeling(); + modelcomponents.stateValuations = pomdp.getOptionalStateValuations(); + modelcomponents.choiceLabeling = pomdp.getChoiceLabeling(); + modelcomponents.choiceLabeling->permuteItems(permutation); return std::make_shared>(modelcomponents, true); } From e4a4214fc4e1f4b007e5aa7d5073572e4fe83681 Mon Sep 17 00:00:00 2001 From: Sebastian Junges Date: Thu, 23 Apr 2020 15:52:36 -0700 Subject: [PATCH 09/11] towards more helpful output when POMDP is not canonical --- .../transformer/MakePOMDPCanonic.cpp | 20 +++++++++++++++++-- 1 file changed, 18 insertions(+), 2 deletions(-) diff --git a/src/storm-pomdp/transformer/MakePOMDPCanonic.cpp b/src/storm-pomdp/transformer/MakePOMDPCanonic.cpp index 9b83f898d..f0a9a3c26 100644 --- a/src/storm-pomdp/transformer/MakePOMDPCanonic.cpp +++ b/src/storm-pomdp/transformer/MakePOMDPCanonic.cpp @@ -175,7 +175,15 @@ namespace storm { if (moreActionObservations.get(observation)) { // We have seen this observation previously with multiple actions. Error! // TODO provide more diagnostic information - STORM_LOG_THROW(false, storm::exceptions::AmbiguousModelException, "Observation " << observation << " sometimes provides multiple actions, but in state " << state << " provides one action."); + std::string stateval =""; + if (pomdp.hasStateValuations()) { + stateval = " (" + pomdp.getStateValuations().getStateInfo(state) + ") "; + } + std::string actionval= ""; + if (pomdp.hasChoiceLabeling()) { + actionval = *pomdp.getChoiceLabeling().getLabelsOfChoice(rowIndexFrom).begin(); + } + STORM_LOG_THROW(false, storm::exceptions::AmbiguousModelException, "Observation " << observation << " sometimes provides multiple actions, but in state " << state << stateval << " provides only one action " << actionval << "."); } oneActionObservations.set(observation); @@ -184,7 +192,15 @@ namespace storm { } else { if (oneActionObservations.get(observation)) { // We have seen this observation previously with one action. Error! - STORM_LOG_THROW(false, storm::exceptions::AmbiguousModelException, "Observation " << observation << " sometimes provides one action, but in state " << state << " provides multiple actions."); + std::string stateval =""; + if (pomdp.hasStateValuations()) { + stateval = " (" + pomdp.getStateValuations().getStateInfo(state) + ") "; + } +// std::string actionval= ""; +// if (pomdp.hasChoiceLabeling()) { +// actionval = *pomdp.getChoiceLabeling().getLabelsOfChoice(rowIndexFrom).begin(); +// } + STORM_LOG_THROW(false, storm::exceptions::AmbiguousModelException, "Observation " << observation << " sometimes provides one action, but in state " << state << stateval << " provides " << rowIndexTo - rowIndexFrom << " actions."); } moreActionObservations.set(observation); } From 91cad8164fd7f9842de24ccfa2c9a5960371ac1d Mon Sep 17 00:00:00 2001 From: Sebastian Junges Date: Thu, 23 Apr 2020 15:54:19 -0700 Subject: [PATCH 10/11] trace outputs towards debugging the JaniScopeChanger --- .../storage/expressions/ExpressionManager.cpp | 2 +- src/storm/storage/jani/JaniScopeChanger.cpp | 21 ++++++++++++++++++- 2 files changed, 21 insertions(+), 2 deletions(-) diff --git a/src/storm/storage/expressions/ExpressionManager.cpp b/src/storm/storage/expressions/ExpressionManager.cpp index 219e5a521..07a44dd3b 100644 --- a/src/storm/storage/expressions/ExpressionManager.cpp +++ b/src/storm/storage/expressions/ExpressionManager.cpp @@ -324,7 +324,7 @@ namespace storm { out << "manager {" << std::endl; for (auto const& variableTypePair : manager) { - out << "\t" << variableTypePair.second << " " << variableTypePair.first.getName() << " [offset " << variableTypePair.first.getOffset() << "]" << std::endl; + out << "\t" << variableTypePair.second << " " << variableTypePair.first.getName() << " [offset " << variableTypePair.first.getOffset() << ", " << variableTypePair.first.getIndex() <<" ]" << std::endl; } out << "}" << std::endl; diff --git a/src/storm/storage/jani/JaniScopeChanger.cpp b/src/storm/storage/jani/JaniScopeChanger.cpp index 752dc033e..58e0bbbb1 100644 --- a/src/storm/storage/jani/JaniScopeChanger.cpp +++ b/src/storm/storage/jani/JaniScopeChanger.cpp @@ -24,6 +24,9 @@ namespace storm { bool* result = boost::any_cast(data); if (*result) { return; } *result = expression.containsVariable(varSet); + if (*result) { + STORM_LOG_TRACE("The expression " << expression << " 'contains' a variable in the variable set."); + } } private: @@ -34,6 +37,7 @@ namespace storm { std::set res; for (uint64_t i = 0; i < model.getNumberOfAutomata(); ++i) { if (model.getAutomaton(i).getVariables().hasVariable(variable)) { + STORM_LOG_TRACE("The automaton " << model.getAutomaton(i).getName() << " 'has' the variable " << variable.getName() << "."); res.insert(i); } else { VariableAccessedTraverser vat({variable}); @@ -41,6 +45,8 @@ namespace storm { vat.traverse(model.getAutomaton(i), &varAccessed); if (varAccessed) { res.insert(i); + STORM_LOG_TRACE("The automaton " << model.getAutomaton(i).getName() << " 'accesses' the variable " << variable.getName() << "."); + } } } @@ -72,6 +78,8 @@ namespace storm { } bool JaniScopeChanger::canMakeVariableGlobal(storm::expressions::Variable const& variable, Model const& model) const { + STORM_LOG_TRACE("Can the variable " << variable.getName() << " be made global?"); + if (model.hasGlobalVariable(variable.getName())) { return false; } @@ -87,8 +95,12 @@ namespace storm { } return foundVar; } - + + + std::pair JaniScopeChanger::canMakeVariableLocal(storm::expressions::Variable const& variable, Model const& model, std::vector const& properties, boost::optional automatonIndex) const { + STORM_LOG_TRACE("Can the variable " << variable.getName() << " be made local?"); + uint64_t index = model.getNumberOfAutomata(); if (!model.getGlobalVariables().hasVariable(variable)) { @@ -97,26 +109,32 @@ namespace storm { auto accessingAutomata = detail::getAutomataAccessingVariable(variable, model); if (accessingAutomata.size() > 1 || (automatonIndex.is_initialized() && accessingAutomata.count(automatonIndex.get()) == 0)) { + STORM_LOG_TRACE(".. no!, multiple automata access the variable, e.g. automata " << model.getAutomaton(*accessingAutomata.begin()).getName() << " and " << model.getAutomaton(*accessingAutomata.rbegin()).getName()); return {false, index}; } if (model.getInitialStatesRestriction().containsVariable({variable})) { + STORM_LOG_TRACE(".. no!, initial states restriction contains variable"); return {false, index}; } for (auto const& rewExp : model.getNonTrivialRewardExpressions()) { if (rewExp.second.containsVariable({variable})) { + STORM_LOG_TRACE(".. no!, non trivial reward expression "); return {false, index}; } } for (auto const& funDef : model.getGlobalFunctionDefinitions()) { if (funDef.second.getFunctionBody().containsVariable({variable})) { + STORM_LOG_TRACE(".. no!, function definition: "); return {false, index}; } } for (auto const& p : properties) { if (p.getUsedVariablesAndConstants().count(variable) > 0) { + STORM_LOG_TRACE(".. no!, used variables definition: "); return {false, index}; } if (p.getUsedLabels().count(variable.getName()) > 0) { + STORM_LOG_TRACE(".. no!, used labels definition: "); return {false, index}; } } @@ -127,6 +145,7 @@ namespace storm { index = *accessingAutomata.begin(); assert(!automatonIndex.is_initialized() || index == automatonIndex.get()); } + STORM_LOG_TRACE(".. Yes, made local in automaton with index " << index ); return {true, index}; } From 493090894237037758b8fa066466e8797fd40656 Mon Sep 17 00:00:00 2001 From: Sebastian Junges Date: Thu, 23 Apr 2020 15:57:16 -0700 Subject: [PATCH 11/11] new version of containsVariable, with better performance and somewhat better to debug --- src/storm/storage/expressions/Expression.cpp | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/src/storm/storage/expressions/Expression.cpp b/src/storm/storage/expressions/Expression.cpp index 0221628db..520bf1db4 100644 --- a/src/storm/storage/expressions/Expression.cpp +++ b/src/storm/storage/expressions/Expression.cpp @@ -133,9 +133,12 @@ namespace storm { bool Expression::containsVariable(std::set const& variables) const { std::set appearingVariables = this->getVariables(); - std::set intersection; - std::set_intersection(variables.begin(), variables.end(), appearingVariables.begin(), appearingVariables.end(), std::inserter(intersection, intersection.begin())); - return !intersection.empty(); + for (auto const& v : variables) { + if (appearingVariables.count(v) > 0) { + return true; + } + } + return false; } bool Expression::containsVariableInITEGuard(std::set const& variables) const {