Browse Source

fixed abstraction by considering related predicates for all sources. started on command decomposition

tempestpy_adaptions
dehnert 8 years ago
parent
commit
b0b1f8911e
  1. 2
      src/storm/abstraction/AbstractionInformation.cpp
  2. 9
      src/storm/abstraction/LocalExpressionInformation.cpp
  3. 10
      src/storm/abstraction/LocalExpressionInformation.h
  4. 4
      src/storm/abstraction/MenuGameRefiner.cpp
  5. 71
      src/storm/abstraction/prism/CommandAbstractor.cpp
  6. 7
      src/storm/abstraction/prism/CommandAbstractor.h
  7. 2
      src/storm/settings/modules/AbstractionSettings.cpp
  8. 1
      src/storm/storage/SymbolicModelDescription.cpp
  9. 14
      src/storm/storage/jani/Model.cpp

2
src/storm/abstraction/AbstractionInformation.cpp

@ -490,6 +490,6 @@ namespace storm {
template class AbstractionInformation<storm::dd::DdType::Sylvan>; template class AbstractionInformation<storm::dd::DdType::Sylvan>;
template std::map<uint_fast64_t, std::pair<storm::storage::BitVector, double>> AbstractionInformation<storm::dd::DdType::CUDD>::decodeChoiceToUpdateSuccessorMapping(storm::dd::Bdd<storm::dd::DdType::CUDD> const& choice) const; template std::map<uint_fast64_t, std::pair<storm::storage::BitVector, double>> AbstractionInformation<storm::dd::DdType::CUDD>::decodeChoiceToUpdateSuccessorMapping(storm::dd::Bdd<storm::dd::DdType::CUDD> const& choice) const;
template std::map<uint_fast64_t, std::pair<storm::storage::BitVector, double>> AbstractionInformation<storm::dd::DdType::Sylvan>::decodeChoiceToUpdateSuccessorMapping(storm::dd::Bdd<storm::dd::DdType::Sylvan> const& choice) const;
template std::map<uint_fast64_t, std::pair<storm::storage::BitVector, double>> AbstractionInformation<storm::dd::DdType::Sylvan>::decodeChoiceToUpdateSuccessorMapping(storm::dd::Bdd<storm::dd::DdType::Sylvan > const& choice) const;
} }
} }

9
src/storm/abstraction/LocalExpressionInformation.cpp

@ -141,6 +141,15 @@ namespace storm {
return this->variableToBlockMapping.find(variable)->second; return this->variableToBlockMapping.find(variable)->second;
} }
template <storm::dd::DdType DdType>
std::set<uint_fast64_t> LocalExpressionInformation<DdType>::getBlockIndicesOfVariables(std::set<storm::expressions::Variable> const& variables) const {
std::set<uint_fast64_t> result;
for (auto const& variable : variables) {
result.insert(getBlockIndexOfVariable(variable));
}
return result;
}
template <storm::dd::DdType DdType> template <storm::dd::DdType DdType>
std::set<uint_fast64_t> const& LocalExpressionInformation<DdType>::getRelatedExpressions(storm::expressions::Variable const& variable) const { std::set<uint_fast64_t> const& LocalExpressionInformation<DdType>::getRelatedExpressions(storm::expressions::Variable const& variable) const {
return this->expressionBlocks[getBlockIndexOfVariable(variable)]; return this->expressionBlocks[getBlockIndexOfVariable(variable)];

10
src/storm/abstraction/LocalExpressionInformation.h

@ -75,7 +75,15 @@ namespace storm {
* @return The block index of the given variable. * @return The block index of the given variable.
*/ */
uint_fast64_t getBlockIndexOfVariable(storm::expressions::Variable const& variable) const; uint_fast64_t getBlockIndexOfVariable(storm::expressions::Variable const& variable) const;
/*!
* Retrieves the block indices of the given variables.
*
* @param variables The variables for which to retrieve the blocks.
* @return The block indices of the given variables.
*/
std::set<uint_fast64_t> getBlockIndicesOfVariables(std::set<storm::expressions::Variable> const& variables) const;
/*! /*!
* Retrieves the number of blocks of the variable partition. * Retrieves the number of blocks of the variable partition.
* *

4
src/storm/abstraction/MenuGameRefiner.cpp

@ -253,6 +253,10 @@ namespace storm {
STORM_LOG_DEBUG("Derived new predicate (based on guard): " << newPredicate); STORM_LOG_DEBUG("Derived new predicate (based on guard): " << newPredicate);
} else { } else {
STORM_LOG_TRACE("No bottom state successor. Deriving a new predicate using weakest precondition."); STORM_LOG_TRACE("No bottom state successor. Deriving a new predicate using weakest precondition.");
// player1Choice.template toAdd<ValueType>().exportToDot("choice.dot");
// lowerChoice.template toAdd<ValueType>().exportToDot("lower.dot");
// upperChoice.template toAdd<ValueType>().exportToDot("upper.dot");
// Decode both choices to explicit mappings. // Decode both choices to explicit mappings.
std::map<uint64_t, std::pair<storm::storage::BitVector, ValueType>> lowerChoiceUpdateToSuccessorMapping = abstractionInformation.template decodeChoiceToUpdateSuccessorMapping<ValueType>(lowerChoice); std::map<uint64_t, std::pair<storm::storage::BitVector, ValueType>> lowerChoiceUpdateToSuccessorMapping = abstractionInformation.template decodeChoiceToUpdateSuccessorMapping<ValueType>(lowerChoice);

71
src/storm/abstraction/prism/CommandAbstractor.cpp

@ -69,6 +69,68 @@ namespace storm {
std::map<storm::expressions::Variable, storm::expressions::Expression> CommandAbstractor<DdType, ValueType>::getVariableUpdates(uint64_t auxiliaryChoice) const { std::map<storm::expressions::Variable, storm::expressions::Expression> CommandAbstractor<DdType, ValueType>::getVariableUpdates(uint64_t auxiliaryChoice) const {
return command.get().getUpdate(auxiliaryChoice).getAsVariableToExpressionMap(); return command.get().getUpdate(auxiliaryChoice).getAsVariableToExpressionMap();
} }
template <storm::dd::DdType DdType, typename ValueType>
void CommandAbstractor<DdType, ValueType>::recomputeCachedBddUsingDecomposition() {
STORM_LOG_TRACE("Recomputing BDD for command " << command.get() << " using the decomposition.");
auto start = std::chrono::high_resolution_clock::now();
// compute a decomposition of the command
// * start with all relevant blocks: blocks of assignment variables and variables in the rhs of assignments
// * go through all assignments of all updates and merge relevant blocks that are related via an assignment
// * repeat this until nothing changes anymore
// * the resulting blocks are the decomposition
// Start by constructing the relevant blocks.
std::set<uint64_t> allRelevantBlocks;
std::map<storm::expressions::Variable, uint64_t> variableToBlockIndex;
for (auto const& update : command.get().getUpdates()) {
for (auto const& assignment : update.getAssignments()) {
allRelevantBlocks.insert(localExpressionInformation.getBlockIndexOfVariable(assignment.getVariable()));
auto rhsVariableBlocks = localExpressionInformation.getBlockIndicesOfVariables(assignment.getExpression().getVariables());
allRelevantBlocks.insert(rhsVariableBlocks.begin(), rhsVariableBlocks.end());
}
}
STORM_LOG_TRACE("Found " << allRelevantBlocks.size() << " relevant block.");
// Create a block partition.
std::vector<std::set<uint64_t>> relevantBlockPartition;
std::map<storm::expressions::Variable, uint64_t> variableToLocalBlockIndex;
uint64_t index = 0;
for (auto const& blockIndex : allRelevantBlocks) {
relevantBlockPartition.emplace_back(std::set<uint64_t>({blockIndex}));
for (auto const& variable : localExpressionInformation.getVariableBlockWithIndex(blockIndex)) {
variableToLocalBlockIndex[variable] = index;
}
++index;
}
// Proceed by relating the blocks via assignments until nothing changes anymore.
bool changed = false;
do {
for (auto const& update : command.get().getUpdates()) {
for (auto const& assignment : update.getAssignments()) {
std::set<storm::expressions::Variable> rhsVariables = assignment.getExpression().getVariables();
}
}
} while (changed);
// if the decomposition has size 1, use the plain technique from before
// otherwise, enumerate the abstract guard so we do this only once
// then enumerate the solutions for each of the blocks of the decomposition
// multiply the results
// multiply with the abstract guard
// multiply with missing identities
// cache and return result
}
template <storm::dd::DdType DdType, typename ValueType> template <storm::dd::DdType DdType, typename ValueType>
void CommandAbstractor<DdType, ValueType>::recomputeCachedBdd() { void CommandAbstractor<DdType, ValueType>::recomputeCachedBdd() {
@ -140,12 +202,12 @@ namespace storm {
std::set<storm::expressions::Variable> assignedVariables; std::set<storm::expressions::Variable> assignedVariables;
for (auto const& assignment : assignments) { for (auto const& assignment : assignments) {
// Also, variables appearing on the right-hand side of an assignment are relevant for source state. // Also, variables appearing on the right-hand side of an assignment are relevant for source state.
auto const& rightHandSidePredicates = localExpressionInformation.getExpressionsUsingVariables(assignment.getExpression().getVariables());
auto const& rightHandSidePredicates = localExpressionInformation.getRelatedExpressions(assignment.getExpression().getVariables());
result.first.insert(rightHandSidePredicates.begin(), rightHandSidePredicates.end()); result.first.insert(rightHandSidePredicates.begin(), rightHandSidePredicates.end());
// Variables that are being assigned are relevant for the successor state. // Variables that are being assigned are relevant for the successor state.
storm::expressions::Variable const& assignedVariable = assignment.getVariable(); storm::expressions::Variable const& assignedVariable = assignment.getVariable();
auto const& leftHandSidePredicates = localExpressionInformation.getExpressionsUsingVariable(assignedVariable);
auto const& leftHandSidePredicates = localExpressionInformation.getRelatedExpressions(assignedVariable);
result.second.insert(leftHandSidePredicates.begin(), leftHandSidePredicates.end()); result.second.insert(leftHandSidePredicates.begin(), leftHandSidePredicates.end());
// Keep track of all assigned variables, so we can find the related predicates later. // Keep track of all assigned variables, so we can find the related predicates later.
@ -165,7 +227,7 @@ namespace storm {
std::pair<std::set<uint_fast64_t>, std::vector<std::set<uint_fast64_t>>> result; std::pair<std::set<uint_fast64_t>, std::vector<std::set<uint_fast64_t>>> result;
// To start with, all predicates related to the guard are relevant source predicates. // To start with, all predicates related to the guard are relevant source predicates.
result.first = localExpressionInformation.getExpressionsUsingVariables(command.get().getGuardExpression().getVariables());
result.first = localExpressionInformation.getRelatedExpressions(command.get().getGuardExpression().getVariables());
// Then, we add the predicates that become relevant, because of some update. // Then, we add the predicates that become relevant, because of some update.
for (auto const& update : command.get().getUpdates()) { for (auto const& update : command.get().getUpdates()) {
@ -223,13 +285,10 @@ namespace storm {
template <storm::dd::DdType DdType, typename ValueType> template <storm::dd::DdType DdType, typename ValueType>
storm::dd::Bdd<DdType> CommandAbstractor<DdType, ValueType>::getSourceStateBdd(storm::solver::SmtSolver::ModelReference const& model) const { storm::dd::Bdd<DdType> CommandAbstractor<DdType, ValueType>::getSourceStateBdd(storm::solver::SmtSolver::ModelReference const& model) const {
storm::dd::Bdd<DdType> result = this->getAbstractionInformation().getDdManager().getBddOne(); storm::dd::Bdd<DdType> result = this->getAbstractionInformation().getDdManager().getBddOne();
// std::cout << "new model ----------------" << std::endl;
for (auto const& variableIndexPair : relevantPredicatesAndVariables.first) { for (auto const& variableIndexPair : relevantPredicatesAndVariables.first) {
if (model.getBooleanValue(variableIndexPair.first)) { if (model.getBooleanValue(variableIndexPair.first)) {
// std::cout << this->getAbstractionInformation().getPredicateByIndex(variableIndexPair.second) << " is true" << std::endl;
result &= this->getAbstractionInformation().encodePredicateAsSource(variableIndexPair.second); result &= this->getAbstractionInformation().encodePredicateAsSource(variableIndexPair.second);
} else { } else {
// std::cout << this->getAbstractionInformation().getPredicateByIndex(variableIndexPair.second) << " is false" << std::endl;
result &= !this->getAbstractionInformation().encodePredicateAsSource(variableIndexPair.second); result &= !this->getAbstractionInformation().encodePredicateAsSource(variableIndexPair.second);
} }
} }

7
src/storm/abstraction/prism/CommandAbstractor.h

@ -161,6 +161,11 @@ namespace storm {
* Recomputes the cached BDD. This needs to be triggered if any relevant predicates change. * Recomputes the cached BDD. This needs to be triggered if any relevant predicates change.
*/ */
void recomputeCachedBdd(); void recomputeCachedBdd();
/*!
* Recomputes the cached BDD using a decomposition. This needs to be triggered if any relevant predicates change.
*/
void recomputeCachedBddUsingDecomposition();
/*! /*!
* Computes the missing state identities. * Computes the missing state identities.
@ -219,7 +224,7 @@ namespace storm {
// The set of all relevant predicates. // The set of all relevant predicates.
std::set<uint64_t> allRelevantPredicates; std::set<uint64_t> allRelevantPredicates;
// The most recent result of a call to computeDd. If nothing has changed regarding the relevant // The most recent result of a call to computeDd. If nothing has changed regarding the relevant
// predicates, this result may be reused. // predicates, this result may be reused.
GameBddResult<DdType> cachedDd; GameBddResult<DdType> cachedDd;

2
src/storm/settings/modules/AbstractionSettings.cpp

@ -39,7 +39,7 @@ namespace storm {
std::vector<std::string> invalidBlockStrategies = {"none", "local", "global"}; std::vector<std::string> invalidBlockStrategies = {"none", "local", "global"};
this->addOption(storm::settings::OptionBuilder(moduleName, invalidBlockStrategyOptionName, true, "Sets the strategy to detect invalid blocks.") this->addOption(storm::settings::OptionBuilder(moduleName, invalidBlockStrategyOptionName, true, "Sets the strategy to detect invalid blocks.")
.addArgument(storm::settings::ArgumentBuilder::createStringArgument("name", "The name of an available strategy. Available are: 'none', 'local' and 'global'.").addValidationFunctionString(storm::settings::ArgumentValidators::stringInListValidator(invalidBlockStrategies)).setDefaultValueString("global").build()).build());
.addArgument(storm::settings::ArgumentBuilder::createStringArgument("name", "The name of an available strategy. Available are: 'none', 'local' and 'global'.").addValidationFunctionString(storm::settings::ArgumentValidators::stringInListValidator(invalidBlockStrategies)).setDefaultValueString("local").build()).build());
this->addOption(storm::settings::OptionBuilder(moduleName, reuseQualitativeResultsOptionName, true, "Sets whether to reuse qualitative results.").build()); this->addOption(storm::settings::OptionBuilder(moduleName, reuseQualitativeResultsOptionName, true, "Sets whether to reuse qualitative results.").build());
this->addOption(storm::settings::OptionBuilder(moduleName, reuseQuantitativeResultsOptionName, true, "Sets whether to reuse quantitative results.").build()); this->addOption(storm::settings::OptionBuilder(moduleName, reuseQuantitativeResultsOptionName, true, "Sets whether to reuse quantitative results.").build());

1
src/storm/storage/SymbolicModelDescription.cpp

@ -119,7 +119,6 @@ namespace storm {
if (this->isJaniModel()) { if (this->isJaniModel()) {
std::map<storm::expressions::Variable, storm::expressions::Expression> substitution = storm::utility::jani::parseConstantDefinitionString(this->asJaniModel(), constantDefinitionString); std::map<storm::expressions::Variable, storm::expressions::Expression> substitution = storm::utility::jani::parseConstantDefinitionString(this->asJaniModel(), constantDefinitionString);
storm::jani::Model preparedModel = this->asJaniModel().defineUndefinedConstants(substitution).substituteConstants(); storm::jani::Model preparedModel = this->asJaniModel().defineUndefinedConstants(substitution).substituteConstants();
preparedModel = preparedModel.flattenComposition();
return SymbolicModelDescription(preparedModel); return SymbolicModelDescription(preparedModel);
} else if (this->isPrismProgram()) { } else if (this->isPrismProgram()) {
std::map<storm::expressions::Variable, storm::expressions::Expression> substitution = storm::utility::prism::parseConstantDefinitionString(this->asPrismProgram(), constantDefinitionString); std::map<storm::expressions::Variable, storm::expressions::Expression> substitution = storm::utility::prism::parseConstantDefinitionString(this->asPrismProgram(), constantDefinitionString);

14
src/storm/storage/jani/Model.cpp

@ -511,7 +511,17 @@ namespace storm {
newAutomaton.changeAssignmentVariables(variableRemapping); newAutomaton.changeAssignmentVariables(variableRemapping);
// Finalize the flattened model. // Finalize the flattened model.
storm::expressions::Expression initialStatesRestriction = getManager().boolean(true);
for (auto const& automaton : composedAutomata) {
if (automaton.get().hasInitialStatesRestriction()) {
initialStatesRestriction = initialStatesRestriction && automaton.get().getInitialStatesRestriction();
}
}
newAutomaton.setInitialStatesRestriction(this->getInitialStatesExpression(composedAutomata)); newAutomaton.setInitialStatesRestriction(this->getInitialStatesExpression(composedAutomata));
if (this->hasInitialStatesRestriction()) {
flattenedModel.setInitialStatesRestriction(this->getInitialStatesRestriction());
}
flattenedModel.addAutomaton(newAutomaton); flattenedModel.addAutomaton(newAutomaton);
flattenedModel.setStandardSystemComposition(); flattenedModel.setStandardSystemComposition();
flattenedModel.finalize(); flattenedModel.finalize();
@ -879,6 +889,10 @@ namespace storm {
this->initialStatesRestriction = initialStatesRestriction; this->initialStatesRestriction = initialStatesRestriction;
} }
bool Model::hasInitialStatesRestriction() const {
return this->initialStatesRestriction.isInitialized();
}
storm::expressions::Expression const& Model::getInitialStatesRestriction() const { storm::expressions::Expression const& Model::getInitialStatesRestriction() const {
return initialStatesRestriction; return initialStatesRestriction;
} }

Loading…
Cancel
Save