/* -*- mode: C++; c-basic-offset: 2; indent-tabs-mode: nil -*- */ /* * Main authors: * Christian Schulte * * Copyright: * Christian Schulte, 2003 * * This file is part of Gecode, the generic constraint * development environment: * http://www.gecode.org * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the * "Software"), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to * the following conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. * */ #include namespace Gecode { namespace Int { namespace Linear { /** * \brief Test if only unit-coefficient arrays used * */ template forceinline bool isunit(ViewArray

&, ViewArray&) { return false; } template<> forceinline bool isunit(ViewArray&, ViewArray&) { return true; } template<> forceinline bool isunit(ViewArray&, ViewArray&) { return true; } template<> forceinline bool isunit(ViewArray&, ViewArray&) { return true; } /* * Linear propagators * */ template forceinline Lin::Lin(Home home, ViewArray

& x0, ViewArray& y0, Val c0) : Propagator(home), x(x0), y(y0), c(c0) { x.subscribe(home,*this,pc); y.subscribe(home,*this,pc); } template forceinline Lin::Lin(Space& home, Lin& p) : Propagator(home,p), c(p.c) { x.update(home,p.x); y.update(home,p.y); } template PropCost Lin::cost(const Space&, const ModEventDelta&) const { return PropCost::linear(PropCost::LO, x.size()+y.size()); } template void Lin::reschedule(Space& home) { x.reschedule(home,*this,pc); y.reschedule(home,*this,pc); } template forceinline size_t Lin::dispose(Space& home) { x.cancel(home,*this,pc); y.cancel(home,*this,pc); (void) Propagator::dispose(home); return sizeof(*this); } /* * Reified linear propagators * */ template forceinline ReLin::ReLin (Home home, ViewArray

& x, ViewArray& y, Val c, Ctrl b0) : Lin(home,x,y,c), b(b0) { b.subscribe(home,*this,PC_INT_VAL); } template forceinline ReLin::ReLin (Space& home, ReLin& p) : Lin(home,p) { b.update(home,p.b); } template void ReLin::reschedule(Space& home) { x.reschedule(home,*this,pc); y.reschedule(home,*this,pc); b.reschedule(home,*this,PC_INT_VAL); } template forceinline size_t ReLin::dispose(Space& home) { b.cancel(home,*this,PC_BOOL_VAL); (void) Lin::dispose(home); return sizeof(*this); } /* * Computing bounds * */ template void bounds_p(ModEventDelta med, ViewArray& x, Val& c, Val& sl, Val& su) { int n = x.size(); if (IntView::me(med) == ME_INT_VAL) { for (int i=n; i--; ) { Val m = x[i].min(); if (x[i].assigned()) { c -= m; x[i] = x[--n]; } else { sl -= m; su -= x[i].max(); } } x.size(n); } else { for (int i=0; i void bounds_n(ModEventDelta med, ViewArray& y, Val& c, Val& sl, Val& su) { int n = y.size(); if (IntView::me(med) == ME_INT_VAL) { for (int i=n; i--; ) { Val m = y[i].max(); if (y[i].assigned()) { c += m; y[i] = y[--n]; } else { sl += m; su += y[i].min(); } } y.size(n); } else { for (int i=0; i ExecStatus prop_bnd(Space& home, ModEventDelta med, Propagator& p, ViewArray

& x, ViewArray& y, Val& c) { // Eliminate singletons Val sl = 0; Val su = 0; bounds_p(med, x, c, sl, su); bounds_n(med, y, c, sl, su); if ((IntView::me(med) == ME_INT_VAL) && ((x.size() + y.size()) <= 1)) { if (x.size() == 1) { GECODE_ME_CHECK(x[0].eq(home,c)); return home.ES_SUBSUMED(p); } if (y.size() == 1) { GECODE_ME_CHECK(y[0].eq(home,-c)); return home.ES_SUBSUMED(p); } return (c == static_cast(0)) ? home.ES_SUBSUMED(p) : ES_FAILED; } sl += c; su += c; const int mod_sl = 1; const int mod_su = 2; int mod = mod_sl | mod_su; do { if (mod & mod_sl) { mod -= mod_sl; // Propagate upper bound for positive variables for (int i=0; i forceinline Eq::Eq(Home home, ViewArray

& x, ViewArray& y, Val c) : Lin(home,x,y,c) {} template ExecStatus Eq::post(Home home, ViewArray

& x, ViewArray& y, Val c) { ViewArray nva; if (y.size() == 0) { (void) new (home) Eq(home,x,nva,c); } else if (x.size() == 0) { (void) new (home) Eq(home,y,nva,-c); } else { (void) new (home) Eq(home,x,y,c); } return ES_OK; } template forceinline Eq::Eq(Space& home, Eq& p) : Lin(home,p) {} /** * \brief Rewriting of equality to binary propagators * */ template forceinline Actor* eqtobin(Space&, Propagator&, ViewArray

&, ViewArray&, Val) { return nullptr; } template forceinline Actor* eqtobin(Space& home, Propagator& p, ViewArray& x, ViewArray&, Val c) { assert(x.size() == 2); return new (home) EqBin (home,p,x[0],x[1],c); } template forceinline Actor* eqtobin(Space& home, Propagator& p, ViewArray&, ViewArray& y, Val c) { assert(y.size() == 2); return new (home) EqBin (home,p,y[0],y[1],-c); } template forceinline Actor* eqtobin(Space& home, Propagator& p, ViewArray& x, ViewArray& y, Val c) { if (x.size() == 2) return new (home) EqBin (home,p,x[0],x[1],c); if (x.size() == 1) return new (home) EqBin (home,p,x[0],MinusView(y[0]),c); return new (home) EqBin (home,p,y[0],y[1],-c); } /** * \brief Rewriting of equality to ternary propagators * */ template forceinline Actor* eqtoter(Space&, Propagator&, ViewArray

&, ViewArray&, Val) { return nullptr; } template forceinline Actor* eqtoter(Space& home, Propagator& p, ViewArray& x, ViewArray&, Val c) { assert(x.size() == 3); return new (home) EqTer (home,p,x[0],x[1],x[2],c); } template forceinline Actor* eqtoter(Space& home, Propagator& p, ViewArray&, ViewArray& y, Val c) { assert(y.size() == 3); return new (home) EqTer (home,p,y[0],y[1],y[2],-c); } template forceinline Actor* eqtoter(Space& home, Propagator& p, ViewArray& x, ViewArray& y, Val c) { if (x.size() == 3) return new (home) EqTer (home,p,x[0],x[1],x[2],c); if (x.size() == 2) return new (home) EqTer (home,p,x[0],x[1],MinusView(y[0]),c); if (x.size() == 1) return new (home) EqTer (home,p,y[0],y[1],MinusView(x[0]),-c); return new (home) EqTer (home,p,y[0],y[1],y[2],-c); } template Actor* Eq::copy(Space& home) { if (isunit(x,y)) { // Check whether rewriting is possible if (x.size() + y.size() == 2) return eqtobin(home,*this,x,y,c); if (x.size() + y.size() == 3) return eqtoter(home,*this,x,y,c); } return new (home) Eq(home,*this); } template ExecStatus Eq::propagate(Space& home, const ModEventDelta& med) { return prop_bnd(home,med,*this,x,y,c); } /* * Reified bound consistent linear equation * */ template forceinline ReEq::ReEq(Home home, ViewArray

& x, ViewArray& y, Val c, Ctrl b) : ReLin(home,x,y,c,b) {} template ExecStatus ReEq::post(Home home, ViewArray

& x, ViewArray& y, Val c, Ctrl b) { ViewArray nva; if (y.size() == 0) { (void) new (home) ReEq(home,x,nva,c,b); } else if (x.size() == 0) { (void) new (home) ReEq(home,y,nva,-c,b); } else { (void) new (home) ReEq(home,x,y,c,b); } return ES_OK; } template forceinline ReEq::ReEq(Space& home, ReEq& p) : ReLin(home,p) {} template Actor* ReEq::copy(Space& home) { return new (home) ReEq(home,*this); } template ExecStatus ReEq::propagate(Space& home, const ModEventDelta& med) { if (b.zero()) { if (rm == RM_IMP) return home.ES_SUBSUMED(*this); GECODE_REWRITE(*this,(Nq::post(home(*this),x,y,c))); } if (b.one()) { if (rm == RM_PMI) return home.ES_SUBSUMED(*this); GECODE_REWRITE(*this,(Eq::post(home(*this),x,y,c))); } Val sl = 0; Val su = 0; bounds_p(med, x, c, sl, su); bounds_n(med, y, c, sl, su); if ((-sl == c) && (-su == c)) { if (rm != RM_IMP) GECODE_ME_CHECK(b.one_none(home)); return home.ES_SUBSUMED(*this); } if ((-sl > c) || (-su < c)) { if (rm != RM_PMI) GECODE_ME_CHECK(b.zero_none(home)); return home.ES_SUBSUMED(*this); } return ES_FIX; } /* * Domain consistent linear disequation * */ template forceinline Nq::Nq(Home home, ViewArray

& x, ViewArray& y, Val c) : Lin(home,x,y,c) {} template ExecStatus Nq::post(Home home, ViewArray

& x, ViewArray& y, Val c) { ViewArray nva; if (y.size() == 0) { (void) new (home) Nq(home,x,nva,c); } else if (x.size() == 0) { (void) new (home) Nq(home,y,nva,-c); } else { (void) new (home) Nq(home,x,y,c); } return ES_OK; } template forceinline Nq::Nq(Space& home, Nq& p) : Lin(home,p) {} /** * \brief Rewriting of disequality to binary propagators * */ template forceinline Actor* nqtobin(Space&, Propagator&, ViewArray

&, ViewArray&, Val) { return nullptr; } template forceinline Actor* nqtobin(Space& home, Propagator& p, ViewArray& x, ViewArray&, Val c) { assert(x.size() == 2); return new (home) NqBin (home,p,x[0],x[1],c); } template forceinline Actor* nqtobin(Space& home, Propagator& p, ViewArray&, ViewArray& y, Val c) { assert(y.size() == 2); return new (home) NqBin (home,p,y[0],y[1],-c); } template forceinline Actor* nqtobin(Space& home, Propagator& p, ViewArray& x, ViewArray& y, Val c) { if (x.size() == 2) return new (home) NqBin (home,p,x[0],x[1],c); if (x.size() == 1) return new (home) NqBin (home,p,x[0],MinusView(y[0]),c); return new (home) NqBin (home,p,y[0],y[1],-c); } /** * \brief Rewriting of disequality to ternary propagators * */ template forceinline Actor* nqtoter(Space&, Propagator&,ViewArray

&, ViewArray&, Val) { return nullptr; } template forceinline Actor* nqtoter(Space& home, Propagator& p, ViewArray& x, ViewArray&, Val c) { assert(x.size() == 3); return new (home) NqTer (home,p,x[0],x[1],x[2],c); } template forceinline Actor* nqtoter(Space& home, Propagator& p, ViewArray&, ViewArray& y, Val c) { assert(y.size() == 3); return new (home) NqTer (home,p,y[0],y[1],y[2],-c); } template forceinline Actor* nqtoter(Space& home, Propagator& p, ViewArray& x, ViewArray& y, Val c) { if (x.size() == 3) return new (home) NqTer (home,p,x[0],x[1],x[2],c); if (x.size() == 2) return new (home) NqTer (home,p,x[0],x[1],MinusView(y[0]),c); if (x.size() == 1) return new (home) NqTer (home,p,y[0],y[1],MinusView(x[0]),-c); return new (home) NqTer (home,p,y[0],y[1],y[2],-c); } template Actor* Nq::copy(Space& home) { if (isunit(x,y)) { // Check whether rewriting is possible if (x.size() + y.size() == 2) return nqtobin(home,*this,x,y,c); if (x.size() + y.size() == 3) return nqtoter(home,*this,x,y,c); } return new (home) Nq(home,*this); } template ExecStatus Nq::propagate(Space& home, const ModEventDelta&) { for (int i=x.size(); i--; ) if (x[i].assigned()) { c -= x[i].val(); x.move_lst(i); } for (int i=y.size(); i--; ) if (y[i].assigned()) { c += y[i].val(); y.move_lst(i); } if (x.size() + y.size() <= 1) { if (x.size() == 1) { GECODE_ME_CHECK(x[0].nq(home,c)); return home.ES_SUBSUMED(*this); } if (y.size() == 1) { GECODE_ME_CHECK(y[0].nq(home,-c)); return home.ES_SUBSUMED(*this); } return (c == static_cast(0)) ? ES_FAILED : home.ES_SUBSUMED(*this); } return ES_FIX; } /* * Bound consistent linear inequation * */ template forceinline Lq::Lq(Home home, ViewArray

& x, ViewArray& y, Val c) : Lin(home,x,y,c) {} template ExecStatus Lq::post(Home home, ViewArray

& x, ViewArray& y, Val c) { ViewArray nva; if (y.size() == 0) { (void) new (home) Lq(home,x,nva,c); } else if (x.size() == 0) { (void) new (home) Lq(home,nva,y,c); } else { (void) new (home) Lq(home,x,y,c); } return ES_OK; } template forceinline Lq::Lq(Space& home, Lq& p) : Lin(home,p) {} /** * \brief Rewriting of inequality to binary propagators * */ template forceinline Actor* lqtobin(Space&, Propagator&,ViewArray

&, ViewArray&, Val) { return nullptr; } template forceinline Actor* lqtobin(Space& home, Propagator& p, ViewArray& x, ViewArray&, Val c) { assert(x.size() == 2); return new (home) LqBin (home,p,x[0],x[1],c); } template forceinline Actor* lqtobin(Space& home, Propagator& p, ViewArray&, ViewArray& y, Val c) { assert(y.size() == 2); return new (home) LqBin (home,p,MinusView(y[0]),MinusView(y[1]),c); } template forceinline Actor* lqtobin(Space& home, Propagator& p, ViewArray& x, ViewArray& y, Val c) { if (x.size() == 2) return new (home) LqBin (home,p,x[0],x[1],c); if (x.size() == 1) return new (home) LqBin (home,p,x[0],MinusView(y[0]),c); return new (home) LqBin (home,p,MinusView(y[0]),MinusView(y[1]),c); } /** * \brief Rewriting of inequality to ternary propagators * */ template forceinline Actor* lqtoter(Space&, Propagator&, ViewArray

&, ViewArray&, Val) { return nullptr; } template forceinline Actor* lqtoter(Space& home, Propagator& p, ViewArray& x, ViewArray&, Val c) { assert(x.size() == 3); return new (home) LqTer (home,p,x[0],x[1],x[2],c); } template forceinline Actor* lqtoter(Space& home, Propagator& p, ViewArray&, ViewArray& y, Val c) { assert(y.size() == 3); return new (home) LqTer (home,p,MinusView(y[0]),MinusView(y[1]),MinusView(y[2]),c); } template forceinline Actor* lqtoter(Space& home, Propagator& p, ViewArray& x, ViewArray& y, Val c) { if (x.size() == 3) return new (home) LqTer (home,p,x[0],x[1],x[2],c); if (x.size() == 2) return new (home) LqTer (home,p,x[0],x[1],MinusView(y[0]),c); if (x.size() == 1) return new (home) LqTer (home,p,x[0],MinusView(y[0]),MinusView(y[1]),c); return new (home) LqTer (home,p,MinusView(y[0]),MinusView(y[1]),MinusView(y[2]),c); } template Actor* Lq::copy(Space& home) { if (isunit(x,y)) { // Check whether rewriting is possible if (x.size() + y.size() == 2) return lqtobin(home,*this,x,y,c); if (x.size() + y.size() == 3) return lqtoter(home,*this,x,y,c); } return new (home) Lq(home,*this); } template ExecStatus Lq::propagate(Space& home, const ModEventDelta& med) { // Eliminate singletons Val sl = 0; if (IntView::me(med) == ME_INT_VAL) { for (int i=x.size(); i--; ) { Val m = x[i].min(); if (x[i].assigned()) { c -= m; x.move_lst(i); } else { sl -= m; } } for (int i=y.size(); i--; ) { Val m = y[i].max(); if (y[i].assigned()) { c += m; y.move_lst(i); } else { sl += m; } } if ((x.size() + y.size()) <= 1) { if (x.size() == 1) { GECODE_ME_CHECK(x[0].lq(home,c)); return home.ES_SUBSUMED(*this); } if (y.size() == 1) { GECODE_ME_CHECK(y[0].gq(home,-c)); return home.ES_SUBSUMED(*this); } return (c >= static_cast(0)) ? home.ES_SUBSUMED(*this) : ES_FAILED; } } else { for (int i=0; i forceinline ReLq::ReLq(Home home, ViewArray

& x, ViewArray& y, Val c, BoolView b) : ReLin(home,x,y,c,b) {} template ExecStatus ReLq::post(Home home, ViewArray

& x, ViewArray& y, Val c, BoolView b) { ViewArray nva; if (y.size() == 0) { (void) new (home) ReLq(home,x,nva,c,b); } else if (x.size() == 0) { (void) new (home) ReLq(home,nva,y,c,b); } else { (void) new (home) ReLq(home,x,y,c,b); } return ES_OK; } template forceinline ReLq::ReLq(Space& home, ReLq& p) : ReLin(home,p) {} template Actor* ReLq::copy(Space& home) { return new (home) ReLq(home,*this); } template ExecStatus ReLq::propagate(Space& home, const ModEventDelta& med) { if (b.zero()) { if (rm == RM_IMP) return home.ES_SUBSUMED(*this); GECODE_REWRITE(*this,(Lq::post(home(*this),y,x,-c-1))); } if (b.one()) { if (rm == RM_PMI) return home.ES_SUBSUMED(*this); GECODE_REWRITE(*this,(Lq::post(home(*this),x,y,c))); } // Eliminate singletons Val sl = 0; Val su = 0; bounds_p(med,x,c,sl,su); bounds_n(med,y,c,sl,su); if (-sl > c) { if (rm != RM_PMI) GECODE_ME_CHECK(b.zero_none(home)); return home.ES_SUBSUMED(*this); } if (-su <= c) { if (rm != RM_IMP) GECODE_ME_CHECK(b.one_none(home)); return home.ES_SUBSUMED(*this); } return ES_FIX; } }}} // STATISTICS: int-prop