/* -*- mode: C++; c-basic-offset: 2; indent-tabs-mode: nil -*- */ /* * Main authors: * Guido Tack * Christian Schulte * * Copyright: * Guido Tack, 2004,2006 * Christian Schulte, 2004 * * 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. * */ namespace Gecode { namespace Set { namespace Element { template forceinline ElementUnion:: ElementUnion(Home home, IdxViewArray& iv0, View0 y0, View1 y1) : Propagator(home), iv(iv0), x0(y0), x1(y1) { home.notice(*this,AP_DISPOSE); x0.subscribe(home,*this, PC_SET_ANY); x1.subscribe(home,*this, PC_SET_ANY); iv.subscribe(home,*this, PC_SET_ANY); } template forceinline ElementUnion:: ElementUnion(Space& home, ElementUnion& p) : Propagator(home,p) { x0.update(home,p.x0); x1.update(home,p.x1); iv.update(home,p.iv); } template PropCost ElementUnion::cost(const Space&, const ModEventDelta&) const { return PropCost::linear(PropCost::HI, iv.size()+2); } template void ElementUnion::reschedule(Space& home) { x0.reschedule(home,*this, PC_SET_ANY); x1.reschedule(home,*this, PC_SET_ANY); iv.reschedule(home,*this, PC_SET_ANY); } template forceinline size_t ElementUnion::dispose(Space& home) { home.ignore(*this,AP_DISPOSE); if (!home.failed()) { x0.cancel(home,*this,PC_SET_ANY); x1.cancel(home,*this,PC_SET_ANY); iv.cancel(home,*this,PC_SET_ANY); } (void) Propagator::dispose(home); return sizeof(*this); } template ExecStatus ElementUnion:: post(Home home, IdxViewArray& xs, View0 x0, View1 x1) { int n = xs.size(); // x0 \subseteq {1,...,n} Iter::Ranges::Singleton s(0, n-1); GECODE_ME_CHECK(x0.intersectI(home,s)); (void) new (home) ElementUnion(home,xs,x0,x1); return ES_OK; } template Actor* ElementUnion::copy(Space& home) { return new (home) ElementUnion(home,*this); } template ExecStatus ElementUnion::propagate(Space& home, const ModEventDelta&) { Region r; int n = iv.size(); bool loopVar; do { loopVar = false; // Cache the upper bound iterator, as we have to // modify the upper bound while iterating LubRanges x0ub(x0); Iter::Ranges::Cache x0ubc(r,x0ub); Iter::Ranges::ToValues vx0ub(x0ubc); GlbRanges x0lb(x0); Iter::Ranges::Cache x0lbc(r,x0lb); Iter::Ranges::ToValues vx0(x0lbc); // In the first iteration, compute in before[i] the union // of all the upper bounds of the x_i. At the same time, // exclude inconsistent x_i from x0 and remove them from // the list, cancel their dependencies. GLBndSet sofarBefore(home); LUBndSet selectedInter(home, IntSet (Limits::min, Limits::max)); GLBndSet* before = static_cast(r.ralloc(sizeof(GLBndSet)*n)); int j = 0; int i = 0; unsigned int maxCard = 0; unsigned int minCard = Limits::card; while ( vx0ub() ) { // Remove vars at indices not in the upper bound if (iv[i].idx < vx0ub.val()) { iv[i].view.cancel(home,*this, PC_SET_ANY); ++i; continue; } assert(iv[i].idx == vx0ub.val()); iv[j] = iv[i]; View candidate = iv[j].view; int candidateInd = iv[j].idx; // inter = glb(candidate) & complement(lub(x1)) GlbRanges candlb(candidate); LubRanges x1ub(x1); Iter::Ranges::Diff, LubRanges > diff(candlb, x1ub); bool selectSingleInconsistent = false; if (x0.cardMax() <= 1) { GlbRanges x1lb(x1); LubRanges candub(candidate); Iter::Ranges::Diff, LubRanges > diff2(x1lb, candub); selectSingleInconsistent = diff2() || candidate.cardMax() < x1.cardMin(); } // exclude inconsistent x_i // an x_i is inconsistent if // * at most one x_i can be selected and there are // elements in x_0 that can't be in x_i // (selectSingleInconsistent) // * its min cardinality is greater than maxCard of x1 // * inter is not empty (there are elements in x_i // that can't be in x_0) if (selectSingleInconsistent || candidate.cardMin() > x1.cardMax() || diff()) { ModEvent me = (x0.exclude(home,candidateInd)); loopVar |= me_modified(me); GECODE_ME_CHECK(me); iv[j].view.cancel(home,*this, PC_SET_ANY); ++i; ++vx0ub; continue; } else { // if x_i is consistent, check whether we know // that its index is in x0 if (vx0() && vx0.val()==candidateInd) { // x1 >= candidate, candidate <= x1 GlbRanges candlb(candidate); ModEvent me = x1.includeI(home,candlb); loopVar |= me_modified(me); GECODE_ME_CHECK(me); LubRanges x1ub(x1); me = candidate.intersectI(home,x1ub); loopVar |= me_modified(me); GECODE_ME_CHECK(me); ++vx0; } new (&before[j]) GLBndSet(home); before[j].update(home,sofarBefore); LubRanges cub(candidate); sofarBefore.includeI(home,cub); GlbRanges clb(candidate); selectedInter.intersectI(home,clb); maxCard = std::max(maxCard, candidate.cardMax()); minCard = std::min(minCard, candidate.cardMin()); } ++vx0ub; ++i; ++j; } // cancel the variables with index greater than // max of lub(x0) for (int k=i; k 0) { // Selector is not empty, hence the intersection of the // possibly selected lower bounds is contained in x1 BndSetRanges si(selectedInter); ModEvent me = x1.includeI(home, si); loopVar |= me_modified(me); GECODE_ME_CHECK(me); me = x1.cardMin(home, minCard); loopVar |= me_modified(me); GECODE_ME_CHECK(me); } selectedInter.dispose(home); if (x0.cardMax() <= 1) { ModEvent me = x1.cardMax(home, maxCard); loopVar |= me_modified(me); GECODE_ME_CHECK(me); } { // x1 <= sofarBefore BndSetRanges sfB(sofarBefore); ModEvent me = x1.intersectI(home,sfB); loopVar |= me_modified(me); GECODE_ME_CHECK(me); } sofarBefore.dispose(home); GLBndSet sofarAfter(home); // In the second iteration, this time backwards, compute // sofarAfter as the union of all lub(x_j) with j>i for (int i=n; i--;) { // TODO: check for size of universe here? // if (sofarAfter.size() == 0) break; // extra = inter(before[i], sofarAfter) - lub(x1) BndSetRanges b(before[i]); BndSetRanges s(sofarAfter); GlbRanges x1lb(x1); Iter::Ranges::Union inter(b,s); Iter::Ranges::Diff, Iter::Ranges::Union > diff(x1lb, inter); if (diff()) { ModEvent me = (x0.include(home,iv[i].idx)); loopVar |= me_modified(me); GECODE_ME_CHECK(me); // candidate != extra me = iv[i].view.includeI(home,diff); loopVar |= me_modified(me); GECODE_ME_CHECK(me); } LubRanges iviub(iv[i].view); sofarAfter.includeI(home,iviub); before[i].dispose(home); } sofarAfter.dispose(home); } while (loopVar); // Test whether we determined x0 without determining x1 if (x0.assigned() && !x1.assigned()) { int ubsize = static_cast(x0.lubSize()); if (ubsize > 2) { assert(ubsize==n); ViewArray is(home,ubsize); for (int i=n; i--;) is[i]=iv[i].view; GECODE_REWRITE(*this,(RelOp::UnionN ::post(home(*this),is,x1))); } else if (ubsize == 2) { assert(n==2); View a = iv[0].view; View b = iv[1].view; GECODE_REWRITE(*this,(RelOp::Union ::post(home(*this),a,b,x1))); } else if (ubsize == 1) { assert(n==1); GECODE_REWRITE(*this, (Rel::Eq::post(home(*this),x1,iv[0].view))); } else { GECODE_ME_CHECK(x1.cardMax(home, 0)); return home.ES_SUBSUMED(*this); } } for (int i=iv.size(); i--;) if (!iv[i].view.assigned()) return ES_FIX; if (!x1.assigned() || !x0.assigned()) return ES_FIX; return home.ES_SUBSUMED(*this); } }}} // STATISTICS: set-prop