Browse Source
added some accessor functions and iteration capabilities. started on symbolic jani model builder
added some accessor functions and iteration capabilities. started on symbolic jani model builder
Former-commit-id: 4388aca60d
main
25 changed files with 1118 additions and 37 deletions
-
125src/builder/DdJaniModelBuilder.cpp
-
121src/builder/DdJaniModelBuilder.h
-
4src/storage/jani/Assignment.cpp
-
5src/storage/jani/Assignment.h
-
140src/storage/jani/Automaton.cpp
-
131src/storage/jani/Automaton.h
-
8src/storage/jani/BoundedIntegerVariable.cpp
-
10src/storage/jani/BoundedIntegerVariable.h
-
8src/storage/jani/Constant.cpp
-
10src/storage/jani/Constant.h
-
12src/storage/jani/Edge.cpp
-
15src/storage/jani/Edge.h
-
8src/storage/jani/EdgeDestination.cpp
-
10src/storage/jani/EdgeDestination.h
-
4src/storage/jani/EdgeSet.cpp
-
5src/storage/jani/EdgeSet.h
-
4src/storage/jani/Exporter.cpp
-
138src/storage/jani/Model.cpp
-
61src/storage/jani/Model.h
-
4src/storage/jani/Variable.cpp
-
5src/storage/jani/Variable.h
-
120src/storage/jani/VariableSet.cpp
-
112src/storage/jani/VariableSet.h
-
72src/utility/jani.cpp
-
23src/utility/jani.h
@ -0,0 +1,125 @@ |
|||||
|
#include "src/builder/DdJaniModelBuilder.h"
|
||||
|
|
||||
|
#include <sstream>
|
||||
|
|
||||
|
#include <boost/algorithm/string/join.hpp>
|
||||
|
|
||||
|
#include "src/logic/Formulas.h"
|
||||
|
|
||||
|
#include "src/utility/macros.h"
|
||||
|
#include "src/utility/jani.h"
|
||||
|
#include "src/exceptions/InvalidArgumentException.h"
|
||||
|
|
||||
|
namespace storm { |
||||
|
namespace builder { |
||||
|
|
||||
|
template <storm::dd::DdType Type, typename ValueType> |
||||
|
DdJaniModelBuilder<Type, ValueType>::Options::Options() : buildAllRewardModels(true), rewardModelsToBuild(), constantDefinitions(), terminalStates(), negatedTerminalStates() { |
||||
|
// Intentionally left empty.
|
||||
|
} |
||||
|
|
||||
|
template <storm::dd::DdType Type, typename ValueType> |
||||
|
DdJaniModelBuilder<Type, ValueType>::Options::Options(storm::logic::Formula const& formula) : buildAllRewardModels(false), rewardModelsToBuild(), constantDefinitions(), terminalStates(), negatedTerminalStates() { |
||||
|
this->preserveFormula(formula); |
||||
|
this->setTerminalStatesFromFormula(formula); |
||||
|
} |
||||
|
|
||||
|
template <storm::dd::DdType Type, typename ValueType> |
||||
|
DdJaniModelBuilder<Type, ValueType>::Options::Options(std::vector<std::shared_ptr<storm::logic::Formula const>> const& formulas) : buildAllRewardModels(false), rewardModelsToBuild(), constantDefinitions(), terminalStates(), negatedTerminalStates() { |
||||
|
if (formulas.empty()) { |
||||
|
this->buildAllRewardModels = true; |
||||
|
} else { |
||||
|
for (auto const& formula : formulas) { |
||||
|
this->preserveFormula(*formula); |
||||
|
} |
||||
|
if (formulas.size() == 1) { |
||||
|
this->setTerminalStatesFromFormula(*formulas.front()); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
template <storm::dd::DdType Type, typename ValueType> |
||||
|
void DdJaniModelBuilder<Type, ValueType>::Options::preserveFormula(storm::logic::Formula const& formula) { |
||||
|
// If we already had terminal states, we need to erase them.
|
||||
|
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) { |
||||
|
std::set<std::string> referencedRewardModels = formula.getReferencedRewardModels(); |
||||
|
rewardModelsToBuild.insert(referencedRewardModels.begin(), referencedRewardModels.end()); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
template <storm::dd::DdType Type, typename ValueType> |
||||
|
void DdJaniModelBuilder<Type, ValueType>::Options::setTerminalStatesFromFormula(storm::logic::Formula const& formula) { |
||||
|
if (formula.isAtomicExpressionFormula()) { |
||||
|
terminalStates = formula.asAtomicExpressionFormula().getExpression(); |
||||
|
} else if (formula.isEventuallyFormula()) { |
||||
|
storm::logic::Formula const& sub = formula.asEventuallyFormula().getSubformula(); |
||||
|
if (sub.isAtomicExpressionFormula() || sub.isAtomicLabelFormula()) { |
||||
|
this->setTerminalStatesFromFormula(sub); |
||||
|
} |
||||
|
} else if (formula.isUntilFormula()) { |
||||
|
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 (formula.isProbabilityOperatorFormula()) { |
||||
|
storm::logic::Formula const& sub = formula.asProbabilityOperatorFormula().getSubformula(); |
||||
|
if (sub.isEventuallyFormula() || sub.isUntilFormula()) { |
||||
|
this->setTerminalStatesFromFormula(sub); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
template <storm::dd::DdType Type, typename ValueType> |
||||
|
void DdJaniModelBuilder<Type, ValueType>::Options::addConstantDefinitionsFromString(storm::jani::Model const& model, std::string const& constantDefinitionString) { |
||||
|
std::map<storm::expressions::Variable, storm::expressions::Expression> newConstantDefinitions = storm::utility::jani::parseConstantDefinitionString(model, constantDefinitionString); |
||||
|
|
||||
|
// If there is at least one constant that is defined, and the constant definition map does not yet exist,
|
||||
|
// we need to create it.
|
||||
|
if (!constantDefinitions && !newConstantDefinitions.empty()) { |
||||
|
constantDefinitions = std::map<storm::expressions::Variable, storm::expressions::Expression>(); |
||||
|
} |
||||
|
|
||||
|
// Now insert all the entries that need to be defined.
|
||||
|
for (auto const& entry : newConstantDefinitions) { |
||||
|
constantDefinitions.get().insert(entry); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
template <storm::dd::DdType Type, typename ValueType> |
||||
|
DdJaniModelBuilder<Type, ValueType>::DdJaniModelBuilder(storm::jani::Model const& model, Options const& options) : options(options) { |
||||
|
if (options.constantDefinitions) { |
||||
|
this->model = model.defineUndefinedConstants(options.constantDefinitions.get()); |
||||
|
} else { |
||||
|
this->model = model; |
||||
|
} |
||||
|
|
||||
|
if (this->model->hasUndefinedConstants()) { |
||||
|
std::vector<std::reference_wrapper<storm::jani::Constant const>> undefinedConstants = this->model->getUndefinedConstants(); |
||||
|
std::vector<std::string> strings; |
||||
|
for (auto const& constant : undefinedConstants) { |
||||
|
std::stringstream stream; |
||||
|
stream << constant.get().getName() << " (" << constant.get().getType() << ")"; |
||||
|
strings.push_back(stream.str()); |
||||
|
} |
||||
|
STORM_LOG_THROW(false, storm::exceptions::InvalidArgumentException, "Program still contains these undefined constants: " << boost::join(strings, ", ") << "."); |
||||
|
} |
||||
|
|
||||
|
this->model = this->model->substituteConstants(); |
||||
|
} |
||||
|
|
||||
|
template class DdJaniModelBuilder<storm::dd::DdType::CUDD, double>; |
||||
|
template class DdJaniModelBuilder<storm::dd::DdType::Sylvan, double>; |
||||
|
} |
||||
|
} |
@ -0,0 +1,121 @@ |
|||||
|
#pragma once |
||||
|
|
||||
|
#include <boost/optional.hpp> |
||||
|
|
||||
|
#include "src/storage/dd/DdType.h" |
||||
|
|
||||
|
#include "src/logic/Formula.h" |
||||
|
|
||||
|
#include "src/storage/jani/Model.h" |
||||
|
|
||||
|
namespace storm { |
||||
|
namespace models { |
||||
|
namespace symbolic { |
||||
|
template <storm::dd::DdType Type, typename ValueType> |
||||
|
class Model; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
namespace builder { |
||||
|
|
||||
|
template <storm::dd::DdType Type, typename ValueType = double> |
||||
|
class DdJaniModelBuilder { |
||||
|
public: |
||||
|
struct Options { |
||||
|
/*! |
||||
|
* Creates an object representing the default building options. |
||||
|
*/ |
||||
|
Options(); |
||||
|
|
||||
|
/*! Creates an object representing the suggested building options assuming that the given formula is the |
||||
|
* only one to check. Additional formulas may be preserved by calling <code>preserveFormula</code>. |
||||
|
* |
||||
|
* @param formula The formula based on which to choose the building options. |
||||
|
*/ |
||||
|
Options(storm::logic::Formula const& formula); |
||||
|
|
||||
|
/*! Creates an object representing the suggested building options assuming that the given formulas are |
||||
|
* the only ones to check. Additional formulas may be preserved by calling <code>preserveFormula</code>. |
||||
|
* |
||||
|
* @param formula Thes formula based on which to choose the building options. |
||||
|
*/ |
||||
|
Options(std::vector<std::shared_ptr<storm::logic::Formula const>> const& formulas); |
||||
|
|
||||
|
/*! |
||||
|
* Sets the constants definitions from the given string. The string must be of the form 'X=a,Y=b,Z=c', |
||||
|
* etc. where X,Y,Z are the variable names and a,b,c are the values of the constants. |
||||
|
* |
||||
|
* @param program The program managing the constants that shall be defined. Note that the program itself |
||||
|
* is not modified whatsoever. |
||||
|
* @param constantDefinitionString The string from which to parse the constants' values. |
||||
|
*/ |
||||
|
void addConstantDefinitionsFromString(storm::jani::Model const& model, std::string const& constantDefinitionString); |
||||
|
|
||||
|
/*! |
||||
|
* Changes the options in a way that ensures that the given formula can be checked on the model once it |
||||
|
* has been built. |
||||
|
* |
||||
|
* @param formula The formula that is to be ''preserved''. |
||||
|
*/ |
||||
|
void preserveFormula(storm::logic::Formula const& formula); |
||||
|
|
||||
|
/*! |
||||
|
* Analyzes the given formula and sets an expression for the states states of the model that can be |
||||
|
* treated as terminal states. Note that this may interfere with checking properties different than the |
||||
|
* one provided. |
||||
|
* |
||||
|
* @param formula The formula used to (possibly) derive an expression for the terminal states of the |
||||
|
* model. |
||||
|
*/ |
||||
|
void setTerminalStatesFromFormula(storm::logic::Formula const& formula); |
||||
|
|
||||
|
// A flag that indicates whether or not all reward models are to be build. |
||||
|
bool buildAllRewardModels; |
||||
|
|
||||
|
// A list of reward models to be build in case not all reward models are to be build. |
||||
|
std::set<std::string> rewardModelsToBuild; |
||||
|
|
||||
|
// An optional mapping that, if given, contains defining expressions for undefined constants. |
||||
|
boost::optional<std::map<storm::expressions::Variable, storm::expressions::Expression>> constantDefinitions; |
||||
|
|
||||
|
// 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<storm::expressions::Expression> 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<storm::expressions::Expression> negatedTerminalStates; |
||||
|
}; |
||||
|
|
||||
|
/*! |
||||
|
* Creates a builder for the given model that uses the given options. |
||||
|
*/ |
||||
|
DdJaniModelBuilder(storm::jani::Model const& model, Options const& options = Options()); |
||||
|
|
||||
|
/*! |
||||
|
* Translates the given program into a symbolic model (i.e. one that stores the transition relation as a |
||||
|
* decision diagram). |
||||
|
* |
||||
|
* @param model The model to translate. |
||||
|
* @return A pointer to the resulting model. |
||||
|
*/ |
||||
|
std::shared_ptr<storm::models::symbolic::Model<Type, ValueType>> translate(); |
||||
|
|
||||
|
/*! |
||||
|
* Retrieves the model that was actually translated (i.e. including constant substitutions etc.). Note |
||||
|
* that this function may only be called after a succesful translation. |
||||
|
* |
||||
|
* @return The translated model. |
||||
|
*/ |
||||
|
storm::jani::Model const& getTranslatedModel() const; |
||||
|
|
||||
|
private: |
||||
|
/// The model to translate. |
||||
|
boost::optional<storm::jani::Model> model; |
||||
|
|
||||
|
/// The options to use for building the model. |
||||
|
Options options; |
||||
|
}; |
||||
|
|
||||
|
} |
||||
|
} |
@ -0,0 +1,72 @@ |
|||||
|
#include "src/utility/jani.h"
|
||||
|
|
||||
|
#include <boost/algorithm/string.hpp>
|
||||
|
|
||||
|
#include "src/storage/expressions/ExpressionManager.h"
|
||||
|
#include "src/storage/jani/Model.h"
|
||||
|
|
||||
|
#include "src/utility/macros.h"
|
||||
|
#include "src/exceptions/InvalidArgumentException.h"
|
||||
|
|
||||
|
namespace storm { |
||||
|
namespace utility { |
||||
|
namespace jani { |
||||
|
|
||||
|
std::map<storm::expressions::Variable, storm::expressions::Expression> parseConstantDefinitionString(storm::jani::Model const& model, std::string const& constantDefinitionString) { |
||||
|
std::map<storm::expressions::Variable, storm::expressions::Expression> constantDefinitions; |
||||
|
std::set<storm::expressions::Variable> definedConstants; |
||||
|
|
||||
|
if (!constantDefinitionString.empty()) { |
||||
|
// Parse the string that defines the undefined constants of the model and make sure that it contains exactly
|
||||
|
// one value for each undefined constant of the model.
|
||||
|
std::vector<std::string> definitions; |
||||
|
boost::split(definitions, constantDefinitionString, boost::is_any_of(",")); |
||||
|
for (auto& definition : definitions) { |
||||
|
boost::trim(definition); |
||||
|
|
||||
|
// Check whether the token could be a legal constant definition.
|
||||
|
std::size_t positionOfAssignmentOperator = definition.find('='); |
||||
|
STORM_LOG_THROW(positionOfAssignmentOperator != std::string::npos, storm::exceptions::InvalidArgumentException, "Illegal constant definition string: syntax error."); |
||||
|
|
||||
|
// Now extract the variable name and the value from the string.
|
||||
|
std::string constantName = definition.substr(0, positionOfAssignmentOperator); |
||||
|
boost::trim(constantName); |
||||
|
std::string value = definition.substr(positionOfAssignmentOperator + 1); |
||||
|
boost::trim(value); |
||||
|
|
||||
|
// Check whether the constant is a legal undefined constant of the program and if so, of what type it is.
|
||||
|
if (model.hasConstant(constantName)) { |
||||
|
// Get the actual constant and check whether it's in fact undefined.
|
||||
|
auto const& constant = model.getConstant(constantName); |
||||
|
storm::expressions::Variable variable = constant.getExpressionVariable(); |
||||
|
STORM_LOG_THROW(!constant.isDefined(), storm::exceptions::InvalidArgumentException, "Illegally trying to define already defined constant '" << constantName <<"'."); |
||||
|
STORM_LOG_THROW(definedConstants.find(variable) == definedConstants.end(), storm::exceptions::InvalidArgumentException, "Illegally trying to define constant '" << constantName <<"' twice."); |
||||
|
definedConstants.insert(variable); |
||||
|
|
||||
|
if (constant.getType().isBooleanType()) { |
||||
|
if (value == "true") { |
||||
|
constantDefinitions[variable] = model.getExpressionManager().boolean(true); |
||||
|
} else if (value == "false") { |
||||
|
constantDefinitions[variable] = model.getExpressionManager().boolean(false); |
||||
|
} else { |
||||
|
throw storm::exceptions::InvalidArgumentException() << "Illegal value for boolean constant: " << value << "."; |
||||
|
} |
||||
|
} else if (constant.getType().isIntegerType()) { |
||||
|
int_fast64_t integerValue = std::stoi(value); |
||||
|
constantDefinitions[variable] = model.getExpressionManager().integer(integerValue); |
||||
|
} else if (constant.getType().isRationalType()) { |
||||
|
double doubleValue = std::stod(value); |
||||
|
constantDefinitions[variable] = model.getExpressionManager().rational(doubleValue); |
||||
|
} |
||||
|
} else { |
||||
|
STORM_LOG_THROW(false, storm::exceptions::InvalidArgumentException, "Illegal constant definition string: unknown undefined constant '" << constantName << "'."); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
return constantDefinitions; |
||||
|
} |
||||
|
|
||||
|
} |
||||
|
} |
||||
|
} |
@ -0,0 +1,23 @@ |
|||||
|
#pragma once |
||||
|
|
||||
|
#include <map> |
||||
|
|
||||
|
namespace storm { |
||||
|
namespace expressions { |
||||
|
class Variable; |
||||
|
class Expression; |
||||
|
} |
||||
|
|
||||
|
namespace jani { |
||||
|
class Model; |
||||
|
} |
||||
|
|
||||
|
namespace utility { |
||||
|
namespace jani { |
||||
|
|
||||
|
std::map<storm::expressions::Variable, storm::expressions::Expression> parseConstantDefinitionString(storm::jani::Model const& model, std::string const& constantDefinitionString); |
||||
|
|
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
Write
Preview
Loading…
Cancel
Save
Reference in new issue