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