/* -*- mode: C++; c-basic-offset: 2; indent-tabs-mode: nil -*- */ /* * Main authors: * Christian Schulte * Tias Guns * * Copyright: * Christian Schulte, 2002 * Tias Guns, 2009 * * 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 #include namespace Gecode { namespace Int { namespace Linear { /// Eliminate assigned views forceinline void eliminate(Term* t, int &n, long long int& d) { for (int i=n; i--; ) if (t[i].x.one()) { d -= t[i].a; t[i]=t[--n]; } else if (t[i].x.zero()) { t[i]=t[--n]; } Limits::check(d,"Int::linear"); } /// Rewrite non-strict relations forceinline void rewrite(IntRelType &r, long long int &d) { switch (r) { case IRT_EQ: case IRT_NQ: case IRT_LQ: case IRT_GQ: break; case IRT_LE: d--; r = IRT_LQ; break; case IRT_GR: d++; r = IRT_GQ; break; default: throw UnknownRelation("Int::linear"); } } forceinline void post_pos_unit(Home home, Term* t_p, int n_p, IntRelType irt, IntView y, int c) { switch (irt) { case IRT_EQ: { ViewArray x(home,n_p); for (int i=0; i ::post(home,x,y,c))); } break; case IRT_NQ: { ViewArray x(home,n_p); for (int i=0; i ::post(home,x,y,c))); } break; case IRT_GQ: { ViewArray x(home,n_p); for (int i=0; i ::post(home,x,y,c))); } break; case IRT_LQ: { ViewArray x(home,n_p); for (int i=0; i ::post(home,x,z,n_p-c))); } break; default: GECODE_NEVER; } } forceinline void post_pos_unit(Home home, Term* t_p, int n_p, IntRelType irt, ZeroIntView, int c) { switch (irt) { case IRT_EQ: { ViewArray x(home,n_p); for (int i=0; i::post(home,x,c))); } break; case IRT_NQ: { ViewArray x(home,n_p); for (int i=0; i::post(home,x,c))); } break; case IRT_GQ: { ViewArray x(home,n_p); for (int i=0; i::post(home,x,c))); } break; case IRT_LQ: { ViewArray x(home,n_p); for (int i=0; i::post(home,x,n_p-c))); } break; default: GECODE_NEVER; } } forceinline void post_pos_unit(Home home, Term* t_p, int n_p, IntRelType irt, int c, Reify r, IntPropLevel) { switch (irt) { case IRT_EQ: { ViewArray x(home,n_p); for (int i=0; i:: post(home,x,c,r.var()))); break; case RM_IMP: GECODE_ES_FAIL((ReEqBoolInt:: post(home,x,c,r.var()))); break; case RM_PMI: GECODE_ES_FAIL((ReEqBoolInt:: post(home,x,c,r.var()))); break; default: GECODE_NEVER; } } break; case IRT_NQ: { ViewArray x(home,n_p); for (int i=0; i:: post(home,x,c,nb))); break; case RM_IMP: GECODE_ES_FAIL((ReEqBoolInt:: post(home,x,c,nb))); break; case RM_PMI: GECODE_ES_FAIL((ReEqBoolInt:: post(home,x,c,nb))); break; default: GECODE_NEVER; } } break; case IRT_GQ: { ViewArray x(home,n_p); for (int i=0; i:: post(home,x,c,r.var()))); break; case RM_IMP: GECODE_ES_FAIL((ReGqBoolInt:: post(home,x,c,r.var()))); break; case RM_PMI: GECODE_ES_FAIL((ReGqBoolInt:: post(home,x,c,r.var()))); break; default: GECODE_NEVER; } } break; case IRT_LQ: { ViewArray x(home,n_p); for (int i=0; i:: post(home,x,n_p-c,r.var()))); break; case RM_IMP: GECODE_ES_FAIL((ReGqBoolInt:: post(home,x,n_p-c,r.var()))); break; case RM_PMI: GECODE_ES_FAIL((ReGqBoolInt:: post(home,x,n_p-c,r.var()))); break; default: GECODE_NEVER; } } break; default: GECODE_NEVER; } } forceinline void post_neg_unit(Home home, Term* t_n, int n_n, IntRelType irt, IntView y, int c) { switch (irt) { case IRT_EQ: { ViewArray x(home,n_n); for (int i=0; i ::post(home,x,z,-c))); } break; case IRT_NQ: { ViewArray x(home,n_n); for (int i=0; i ::post(home,x,z,-c))); } break; case IRT_GQ: { ViewArray x(home,n_n); for (int i=0; i ::post(home,x,y,n_n+c))); } break; case IRT_LQ: { ViewArray x(home,n_n); for (int i=0; i ::post(home,x,z,-c))); } break; default: GECODE_NEVER; } } forceinline void post_neg_unit(Home home, Term* t_n, int n_n, IntRelType irt, ZeroIntView, int c) { switch (irt) { case IRT_EQ: { ViewArray x(home,n_n); for (int i=0; i::post(home,x,-c))); } break; case IRT_NQ: { ViewArray x(home,n_n); for (int i=0; i::post(home,x,-c))); } break; case IRT_GQ: { ViewArray x(home,n_n); for (int i=0; i::post(home,x,n_n+c))); } break; case IRT_LQ: { ViewArray x(home,n_n); for (int i=0; i::post(home,x,-c))); } break; default: GECODE_NEVER; } } forceinline void post_neg_unit(Home home, Term* t_n, int n_n, IntRelType irt, int c, Reify r, IntPropLevel) { switch (irt) { case IRT_EQ: { ViewArray x(home,n_n); for (int i=0; i:: post(home,x,-c,r.var()))); break; case RM_IMP: GECODE_ES_FAIL((ReEqBoolInt:: post(home,x,-c,r.var()))); break; case RM_PMI: GECODE_ES_FAIL((ReEqBoolInt:: post(home,x,-c,r.var()))); break; default: GECODE_NEVER; } } break; case IRT_NQ: { ViewArray x(home,n_n); for (int i=0; i:: post(home,x,-c,nb))); break; case RM_IMP: GECODE_ES_FAIL((ReEqBoolInt:: post(home,x,-c,nb))); break; case RM_PMI: GECODE_ES_FAIL((ReEqBoolInt:: post(home,x,-c,nb))); break; default: GECODE_NEVER; } } break; case IRT_GQ: { ViewArray x(home,n_n); for (int i=0; i:: post(home,x,n_n+c,r.var()))); break; case RM_IMP: GECODE_ES_FAIL((ReGqBoolInt:: post(home,x,n_n+c,r.var()))); break; case RM_PMI: GECODE_ES_FAIL((ReGqBoolInt:: post(home,x,n_n+c,r.var()))); break; default: GECODE_NEVER; } } break; case IRT_LQ: { ViewArray x(home,n_n); for (int i=0; i:: post(home,x,-c,r.var()))); break; case RM_IMP: GECODE_ES_FAIL((ReGqBoolInt:: post(home,x,-c,r.var()))); break; case RM_PMI: GECODE_ES_FAIL((ReGqBoolInt:: post(home,x,-c,r.var()))); break; default: GECODE_NEVER; } } break; default: GECODE_NEVER; } } forceinline void post_mixed(Home home, Term* t_p, int n_p, Term* t_n, int n_n, IntRelType irt, IntView y, int c) { ScaleBoolArray b_p(home,n_p); { ScaleBool* f=b_p.fst(); for (int i=0; i ::post(home,b_p,b_n,y,c))); break; case IRT_NQ: GECODE_ES_FAIL((NqBoolScale ::post(home,b_p,b_n,y,c))); break; case IRT_LQ: GECODE_ES_FAIL((LqBoolScale ::post(home,b_p,b_n,y,c))); break; case IRT_GQ: { MinusView m(y); GECODE_ES_FAIL((LqBoolScale ::post(home,b_n,b_p,m,-c))); } break; default: GECODE_NEVER; } } forceinline void post_mixed(Home home, Term* t_p, int n_p, Term* t_n, int n_n, IntRelType irt, ZeroIntView y, int c) { ScaleBoolArray b_p(home,n_p); { ScaleBool* f=b_p.fst(); for (int i=0; i ::post(home,b_p,b_n,y,c))); break; case IRT_NQ: GECODE_ES_FAIL( (NqBoolScale ::post(home,b_p,b_n,y,c))); break; case IRT_LQ: GECODE_ES_FAIL( (LqBoolScale ::post(home,b_p,b_n,y,c))); break; case IRT_GQ: GECODE_ES_FAIL( (LqBoolScale ::post(home,b_n,b_p,y,-c))); break; default: GECODE_NEVER; } } template forceinline void post_all(Home home, Term* t, int n, IntRelType irt, View x, int c) { Limits::check(c,"Int::linear"); long long int d = c; eliminate(t,n,d); Term *t_p, *t_n; int n_p, n_n, gcd=0; bool unit = normalize(t,n,t_p,n_p,t_n,n_n,gcd); rewrite(irt,d); c = static_cast(d); if (n == 0) { switch (irt) { case IRT_EQ: GECODE_ME_FAIL(x.eq(home,-c)); break; case IRT_NQ: GECODE_ME_FAIL(x.nq(home,-c)); break; case IRT_GQ: GECODE_ME_FAIL(x.lq(home,-c)); break; case IRT_LQ: GECODE_ME_FAIL(x.gq(home,-c)); break; default: GECODE_NEVER; } return; } // Check for overflow { long long int sl = static_cast(x.max())+c; long long int su = static_cast(x.min())+c; for (int i=0; i* t, int n, IntRelType irt, IntView x, int c, IntPropLevel) { post_all(home,t,n,irt,x,c); } void post(Home home, Term* t, int n, IntRelType irt, int c, IntPropLevel) { ZeroIntView x; post_all(home,t,n,irt,x,c); } void post(Home home, Term* t, int n, IntRelType irt, IntView x, Reify r, IntPropLevel ipl) { int l, u; estimate(t,n,0,l,u); IntVar z(home,l,u); IntView zv(z); post_all(home,t,n,IRT_EQ,zv,0); rel(home,z,irt,x,r,ipl); } void post(Home home, Term* t, int n, IntRelType irt, int c, Reify r, IntPropLevel ipl) { if (r.var().one()) { if (r.mode() != RM_PMI) post(home,t,n,irt,c,ipl); return; } if (r.var().zero()) { if (r.mode() != RM_IMP) post(home,t,n,neg(irt),c,ipl); return; } Limits::check(c,"Int::linear"); long long int d = c; eliminate(t,n,d); Term *t_p, *t_n; int n_p, n_n, gcd=1; bool unit = normalize(t,n,t_p,n_p,t_n,n_n,gcd); rewrite(irt,d); // Divide by gcd if (gcd > 1) { switch (irt) { case IRT_EQ: if ((d % gcd) != 0) { if (r.mode() != RM_PMI) GECODE_ME_FAIL(BoolView(r.var()).zero(home)); return; } d /= gcd; break; case IRT_NQ: if ((d % gcd) == 0) { if (r.mode() != RM_IMP) GECODE_ME_FAIL(BoolView(r.var()).one(home)); return; } d /= gcd; break; case IRT_LQ: d = floor_div_xp(d,static_cast(gcd)); break; case IRT_GQ: d = ceil_div_xp(d,static_cast(gcd)); break; default: GECODE_NEVER; } } c = static_cast(d); if (n == 0) { bool fail = false; switch (irt) { case IRT_EQ: fail = (0 != c); break; case IRT_NQ: fail = (0 == c); break; case IRT_GQ: fail = (0 < c); break; case IRT_LQ: fail = (0 > c); break; default: GECODE_NEVER; } if (fail) { if (r.mode() != RM_PMI) GECODE_ME_FAIL(BoolView(r.var()).zero(home)); } else { if (r.mode() != RM_IMP) GECODE_ME_FAIL(BoolView(r.var()).one(home)); } return; } // Check for overflow { long long int sl = c; long long int su = c; for (int i=0; i