/* -*- mode: C++; c-basic-offset: 2; indent-tabs-mode: nil -*- */ /* * Main authors: * Christian Schulte * * Copyright: * Christian Schulte, 2009 * * Last modified: * $Date: 2011-05-26 00:56:41 +1000 (Thu, 26 May 2011) $ by $Author: schulte $ * $Revision: 12022 $ * * 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 Unary { /* * Omega tree */ forceinline void OmegaNode::init(const OmegaNode&, const OmegaNode&) { p = 0; ect = -Int::Limits::infinity; } forceinline void OmegaNode::update(const OmegaNode& l, const OmegaNode& r) { p = l.p + r.p; ect = std::max(plus(l.ect,r.p), r.ect); } template OmegaTree::OmegaTree(Region& r, const TaskViewArray& t) : TaskTree(r,t) { for (int i=tasks.size(); i--; ) { leaf(i).p = 0; leaf(i).ect = -Int::Limits::infinity; } init(); } template forceinline void OmegaTree::insert(int i) { leaf(i).p = tasks[i].pmin(); leaf(i).ect = tasks[i].est()+tasks[i].pmin(); update(i); } template forceinline void OmegaTree::remove(int i) { leaf(i).p = 0; leaf(i).ect = -Int::Limits::infinity; update(i); } template forceinline int OmegaTree::ect(void) const { return root().ect; } template forceinline int OmegaTree::ect(int i) const { // Check whether task i is in? OmegaTree& o = const_cast&>(*this); if (o.leaf(i).ect != -Int::Limits::infinity) { o.remove(i); int ect = o.root().ect; o.insert(i); return ect; } else { return root().ect; } } /* * Ome lambda tree */ forceinline void OmegaLambdaNode::init(const OmegaLambdaNode& l, const OmegaLambdaNode& r) { OmegaNode::init(l,r); lp = p; lect = ect; resEct = undef; resLp = undef; } forceinline void OmegaLambdaNode::update(const OmegaLambdaNode& l, const OmegaLambdaNode& r) { OmegaNode::update(l,r); if (l.lp + r.p > l.p + r.lp) { resLp = l.resLp; lp = l.lp + r.p; } else { resLp = r.resLp; lp = l.p + r.lp; } if ((r.lect >= plus(l.ect,r.lp)) && (r.lect >= plus(l.lect,r.p))) { lect = r.lect; resEct = r.resEct; } else if (plus(l.ect,r.lp) >= plus(l.lect,r.p)) { assert(plus(l.ect,r.lp) > r.lect); lect = plus(l.ect,r.lp); resEct = r.resLp; } else { assert((plus(l.lect,r.p) > r.lect) && (plus(l.lect,r.p) > plus(l.ect,r.lp))); lect = plus(l.lect,r.p); resEct = l.resEct; } } template OmegaLambdaTree::OmegaLambdaTree(Region& r, const TaskViewArray& t, bool inc) : TaskTree(r,t) { if (inc) { // Enter all tasks into tree (omega = all tasks, lambda = empty) for (int i=tasks.size(); i--; ) { leaf(i).p = leaf(i).lp = tasks[i].pmin(); leaf(i).ect = leaf(i).lect = tasks[i].est()+tasks[i].pmin(); leaf(i).resEct = OmegaLambdaNode::undef; leaf(i).resLp = OmegaLambdaNode::undef; } update(); } else { // Enter no tasks into tree (omega = empty, lambda = empty) for (int i=tasks.size(); i--; ) { leaf(i).p = leaf(i).lp = 0; leaf(i).ect = leaf(i).lect = -Int::Limits::infinity; leaf(i).resEct = OmegaLambdaNode::undef; leaf(i).resLp = OmegaLambdaNode::undef; } init(); } } template forceinline void OmegaLambdaTree::shift(int i) { // That means that i is in omega assert(leaf(i).ect > -Int::Limits::infinity); leaf(i).p = 0; leaf(i).ect = -Int::Limits::infinity; leaf(i).resEct = i; leaf(i).resLp = i; update(i); } template forceinline void OmegaLambdaTree::oinsert(int i) { leaf(i).p = tasks[i].pmin(); leaf(i).ect = tasks[i].est()+tasks[i].pmin(); update(i); } template forceinline void OmegaLambdaTree::linsert(int i) { leaf(i).lp = tasks[i].pmin(); leaf(i).lect = tasks[i].est()+tasks[i].pmin(); leaf(i).resEct = i; leaf(i).resLp = i; update(i); } template forceinline void OmegaLambdaTree::lremove(int i) { leaf(i).lp = 0; leaf(i).lect = -Int::Limits::infinity; leaf(i).resEct = OmegaLambdaNode::undef; leaf(i).resLp = OmegaLambdaNode::undef; update(i); } template forceinline bool OmegaLambdaTree::lempty(void) const { return root().resEct < 0; } template forceinline int OmegaLambdaTree::responsible(void) const { return root().resEct; } template forceinline int OmegaLambdaTree::ect(void) const { return root().ect; } template forceinline int OmegaLambdaTree::lect(void) const { return root().lect; } }}} // STATISTICS: int-prop