/* -*- mode: C++; c-basic-offset: 2; indent-tabs-mode: nil -*- */ /* * Main authors: * Christian Schulte * * Contributing authors: * Mikael Lagerkvist * * Copyright: * Christian Schulte, 2006 * Mikael Lagerkvist, 2006 * * Last modified: * $Date: 2008-01-31 18:29:16 +0100 (Thu, 31 Jan 2008) $ by $Author: tack $ * $Revision: 6017 $ * * 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 Int { namespace Channel { /** * \brief Combine view with information for domain propagation * */ template class DomInfo { public: View view; unsigned int card; int min; int max; /// Allocate memory from space \a home for \a n elements static DomInfo* allocate(Space* home, int n); /// Initialize void init(View x, int n); /// Update during cloning void update(Space* home, bool share, DomInfo& vcb); /// Check whether propagation for assignment is to be done bool doval(void) const; /// Check whether propagation for domain is to be done bool dodom(void) const; /// Record that view got assigned void assigned(void); /// Record that one value got removed void removed(int i); /// Update the cardinality and bounds information after pruning void done(void); }; template forceinline DomInfo* DomInfo::allocate(Space* home, int n) { return static_cast*> (home->alloc(n*sizeof(DomInfo))); } template forceinline void DomInfo::init(View x, int n) { view = x; card = static_cast(n); min = 0; max = n-1; } template forceinline void DomInfo::update(Space* home, bool share, DomInfo& di) { view.update(home,share,di.view); card = di.card; min = di.min; max = di.max; } template forceinline bool DomInfo::doval(void) const { return (card != 1) && view.assigned(); } template forceinline bool DomInfo::dodom(void) const { return card != view.size(); } template forceinline void DomInfo::assigned(void) { card = 1; } template forceinline void DomInfo::removed(int i) { card--; if (i == min) min++; else if (i == max) max--; } template forceinline void DomInfo::done(void) { card = view.size(); min = view.min(); max = view.max(); } // Propagate domain information from x to y template ExecStatus prop_dom(Space* home, int n, DomInfo* x, DomInfo* y, ProcessStack& ya) { for (int i = n; i--; ) // Only views with not yet propagated missing values if (x[i].dodom()) { // Iterate the values in the complement of x[i] ViewRanges xir(x[i].view); Iter::Ranges::ComplVal > xirc(x[i].min,x[i].max,xir); Iter::Ranges::ToValues > > jv(xirc); while (jv()) { // j is not in the domain of x[i], so prune i from y[j] int j = jv.val(); ModEvent me = y[j].view.nq(home,i); if (me_failed(me)) return ES_FAILED; if (me_modified(me)) if (me == ME_INT_VAL) { // Record that y[j] has been assigned and must be propagated ya.push(j); } else { // Obvious as x[i] is different from j y[j].removed(i); } ++jv; } // Update which values have been propagated and what are the new bounds x[i].done(); } return ES_OK; } /* * The actual propagator * */ template forceinline Dom::Dom(Space* home, int n, DomInfo* xy) : Base,PC_INT_DOM>(home,n,xy) {} template forceinline Dom::Dom(Space* home, bool share, Dom& p) : Base,PC_INT_DOM>(home,share,p) {} template Actor* Dom::copy(Space* home, bool share) { return new (home) Dom(home,share,*this); } template PropCost Dom::cost(ModEventDelta med) const { return (View::me(med) == ME_INT_VAL) ? PC_QUADRATIC_LO : PC_CUBIC_HI; } template Support::Symbol Dom::ati(void) { return Reflection::mangle("Gecode::Int::Channel::Dom"); } template Reflection::ActorSpec Dom::spec(const Space* home, Reflection::VarMap& m) const { return Base,PC_INT_DOM>::spec(home, m, ati()); } template void Dom::post(Space* home, Reflection::VarMap& vars, const Reflection::ActorSpec& spec) { const int n = spec.noOfArgs(); DomInfo* vi = DomInfo::allocate(home,n); for (int i=0; i(home,n/2,vi); } template ExecStatus Dom::propagate(Space* home, ModEventDelta med) { GECODE_AUTOSTACK(int,-1, xa, n); GECODE_AUTOSTACK(int,-1, ya, n); DomInfo* x = xy; DomInfo* y = xy+n; if (View::me(med) == ME_INT_VAL) { // Scan x and y for assigned but not yet propagated views for (int i = n; i--; ) { if (x[i].doval()) xa.push(i); if (y[i].doval()) ya.push(i); } if (shared) { do { // Propagate assigned views for x GECODE_ES_CHECK((prop_val > (home,n,x,y,n_na,xa,ya))); // Propagate assigned views for y GECODE_ES_CHECK((prop_val > (home,n,y,x,n_na,ya,xa))); assert(ya.empty()); } while (!xa.empty() || !ya.empty()); return ES_NOFIX_PARTIAL(this,View::med(ME_INT_DOM)); } else { do { // Propagate assigned views for x GECODE_ES_CHECK((prop_val > (home,n,x,y,n_na,xa,ya))); // Propagate assigned views for y GECODE_ES_CHECK((prop_val > (home,n,y,x,n_na,ya,xa))); assert(ya.empty()); } while (!xa.empty()); return ES_FIX_PARTIAL(this,View::med(ME_INT_DOM)); } } // Process changed views for y // This propagates from y to x and possibly records xs that // got assigned GECODE_ES_CHECK(prop_dom(home,n,y,x,xa)); // Process assigned views for x GECODE_ES_CHECK((prop_val >(home,n,x,y,n_na,xa,ya))); // Perform domain consistent propagation for distinct on the x views if (dc.available()) { GECODE_ES_CHECK(dc.sync()); } else { GECODE_AUTOARRAY(View,xv,n); for (int i=n; i--; ) xv[i] = x[i].view; GECODE_ES_CHECK(dc.init(home,n,&xv[0])); } bool assigned; GECODE_ES_CHECK(dc.propagate(home,assigned)); if (assigned) { // This has assigned some more views in x which goes unnoticed // (that is, not recorded in xa). This must be checked and propagated // to the y views, however the distinctness on x is already // propagated. for (int i=n; i--; ) if (x[i].doval()) { int j = x[i].view.val(); // Assign the y variable to i (or test if already assigned!) ModEvent me = y[j].view.eq(home,i); if (me_failed(me)) return ES_FAILED; if (me_modified(me)) { // Record that y[j] has been assigned and must be propagated assert(me == ME_INT_VAL); // Otherwise the modification event would not be ME_INT_VAL ya.push(j); } x[i].assigned(); n_na--; } } // Process changed views for x // This propagates from x to y and possibly records ys that // got assigned GECODE_ES_CHECK(prop_dom(home,n,x,y,ya)); // Process assigned view again (some might have been found above) while (!xa.empty() || !ya.empty()) { // Process assigned views for x GECODE_ES_CHECK((prop_val >(home,n,x,y,n_na,xa,ya))); // Process assigned views for y GECODE_ES_CHECK((prop_val >(home,n,y,x,n_na,ya,xa))); }; if (shared) { for (int i=2*n; i--; ) if (!xy[i].view.assigned()) return ES_NOFIX; return ES_SUBSUMED(this,home); } else { return (n_na == 0) ? ES_SUBSUMED(this,home) : ES_FIX; } } template ExecStatus Dom::post(Space* home, int n, DomInfo* xy) { assert(n > 0); if (n == 1) { GECODE_ME_CHECK(xy[0].view.eq(home,0)); GECODE_ME_CHECK(xy[1].view.eq(home,0)); return ES_OK; } for (int i=2*n; i--; ) { GECODE_ME_CHECK(xy[i].view.gq(home,0)); GECODE_ME_CHECK(xy[i].view.le(home,n)); } (void) new (home) Dom(home,n,xy); return ES_OK; } }}} // STATISTICS: int-prop