/* * Z3ExpressionAdapter.h * * Created on: 04.10.2013 * Author: Christian Dehnert */ #ifndef STORM_ADAPTERS_Z3EXPRESSIONADAPTER_H_ #define STORM_ADAPTERS_Z3EXPRESSIONADAPTER_H_ #include #include "src/ir/expressions/ExpressionVisitor.h" namespace storm { namespace adapters { class Z3ExpressionAdapter : public storm::ir::expressions::ExpressionVisitor { public: /*! * Creates a Z3ExpressionAdapter over the given Z3 context. * * @param context The Z3 context over which to build the expressions. */ Z3ExpressionAdapter(z3::context const& context, std::map const& variableToExpressionMap) : context(context), stack(), variableToExpressionMap(variableToExpressionMap) { // Intentionally left empty. } /*! * Translates the given expression to an equivalent expression for Z3. * * @param expression The expression to translate. * @return An equivalent expression for Z3. */ z3::expr translateExpression(std::shared_ptr expression) { expression->accept(this); return stack.top(); } virtual void visit(BinaryBooleanFunctionExpression* expression) { expression->getLeft()->accept(this); expression->getRight()->accept(this); z3::expr rightResult = stack.top(); stack.pop(); z3::expr leftResult = stack.top(); stack.pop(); switch(expression->getFunctionType()) { case storm::ir::expressions::BinaryBooleanFunctionExpression::AND: stack.push(leftResult && rightResult); break; case storm::ir::expressions::BinaryBooleanFunctionExpression::OR: stack.push(leftResult || rightResult); break; default: throw storm::exceptions::ExpressionEvaluationException() << "Cannot evaluate expression: " << "Unknown boolean binary operator: '" << expression->getFunctionType() << "'."; } } virtual void visit(BinaryNumericalFunctionExpression* expression) { expression->getLeft()->accept(this); expression->getRight()->accept(this); z3::expr rightResult = stack.top(); stack.pop(); z3::expr leftResult = stack.top(); stack.pop(); switch(expression->getFunctionType()) { case storm::ir::expressions::BinaryNumericalFunctionExpression::PLUS: stack.push(leftResult + rightResult); break; case storm::ir::expressions::BinaryNumericalFunctionExpression::MINUS: stack.push(leftResult - rightResult); break; case storm::ir::expressions::BinaryNumericalFunctionExpression::TIMES: stack.push(leftResult * rightResult); break; case storm::ir::expressions::BinaryNumericalFunctionExpression::DIVIDE: stack.push(leftResult / rightResult); break; default: throw storm::exceptions::ExpressionEvaluationException() << "Cannot evaluate expression: " << "Unknown boolean binary operator: '" << expression->getFunctionType() << "'."; } } virtual void visit(BinaryRelationExpression* expression) { expression->getLeft()->accept(this); expression->getRight()->accept(this); z3::expr rightResult = stack.top(); stack.pop(); z3::expr leftResult = stack.top(); stack.pop(); switch(expression->getRelationType()) { case storm::ir::expressions::BinaryRelationExpression::EQUAL: stack.push(leftResult == rightResult); break; case storm::ir::expressions::BinaryRelationExpression::NOT_EQUAL: stack.push(leftResult != rightResult); break; case storm::ir::expressions::BinaryRelationExpression::LESS: stack.push(leftResult < rightResult); break; case storm::ir::expressions::BinaryRelationExpression::LESS_OR_EQUAL: stack.push(leftResult <= rightResult); break; case storm::ir::expressions::BinaryRelationExpression::GREATER: stack.push(leftResult > rightResult); break; case storm::ir::expressions::BinaryRelationExpression::GREATER_OR_EQUAL: stack.push(leftResult >= rightResult); break; default: throw storm::exceptions::ExpressionEvaluationException() << "Cannot evaluate expression: " << "Unknown boolean binary operator: '" << expression->getRelationType() << "'."; } } virtual void visit(BooleanConstantExpression* expression) { if (!expression->isDefined()) { throw storm::exceptions::ExpressionEvaluationException() << "Cannot evaluate expression: " << ". Boolean constant '" << expression->getConstantName() << "' is undefined."; } stack.push(context.bool_val(expression->getValue())); } virtual void visit(BooleanLiteralExpression* expression) { stack.push(context.bool_val(expression->getValueAsBool(nullptr)))); } virtual void visit(DoubleConstantExpression* expression) { if (!expression->isDefined()) { throw storm::exceptions::ExpressionEvaluationException() << "Cannot evaluate expression: " << ". Double constant '" << expression->getConstantName() << "' is undefined."; } // FIXME: convert double value to suitable format. stack.push(context.real_val(expression->getValue())); } virtual void visit(DoubleLiteralExpression* expression) { // FIXME: convert double value to suitable format. stack.push(context.real_val(expression->getValue())); } virtual void visit(IntegerConstantExpression* expression) { if (!expression->isDefined()) { throw storm::exceptions::ExpressionEvaluationException() << "Cannot evaluate expression: " << ". Integer constant '" << expression->getConstantName() << "' is undefined."; } stack.push(context.int_val(expression->getValue())); } virtual void visit(IntegerLiteralExpression* expression) { stack.push(context.int_val(expression->getValue())); } virtual void visit(UnaryBooleanFunctionExpression* expression) { expression->getChild()->accept(this); z3::expr childResult = stack.top(); stack.pop(); switch (expression->getFunctionType()) { case storm::ir::expressions::UnaryBooleanFunctionExpression::NOT: stack.push(!childResult); break; default: throw storm::exceptions::ExpressionEvaluationException() << "Cannot evaluate expression: " << "Unknown boolean unary operator: '" << expression->getFunctionType() << "'."; } } virtual void visit(UnaryNumericalFunctionExpression* expression) { expression->getChild()->accept(this); z3::expr childResult = stack.top(); stack.pop(); switch(expression->getFunctionType()) { case storm::ir::expressions::UnaryNumericalFunctionExpression::MINUS: stack.push(0 - childResult); break; default: throw storm::exceptions::ExpressionEvaluationException() << "Cannot evaluate expression: " << "Unknown numerical unary operator: '" << expression->getFunctionType() << "'."; } } virtual void visit(VariableExpression* expression) { stack.push(variableToExpressionMap.at(expression->getVariableName()); } private: z3::context context; std::stack stack; std::map variableToExpressionMap } } // namespace adapters } // namespace storm #endif /* STORM_ADAPTERS_Z3EXPRESSIONADAPTER_H_ */