/* -*- mode: C++; c-basic-offset: 2; indent-tabs-mode: nil -*- */ /* * Main authors: * Guido Tack */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include #include #include namespace MiniZinc { void OptimizeRegistry::reg(const MiniZinc::ASTString& call, optimizer opt) { _m.insert(std::make_pair(call, opt)); } OptimizeRegistry::ConstraintStatus OptimizeRegistry::process(EnvI& env, MiniZinc::Item* i, MiniZinc::Call* c, Expression*& rewrite) { ASTStringMap::t::iterator it = _m.find(c->id()); if (it != _m.end()) { return it->second(env, i, c, rewrite); } return CS_NONE; } OptimizeRegistry& OptimizeRegistry::registry(void) { static OptimizeRegistry reg; return reg; } namespace Optimizers { OptimizeRegistry::ConstraintStatus o_linear(EnvI& env, Item* ii, Call* c, Expression*& rewrite) { ArrayLit* al_c = eval_array_lit(env, c->arg(0)); std::vector coeffs(al_c->size()); for (unsigned int i = 0; i < al_c->size(); i++) { coeffs[i] = eval_int(env, (*al_c)[i]); } ArrayLit* al_x = eval_array_lit(env, c->arg(1)); std::vector x(al_x->size()); for (unsigned int i = 0; i < al_x->size(); i++) { x[i] = (*al_x)[i]; } IntVal d = 0; simplify_lin(coeffs, x, d); if (coeffs.size() == 0) { bool failed; if (c->id() == constants().ids.int_.lin_le) { failed = (d > eval_int(env, c->arg(2))); } else if (c->id() == constants().ids.int_.lin_eq) { failed = (d != eval_int(env, c->arg(2))); } else { failed = (d == eval_int(env, c->arg(2))); } if (failed) { return OptimizeRegistry::CS_FAILED; } else { return OptimizeRegistry::CS_ENTAILED; } } else if (coeffs.size() == 1 && (ii->isa() || ii->cast()->e()->ti()->domain() == constants().lit_true)) { VarDecl* vd = x[0]()->cast()->decl(); IntSetVal* domain = vd->ti()->domain() ? eval_intset(env, vd->ti()->domain()) : NULL; if (c->id() == constants().ids.int_.lin_eq) { IntVal rd = eval_int(env, c->arg(2)) - d; if (rd % coeffs[0] == 0) { IntVal nd = rd / coeffs[0]; if (domain && !domain->contains(nd)) return OptimizeRegistry::CS_FAILED; std::vector args(2); args[0] = x[0](); args[1] = IntLit::a(nd); Call* c = new Call(Location(), constants().ids.int_.eq, args); c->type(Type::varbool()); rewrite = c; return OptimizeRegistry::CS_REWRITE; } else { return OptimizeRegistry::CS_FAILED; } } else if (c->id() == constants().ids.int_.lin_le) { IntVal ac = std::abs(coeffs[0]); IntVal rd = eval_int(env, c->arg(2)) - d; IntVal ad = std::abs(rd); IntVal nd; if (ad % ac == 0) { nd = rd / coeffs[0]; } else { double nd_d = static_cast(ad.toInt()) / static_cast(ac.toInt()); if (coeffs[0] >= 0 && rd >= 0) { nd = static_cast(std::floor(nd_d)); } else if (rd >= 0) { nd = -static_cast(std::floor(nd_d)); } else if (coeffs[0] >= 0) { nd = -static_cast(std::ceil(nd_d)); } else { nd = static_cast(std::ceil(nd_d)); } } bool swapSign = coeffs[0] < 0; if (domain) { if (swapSign) { if (domain->max() < nd) { return OptimizeRegistry::CS_FAILED; } else if (domain->min() >= nd) return OptimizeRegistry::CS_ENTAILED; } else { if (domain->min() > nd) { return OptimizeRegistry::CS_FAILED; } else if (domain->max() <= nd) return OptimizeRegistry::CS_ENTAILED; } std::vector args(2); args[0] = x[0](); args[1] = IntLit::a(nd); if (swapSign) std::swap(args[0], args[1]); Call* nc = new Call(Location(), constants().ids.int_.le, args); nc->type(Type::varbool()); rewrite = nc; return OptimizeRegistry::CS_REWRITE; } } } else if (c->id() == constants().ids.int_.lin_eq && coeffs.size() == 2 && ((coeffs[0] == 1 && coeffs[1] == -1) || (coeffs[1] == 1 && coeffs[0] == -1)) && eval_int(env, c->arg(2)) - d == 0) { std::vector args(2); args[0] = x[0](); args[1] = x[1](); Call* c = new Call(Location(), constants().ids.int_.eq, args); rewrite = c; return OptimizeRegistry::CS_REWRITE; } if (coeffs.size() < al_c->size()) { std::vector coeffs_e(coeffs.size()); std::vector x_e(coeffs.size()); for (unsigned int i = 0; i < coeffs.size(); i++) { coeffs_e[i] = IntLit::a(coeffs[i]); x_e[i] = x[i](); } ArrayLit* al_c_new = new ArrayLit(al_c->loc(), coeffs_e); al_c_new->type(Type::parint(1)); ArrayLit* al_x_new = new ArrayLit(al_x->loc(), x_e); al_x_new->type(al_x->type()); std::vector args(3); args[0] = al_c_new; args[1] = al_x_new; args[2] = IntLit::a(eval_int(env, c->arg(2)) - d); Call* nc = new Call(Location(), c->id(), args); nc->type(Type::varbool()); for (ExpressionSetIter it = c->ann().begin(); it != c->ann().end(); ++it) { nc->addAnnotation(*it); } rewrite = nc; return OptimizeRegistry::CS_REWRITE; } return OptimizeRegistry::CS_OK; } OptimizeRegistry::ConstraintStatus o_lin_exp(EnvI& env, Item* i, Call* c, Expression*& rewrite) { if (c->type().isint()) { ArrayLit* al_c = eval_array_lit(env, c->arg(0)); std::vector coeffs(al_c->size()); for (unsigned int i = 0; i < al_c->size(); i++) { coeffs[i] = eval_int(env, (*al_c)[i]); } ArrayLit* al_x = eval_array_lit(env, c->arg(1)); std::vector x(al_x->size()); for (unsigned int i = 0; i < al_x->size(); i++) { x[i] = (*al_x)[i]; } IntVal d = eval_int(env, c->arg(2)); simplify_lin(coeffs, x, d); if (coeffs.size() == 0) { rewrite = IntLit::a(d); return OptimizeRegistry::CS_REWRITE; } else if (coeffs.size() < al_c->size()) { if (coeffs.size() == 1 && coeffs[0] == 1 && d == 0) { rewrite = x[0](); return OptimizeRegistry::CS_REWRITE; } std::vector coeffs_e(coeffs.size()); std::vector x_e(coeffs.size()); for (unsigned int i = 0; i < coeffs.size(); i++) { coeffs_e[i] = IntLit::a(coeffs[i]); x_e[i] = x[i](); } ArrayLit* al_c_new = new ArrayLit(al_c->loc(), coeffs_e); al_c_new->type(Type::parint(1)); ArrayLit* al_x_new = new ArrayLit(al_x->loc(), x_e); al_x_new->type(al_x->type()); std::vector args(3); args[0] = al_c_new; args[1] = al_x_new; args[2] = IntLit::a(d); Call* nc = new Call(Location(), c->id(), args); nc->type(c->type()); for (ExpressionSetIter it = c->ann().begin(); it != c->ann().end(); ++it) { nc->addAnnotation(*it); } rewrite = nc; return OptimizeRegistry::CS_REWRITE; } } return OptimizeRegistry::CS_OK; } OptimizeRegistry::ConstraintStatus o_element(EnvI& env, Item* i, Call* c, Expression*& rewrite) { if (c->arg(0)->isa()) { IntVal idx = eval_int(env, c->arg(0)); ArrayLit* al = eval_array_lit(env, c->arg(1)); if (idx < 1 || idx > al->size()) { return OptimizeRegistry::CS_FAILED; } Expression* result = (*al)[static_cast(idx.toInt()) - 1]; std::vector args(2); args[0] = result; args[1] = c->arg(2); Call* eq = new Call(Location(), constants().ids.int_.eq, args); rewrite = eq; return OptimizeRegistry::CS_REWRITE; } return OptimizeRegistry::CS_OK; } OptimizeRegistry::ConstraintStatus o_clause(EnvI& env, Item* i, Call* c, Expression*& rewrite) { std::vector pos; std::vector neg; ArrayLit* al_pos = eval_array_lit(env, c->arg(0)); for (unsigned int i = 0; i < al_pos->size(); i++) { if (Id* ident = (*al_pos)[i]->dyn_cast()) { if (ident->decl()->ti()->domain() == NULL) pos.push_back(ident->decl()); } } ArrayLit* al_neg = eval_array_lit(env, c->arg(1)); for (unsigned int i = 0; i < al_neg->size(); i++) { if (Id* ident = (*al_neg)[i]->dyn_cast()) { if (ident->decl()->ti()->domain() == NULL) neg.push_back(ident->decl()); } } bool subsumed = false; if (pos.size() > 0 && neg.size() > 0) { std::sort(pos.begin(), pos.end()); std::sort(neg.begin(), neg.end()); unsigned int ix = 0; unsigned int iy = 0; for (;;) { if (pos[ix] == neg[iy]) { subsumed = true; break; } if (pos[ix] < neg[iy]) { ix++; } else { iy++; } if (ix == pos.size() || iy == neg.size()) break; } } if (subsumed) { return OptimizeRegistry::CS_ENTAILED; } else { return OptimizeRegistry::CS_OK; } } OptimizeRegistry::ConstraintStatus o_div(EnvI& env, Item* i, Call* c, Expression*& rewrite) { if (c->arg(1)->type().ispar()) { IntVal c1v = eval_int(env, c->arg(1)); if (c->arg(0)->type().ispar() && c->n_args() == 3 && c->arg(2)->type().ispar()) { IntVal c0v = eval_int(env, c->arg(0)); IntVal c2v = eval_int(env, c->arg(2)); return (c0v / c1v == c2v) ? OptimizeRegistry::CS_ENTAILED : OptimizeRegistry::CS_FAILED; } } return OptimizeRegistry::CS_OK; } OptimizeRegistry::ConstraintStatus o_set_in(EnvI& env, Item* i, Call* c, Expression*& rewrite) { if (c->arg(1)->type().ispar()) { if (c->arg(0)->type().ispar()) { IntSetVal* isv = eval_intset(env, c->arg(1)); return isv->contains(eval_int(env, c->arg(0))) ? OptimizeRegistry::CS_ENTAILED : OptimizeRegistry::CS_FAILED; } else if (Id* ident = c->arg(0)->dyn_cast()) { VarDecl* vd = ident->decl(); IntSetVal* isv = eval_intset(env, c->arg(1)); if (vd->ti()->domain()) { IntSetVal* dom = eval_intset(env, vd->ti()->domain()); { IntSetRanges isv_r(isv); IntSetRanges dom_r(dom); if (Ranges::subset(dom_r, isv_r)) return OptimizeRegistry::CS_ENTAILED; } { IntSetRanges isv_r(isv); IntSetRanges dom_r(dom); if (Ranges::disjoint(dom_r, isv_r)) return OptimizeRegistry::CS_FAILED; } } else if (isv->min() == isv->max()) { std::vector args(2); args[0] = vd->id(); args[1] = IntLit::a(isv->min()); Call* eq = new Call(Location(), constants().ids.int_.eq, args); rewrite = eq; return OptimizeRegistry::CS_REWRITE; } } } return OptimizeRegistry::CS_OK; } OptimizeRegistry::ConstraintStatus o_int_ne(EnvI& env, Item* i, Call* c, Expression*& rewrite) { Expression* e0 = c->arg(0); Expression* e1 = c->arg(1); if (e0->type().ispar() && e1->type().ispar()) { return eval_int(env, e0) != eval_int(env, e1) ? OptimizeRegistry::CS_ENTAILED : OptimizeRegistry::CS_FAILED; } if (e1->isa()) { std::swap(e0, e1); } if (Id* ident = e0->dyn_cast()) { if (e1->type().ispar()) { if (ident->decl()->ti()->domain()) { IntVal e1v = eval_int(env, e1); IntSetVal* isv = eval_intset(env, ident->decl()->ti()->domain()); if (!isv->contains(e1v)) return OptimizeRegistry::CS_ENTAILED; if (e1v == isv->min() && e1v == isv->max()) return OptimizeRegistry::CS_FAILED; } } } return OptimizeRegistry::CS_OK; } class Register { private: Model* _keepAliveModel; public: Register(void) { GCLock lock; _keepAliveModel = new Model; ASTString id_element("array_int_element"); ASTString id_var_element("array_var_int_element"); std::vector e; e.push_back(new StringLit(Location(), id_element)); e.push_back(new StringLit(Location(), id_var_element)); _keepAliveModel->addItem(new ConstraintI(Location(), new ArrayLit(Location(), e))); OptimizeRegistry::registry().reg(constants().ids.int_.lin_eq, o_linear); OptimizeRegistry::registry().reg(constants().ids.int_.lin_le, o_linear); OptimizeRegistry::registry().reg(constants().ids.int_.lin_ne, o_linear); OptimizeRegistry::registry().reg(constants().ids.int_.div, o_div); OptimizeRegistry::registry().reg(id_element, o_element); OptimizeRegistry::registry().reg(constants().ids.lin_exp, o_lin_exp); OptimizeRegistry::registry().reg(id_var_element, o_element); OptimizeRegistry::registry().reg(constants().ids.clause, o_clause); OptimizeRegistry::registry().reg(constants().ids.bool_clause, o_clause); OptimizeRegistry::registry().reg(constants().ids.set_in, o_set_in); OptimizeRegistry::registry().reg(constants().ids.int_.ne, o_int_ne); } ~Register(void) { delete _keepAliveModel; } } _r; } // namespace Optimizers } // namespace MiniZinc