#include "storm/storage/geometry/NativePolytope.h" #include "storm/utility/macros.h" #include "storm/utility/solver.h" #include "storm/solver/Z3LpSolver.h" #include "storm/solver/SmtSolver.h" #include "storm/storage/geometry/nativepolytopeconversion/QuickHull.h" #include "storm/storage/geometry/nativepolytopeconversion/HyperplaneEnumeration.h" #include "storm/storage/expressions/ExpressionManager.h" #include "storm/exceptions/InvalidArgumentException.h" #include "storm/exceptions/UnexpectedException.h" #include "storm/exceptions/NotImplementedException.h" namespace storm { namespace storage { namespace geometry { template NativePolytope::NativePolytope(std::vector> const& halfspaces) { if (halfspaces.empty()){ // The polytope is universal emptyStatus = EmptyStatus::Nonempty; } else { StormEigen::Index maxCol = halfspaces.front().normalVector().size(); StormEigen::Index maxRow = halfspaces.size(); A = EigenMatrix(maxRow, maxCol); b = EigenVector(maxRow); for (StormEigen::Index row = 0; row < A.rows(); ++row ){ assert((StormEigen::Index) halfspaces[row].normalVector().size() == maxCol); b(row) = halfspaces[row].offset(); A.row(row) = storm::adapters::EigenAdapter::toEigenVector(halfspaces[row].normalVector()); } emptyStatus = EmptyStatus::Unknown; } } template NativePolytope::NativePolytope(std::vector const& points) { if (points.empty()){ emptyStatus = EmptyStatus::Empty; } else { std::vector eigenPoints; eigenPoints.reserve(points.size()); for (auto const& p : points){ eigenPoints.emplace_back(storm::adapters::EigenAdapter::toEigenVector(p)); } storm::storage::geometry::QuickHull qh; qh.generateHalfspacesFromPoints(eigenPoints, false); A = std::move(qh.getResultMatrix()); b = std::move(qh.getResultVector()); emptyStatus = EmptyStatus::Nonempty; } } template std::shared_ptr> NativePolytope::create(boost::optional>> const& halfspaces, boost::optional> const& points) { if (halfspaces) { STORM_LOG_WARN_COND(!points, "Creating a NativePolytope where halfspaces AND points are given. The points will be ignored."); return std::make_shared>(*halfspaces); } else if (points) { return std::make_shared>(*points); } STORM_LOG_THROW(false, storm::exceptions::UnexpectedException, "Creating a NativePolytope but no representation was given."); return nullptr; } template NativePolytope::NativePolytope(NativePolytope const& other) : emptyStatus(other.emptyStatus), A(other.A), b(other.b) { // Intentionally left empty } template NativePolytope::NativePolytope(NativePolytope&& other) : emptyStatus(std::move(other.emptyStatus)), A(std::move(other.A)), b(std::move(other.b)) { // Intentionally left empty } template NativePolytope::NativePolytope(EmptyStatus const& emptyStatus, EigenMatrix const& halfspaceMatrix, EigenVector const& halfspaceVector) : emptyStatus(emptyStatus), A(halfspaceMatrix), b(halfspaceVector) { // Intentionally left empty } template NativePolytope::NativePolytope(EmptyStatus&& emptyStatus, EigenMatrix&& halfspaceMatrix, EigenVector&& halfspaceVector) : emptyStatus(emptyStatus), A(halfspaceMatrix), b(halfspaceVector) { // Intentionally left empty } template NativePolytope::~NativePolytope() { // Intentionally left empty } template std::vector::Point> NativePolytope::getVertices() const { std::vector eigenVertices = getEigenVertices(); std::vector result; result.reserve(eigenVertices.size()); for (auto const& p : eigenVertices) { result.push_back(storm::adapters::EigenAdapter::toStdVector(p)); } return result; } template std::vector> NativePolytope::getHalfspaces() const { std::vector> result; result.reserve(A.rows()); for (StormEigen::Index row=0; row < A.rows(); ++row){ result.emplace_back(storm::adapters::EigenAdapter::toStdVector(EigenVector(A.row(row))), b(row)); } return result; } template bool NativePolytope::isEmpty() const { if (emptyStatus == EmptyStatus::Unknown) { std::shared_ptr manager(new storm::expressions::ExpressionManager()); std::unique_ptr solver = storm::utility::solver::SmtSolverFactory().create(*manager); std::vector constraints = getConstraints(*manager, declareVariables(*manager, "x")); for (auto const& constraint : constraints) { solver->add(constraint); } switch(solver->check()) { case storm::solver::SmtSolver::CheckResult::Sat: emptyStatus = EmptyStatus::Nonempty; break; case storm::solver::SmtSolver::CheckResult::Unsat: emptyStatus = EmptyStatus::Empty; break; default: STORM_LOG_THROW(false, storm::exceptions::UnexpectedException, "Unexpected result of SMT solver during emptyness-check of Polytope."); break; } } return emptyStatus == EmptyStatus::Empty; } template bool NativePolytope::isUniversal() const { return A.rows()==0; } template bool NativePolytope::contains(Point const& point) const{ EigenVector x = storm::adapters::EigenAdapter::toEigenVector(point); for (StormEigen::Index row=0; row < A.rows(); ++row){ if ((A.row(row) * x)(0) > b(row)){ return false; } } return true; } template bool NativePolytope::contains(std::shared_ptr> const& other) const { STORM_LOG_THROW(other->isNativePolytope(), storm::exceptions::InvalidArgumentException, "Invoked operation between a NativePolytope and a different polytope implementation. This is not supported"); if (this->isUniversal()) { return true; } else if (other->isUniversal()) { return false; } else { // Check whether there is one point in other that is not in this std::shared_ptr manager(new storm::expressions::ExpressionManager()); std::unique_ptr solver = storm::utility::solver::SmtSolverFactory().create(*manager); std::vector variables = declareVariables(*manager, "x"); std::vector constraints = getConstraints(*manager, variables); storm::expressions::Expression constraintsThis = manager->boolean(true); for (auto const& constraint : constraints) { constraintsThis = constraintsThis && constraint; } solver->add(!constraintsThis); constraints = dynamic_cast const&>(*other).getConstraints(*manager, variables); for (auto const& constraint : constraints) { solver->add(constraint); } switch(solver->check()) { case storm::solver::SmtSolver::CheckResult::Sat: return false; case storm::solver::SmtSolver::CheckResult::Unsat: return true; default: STORM_LOG_THROW(false, storm::exceptions::UnexpectedException, "Unexpected result of SMT solver during containment check of two polytopes."); return false; } } } template std::shared_ptr> NativePolytope::intersection(std::shared_ptr> const& rhs) const{ STORM_LOG_THROW(rhs->isNativePolytope(), storm::exceptions::InvalidArgumentException, "Invoked operation between a NativePolytope and a different polytope implementation. This is not supported"); NativePolytope const& nativeRhs = dynamic_cast const&>(*rhs); EigenMatrix resultA(A.rows() + nativeRhs.A.rows(), A.cols()); resultA << A, nativeRhs.A; EigenVector resultb(resultA.rows()); resultb << b, nativeRhs.b; return std::make_shared>(EmptyStatus::Unknown, std::move(resultA), std::move(resultb)); } template std::shared_ptr> NativePolytope::intersection(Halfspace const& halfspace) const{ if (A.rows() == 0) { // No constraints yet EigenMatrix resultA = storm::adapters::EigenAdapter::toEigenVector(halfspace.normalVector()).transpose(); EigenVector resultb(1); resultb(0) = halfspace.offset(); return std::make_shared>(EmptyStatus::Unknown, std::move(resultA), std::move(resultb)); } EigenMatrix resultA(A.rows() + 1, A.cols()); resultA << A, storm::adapters::EigenAdapter::toEigenVector(halfspace.normalVector()).transpose(); EigenVector resultb(resultA.rows()); resultb << b, halfspace.offset(); return std::make_shared>(EmptyStatus::Unknown, std::move(resultA), std::move(resultb)); } template std::shared_ptr> NativePolytope::convexUnion(std::shared_ptr> const& rhs) const{ STORM_LOG_THROW(rhs->isNativePolytope(), storm::exceptions::InvalidArgumentException, "Invoked operation between a NativePolytope and a different polytope implementation. This is not supported"); if (this->isEmpty()) { return std::make_shared>(dynamic_cast const&>(*rhs)); } else if (rhs->isEmpty()) { return std::make_shared>(*this); } else if (this->isUniversal() || rhs->isUniversal()) { return std::make_shared>(std::vector>()); } STORM_LOG_WARN("Implementation of convex union of two polytopes only works if the polytopes are bounded. This is not checked."); std::vector rhsVertices = dynamic_cast const&>(*rhs).getEigenVertices(); std::vector resultVertices = this->getEigenVertices(); resultVertices.insert(resultVertices.end(), std::make_move_iterator(rhsVertices.begin()), std::make_move_iterator(rhsVertices.end())); storm::storage::geometry::QuickHull qh; qh.generateHalfspacesFromPoints(resultVertices, false); return std::make_shared>(EmptyStatus::Nonempty, std::move(qh.getResultMatrix()), std::move(qh.getResultVector())); } template std::shared_ptr> NativePolytope::minkowskiSum(std::shared_ptr> const& rhs) const { STORM_LOG_THROW(rhs->isNativePolytope(), storm::exceptions::InvalidArgumentException, "Invoked operation between a NativePolytope and a different polytope implementation. This is not supported"); NativePolytope const& nativeRhs = dynamic_cast const&>(*rhs); if (this->isEmpty() || nativeRhs.isEmpty()) { return std::make_shared>(std::vector()); } std::vector> resultConstraints; resultConstraints.reserve(A.rows() + nativeRhs.A.rows()); // evaluation of rhs in directions of lhs for (StormEigen::Index i = 0; i < A.rows(); ++i) { auto optimizationRes = nativeRhs.optimize(A.row(i)); if ( optimizationRes.second ) { resultConstraints.emplace_back(A.row(i), b(i) + (A.row(i) * optimizationRes.first)(0)); } // If optimizationRes.second is false, it means that rhs is unbounded in this direction, i.e., the current constraint is not inserted } // evaluation of lhs in directions of rhs for (StormEigen::Index i = 0; i < nativeRhs.A.rows(); ++i) { auto optimizationRes = optimize(nativeRhs.A.row(i)); if ( optimizationRes.second ) { resultConstraints.emplace_back(nativeRhs.A.row(i), nativeRhs.b(i) + (nativeRhs.A.row(i) * optimizationRes.first)(0)); } // If optimizationRes.second is false, it means that rhs is unbounded in this direction, i.e., the current constraint is not inserted } if (resultConstraints.empty()) { return std::make_shared>(std::vector>()); } else { EigenMatrix newA(resultConstraints.size(), resultConstraints.front().first.rows()); EigenVector newb(resultConstraints.size()); for (StormEigen::Index i = 0; i < newA.rows(); ++i) { newA.row(i) = resultConstraints[i].first; newb(i) = resultConstraints[i].second; } return std::make_shared>(EmptyStatus::Nonempty, std::move(newA), std::move(newb)); } } template std::shared_ptr> NativePolytope::affineTransformation(std::vector const& matrix, Point const& vector) const { STORM_LOG_THROW(!matrix.empty(), storm::exceptions::InvalidArgumentException, "Invoked affine transformation with a matrix without rows."); StormEigen::Index rows = matrix.size(); StormEigen::Index columns = matrix.front().size(); EigenMatrix eigenMatrix(rows, columns); for (StormEigen::Index row = 0; row < rows; ++row) { eigenMatrix.row(row) = storm::adapters::EigenAdapter::toEigenVector(matrix[row]); } EigenVector eigenVector = storm::adapters::EigenAdapter::toEigenVector(vector); StormEigen::FullPivLU luMatrix( eigenMatrix ); STORM_LOG_THROW(luMatrix.isInvertible(), storm::exceptions::NotImplementedException, "Affine Transformation of native polytope only implemented if the transformation matrix is invertable"); EigenMatrix newA = A * luMatrix.inverse(); EigenVector newb = b + (newA * eigenVector); return std::make_shared>(emptyStatus, std::move(newA), std::move(newb)); } template std::pair::Point, bool> NativePolytope::optimize(Point const& direction) const { if (isUniversal()) { return std::make_pair(Point(), false); } storm::solver::Z3LpSolver solver(storm::solver::OptimizationDirection::Maximize); std::vector variables; variables.reserve(A.cols()); for (StormEigen::Index i = 0; i < A.cols(); ++i) { variables.push_back(solver.addUnboundedContinuousVariable("x" + std::to_string(i), direction[i])); } std::vector constraints = getConstraints(solver.getManager(), variables); for (auto const& constraint : constraints) { solver.addConstraint("", constraint); } solver.update(); solver.optimize(); if (solver.isOptimal()) { auto result = std::make_pair(Point(), true); result.first.reserve(variables.size()); for (auto const& var : variables) { result.first.push_back(solver.getContinuousValue(var)); } return result; } else { //solution is infinity or infeasible return std::make_pair(Point(), false); } } template std::pair::EigenVector, bool> NativePolytope::optimize(EigenVector const& direction) const { if (isUniversal()) { return std::make_pair(EigenVector(), false); } storm::solver::Z3LpSolver solver(storm::solver::OptimizationDirection::Maximize); std::vector variables; variables.reserve(A.cols()); for (StormEigen::Index i = 0; i < A.cols(); ++i) { variables.push_back(solver.addUnboundedContinuousVariable("x" + std::to_string(i), static_cast(direction(i)))); } std::vector constraints = getConstraints(solver.getManager(), variables); for (auto const& constraint: constraints) { solver.addConstraint("", constraint); } solver.update(); solver.optimize(); if (solver.isOptimal()) { auto result = std::make_pair(EigenVector(A.cols()), true); for (StormEigen::Index i = 0; i < A.cols(); ++i) { result.first(i) = solver.getContinuousValue(variables[i]); } return result; } else { //solution is infinity or infeasible return std::make_pair(EigenVector(), false); } } template bool NativePolytope::isNativePolytope() const { return true; } template std::vector::EigenVector> NativePolytope::getEigenVertices() const { storm::storage::geometry::HyperplaneEnumeration he; he.generateVerticesFromConstraints(A, b, false); return he.getResultVertices(); } template std::vector NativePolytope::declareVariables(storm::expressions::ExpressionManager& manager, std::string const& namePrefix) const { std::vector result; result.reserve(A.cols()); for (StormEigen::Index col=0; col < A.cols(); ++col){ result.push_back(manager.declareVariable(namePrefix + std::to_string(col), manager.getRationalType())); } return result; } template std::vector NativePolytope::getConstraints(storm::expressions::ExpressionManager const& manager, std::vector const& variables) const { std::vector result; for (StormEigen::Index row = 0; row < A.rows(); ++row) { storm::expressions::Expression lhs = manager.rational(A(row,0)) * variables[0].getExpression(); for (StormEigen::Index col=1; col < A.cols(); ++col) { lhs = lhs + manager.rational(A(row,col)) * variables[col].getExpression(); } result.push_back(lhs <= manager.rational(b(row))); } return result; } template std::shared_ptr> NativePolytope::clean() { return create(boost::none, getVertices()); } template class NativePolytope; #ifdef STORM_HAVE_CARL template class NativePolytope; #endif } } }