/* -*- mode: C++; c-basic-offset: 2; indent-tabs-mode: nil -*- */ /* * Main authors: * Christian Schulte * * Copyright: * Christian Schulte, 2011 * * Last modified: * $Date: 2011-08-23 05:43:31 +1000 (Tue, 23 Aug 2011) $ by $Author: schulte $ * $Revision: 12335 $ * * 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 Member { template forceinline ReProp::ReProp(Home home, ValSet& vs, ViewArray& x, View y, BoolView b0) : Prop(home,vs,x,y), b(b0) { b.subscribe(home,*this,PC_BOOL_VAL); } template inline ExecStatus ReProp::post(Home home, ViewArray& x, View y, BoolView b) { if (x.size() == 0) { GECODE_ME_CHECK(b.zero(home)); return ES_OK; } x.unique(home); if (x.size() == 1) return Rel::ReEqDom::post(home,x[0],y,b); if (x.same(home,y)) { GECODE_ME_CHECK(b.one(home)); return ES_OK; } // Eliminate assigned views and store them into the value set ValSet vs; add(home, vs, x); switch (vs.compare(y)) { case Iter::Ranges::CS_SUBSET: GECODE_ME_CHECK(b.one(home)); return ES_OK; case Iter::Ranges::CS_DISJOINT: if (x.size() == 0) { GECODE_ME_CHECK(b.zero(home)); return ES_OK; } break; case Iter::Ranges::CS_NONE: break; default: GECODE_NEVER; } (void) new (home) ReProp(home, vs, x, y, b); return ES_OK; } template forceinline ReProp::ReProp(Space& home, bool share, ReProp& p) : Prop(home, share, p) { b.update(home, share, p.b); } template Propagator* ReProp::copy(Space& home, bool share) { return new (home) ReProp(home, share, *this); } template forceinline size_t ReProp::dispose(Space& home) { b.cancel(home, *this, PC_BOOL_VAL); (void) Prop::dispose(home); return sizeof(*this); } template ExecStatus ReProp::propagate(Space& home, const ModEventDelta& med) { // Add assigned views to value set if (View::me(med) == ME_INT_VAL) add(home,vs,x); if (b.one()) { ValSet vsc(vs); vs.flush(); GECODE_REWRITE(*this,Prop::post(home,vsc,x,y)); } if (b.zero()) { ValSet::Ranges vsr(vs); GECODE_ME_CHECK(y.minus_r(home,vsr,false)); for (int i=x.size(); i--; ) GECODE_ES_CHECK(Rel::Nq::post(home,x[i],y)); return home.ES_SUBSUMED(*this); } // Eliminate views from x eliminate(home); switch (vs.compare(y)) { case Iter::Ranges::CS_SUBSET: GECODE_ME_CHECK(b.one(home)); return home.ES_SUBSUMED(*this); case Iter::Ranges::CS_DISJOINT: if (x.size() == 0) { GECODE_ME_CHECK(b.zero(home)); return home.ES_SUBSUMED(*this); } break; case Iter::Ranges::CS_NONE: break; default: GECODE_NEVER; } // Check whether y is in union of x and value set if (x.size() > 0) { Region r(home); ValSet::Ranges vsr(vs); ViewRanges xsr(x[x.size()-1]); Iter::Ranges::NaryUnion u(r,vsr,xsr); for (int i=x.size()-1; i--; ) { ViewRanges xir(x[i]); u |= xir; } ViewRanges yr(y); if (Iter::Ranges::disjoint(u,yr)) { GECODE_ME_CHECK(b.zero(home)); return home.ES_SUBSUMED(*this); } } return ES_FIX; } }}} // STATISTICS: int-prop