/* -*- mode: C++; c-basic-offset: 2; indent-tabs-mode: nil -*- */ /* * Main authors: * Christian Schulte * * Copyright: * Christian Schulte, 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 "test/int.hh" #include namespace Test { namespace Int { /// Tests for cumulative scheduling constraints namespace Cumulative { /** * \defgroup TaskTestIntCumulative Cumulative scheduling constraints * \ingroup TaskTestInt */ //@{ /// Test for cumulative constraint with mandatory tasks class ManFixPCumulative : public Test { protected: /// Capacity of resource int c; /// The processing times Gecode::IntArgs p; /// The resource usage Gecode::IntArgs u; /// Get a reasonable maximal start time static int st(int c, const Gecode::IntArgs& p, const Gecode::IntArgs& u) { double e = 0; for (int i=p.size(); i--; ) e += static_cast(p[i])*u[i]; return e / std::max(1,std::abs(c)); } /// Offset int o; public: /// Create and register test ManFixPCumulative(int c0, const Gecode::IntArgs& p0, const Gecode::IntArgs& u0, int o0, Gecode::IntPropLevel ipl0) : Test("Cumulative::Man::Fix::"+str(o0)+"::"+ str(c0)+"::"+str(p0)+"::"+str(u0)+"::"+str(ipl0), (c0 >= 0) ? p0.size():p0.size()+1,0,st(c0,p0,u0),false,ipl0), c(c0), p(p0), u(u0), o(o0) { testsearch = false; testfix = false; contest = CTL_NONE; } /// Create and register initial assignment virtual Assignment* assignment(void) const { return new RandomAssignment(arity, dom, 500, _rand); } /// Test whether \a x is solution virtual bool solution(const Assignment& x) const { int cmax = (c >= 0) ? c : x[x.size()-1]; int n = (c >= 0) ? x.size() : x.size()-1; if (c < 0 && x[n] > -c) return false; // Compute maximal time int t = 0; for (int i=0; i cmax) { delete [] used; return false; } // Compute resource usage (only internal) for (int i=0; i cmax) { delete [] used; return false; } delete [] used; return true; } /// Post constraint on \a x virtual void post(Gecode::Space& home, Gecode::IntVarArray& x) { int n = (c >= 0) ? x.size() : x.size()-1; Gecode::IntVarArgs xx; if (o==0) { xx=x.slice(0,1,n); } else { xx=Gecode::IntVarArgs(n); for (int i=n; i--;) xx[i]=Gecode::expr(home,x[i]+o,Gecode::IPL_DOM); } if (c >= 0) { Gecode::cumulative(home, c, xx, p, u, ipl); } else { Gecode::rel(home, x[n] <= -c); Gecode::cumulative(home, x[n], xx, p, u, ipl); } } }; /// Test for cumulative constraint with optional tasks class OptFixPCumulative : public Test { protected: /// Capacity of resource int c; /// The processing times Gecode::IntArgs p; /// The resource usage Gecode::IntArgs u; /// Limit for optional tasks int l; /// Offset int o; /// Get a reasonable maximal start time static int st(int c, const Gecode::IntArgs& p, const Gecode::IntArgs& u) { double e = 0; for (int i=p.size(); i--; ) e += static_cast(p[i])*u[i]; return e / std::max(1,std::abs(c)); } public: /// Create and register test OptFixPCumulative(int c0, const Gecode::IntArgs& p0, const Gecode::IntArgs& u0, int o0, Gecode::IntPropLevel ipl0) : Test("Cumulative::Opt::Fix::"+str(o0)+"::"+ str(c0)+"::"+str(p0)+"::"+str(u0)+"::"+str(ipl0), (c0 >= 0) ? 2*p0.size() : 2*p0.size()+1,0,st(c0,p0,u0), false,ipl0), c(c0), p(p0), u(u0), l(st(c,p,u)/2), o(o0) { testsearch = false; testfix = false; contest = CTL_NONE; } /// Create and register initial assignment virtual Assignment* assignment(void) const { return new RandomAssignment(arity, dom, 500, _rand); } /// Test whether \a x is solution virtual bool solution(const Assignment& x) const { int nn = (c >= 0) ? x.size() : x.size()-1; int cmax = (c >= 0) ? c : x[nn]; if (c < 0 && x[nn] > -c) return false; int n = nn / 2; // Compute maximal time int t = 0; for (int i=0; i l) for (int t=0; t cmax) { delete [] used; return false; } } // Compute resource usage (only internal) for (int i=0; i l) { for (int t=1; t l) if (used[x[i]]+u[i] > cmax) { delete [] used; return false; } delete [] used; return true; } /// Post constraint on \a x virtual void post(Gecode::Space& home, Gecode::IntVarArray& x) { int nn=(c >= 0) ? x.size() : x.size()-1; int n=nn / 2; Gecode::IntVarArgs s(n); Gecode::BoolVarArgs m(n); for (int i=0; i= 0) ? x[i] : Gecode::expr(home,x[i]+o,Gecode::IPL_DOM); m[i]=Gecode::expr(home, x[n+i] > l); } if (c >= 0) { Gecode::cumulative(home, c, s, p, u, m, ipl); } else { Gecode::rel(home, x[nn] <= -c); Gecode::cumulative(home, x[nn], s, p, u, m, ipl); } } }; /// Test for cumulative constraint with flexible mandatory tasks class ManFlexCumulative : public Test { protected: /// Capacity of resource int c; /// Minimum processing time int _minP; /// Maximum processing time int _maxP; /// The resource usage Gecode::IntArgs u; /// Get a reasonable maximal start time static int st(int c, int maxP, const Gecode::IntArgs& u) { double e = 0; for (int i=u.size(); i--; ) e += static_cast(maxP)*u[i]; return e / std::max(1,std::abs(c)); } /// Offset int o; public: /// Create and register test ManFlexCumulative(int c0, int minP, int maxP, const Gecode::IntArgs& u0, int o0, Gecode::IntPropLevel ipl0) : Test("Cumulative::Man::Flex::"+str(o0)+"::"+ str(c0)+"::"+str(minP)+"::"+str(maxP)+"::"+str(u0)+ "::"+str(ipl0), (c0 >= 0) ? 2*u0.size() : 2*u0.size()+1, 0,std::max(maxP,st(c0,maxP,u0)),false,ipl0), c(c0), _minP(minP), _maxP(maxP), u(u0), o(o0) { testsearch = false; testfix = false; contest = CTL_NONE; } /// Create and register initial assignment virtual Assignment* assignment(void) const { return new RandomMixAssignment((c >= 0) ? arity / 2 : arity / 2 + 1, dom, arity / 2, Gecode::IntSet(_minP, _maxP), 500, _rand); } /// Test whether \a x is solution virtual bool solution(const Assignment& x) const { int nn = (c >= 0) ? x.size() : x.size()-1; int n = nn/2; int cmax = (c >= 0) ? c : x[n]; int pstart = (c >= 0) ? n : n+1; if (c < 0 && cmax > -c) return false; // Compute maximal time int t = 0; for (int i=0; i cmax) { delete [] used; return false; } // Compute resource usage (only internal) for (int i=0; i cmax) { delete [] used; return false; } delete [] used; return true; } /// Post constraint on \a x virtual void post(Gecode::Space& home, Gecode::IntVarArray& x) { int nn = (c >= 0) ? x.size() : x.size()-1; int n = nn/2; int pstart = (c >= 0) ? n : n+1; Gecode::IntVarArgs s(n); Gecode::IntVarArgs px(x.slice(pstart,1,n)); Gecode::IntVarArgs e(home,n, Gecode::Int::Limits::min, Gecode::Int::Limits::max); for (int i=s.size(); i--;) { s[i] = expr(home, o+x[i], Gecode::IPL_DOM); rel(home, s[i]+px[i] == e[i]); rel(home, _minP <= px[i]); rel(home, _maxP >= px[i]); } if (c >= 0) { Gecode::cumulative(home, c, s, px, e, u, ipl); } else { rel(home, x[n] <= -c); Gecode::cumulative(home, x[n], s, px, e, u, ipl); } } }; /// Test for cumulative constraint with optional flexible tasks class OptFlexCumulative : public Test { protected: /// Capacity of resource int c; /// Minimum processing time int _minP; /// Maximum processing time int _maxP; /// The resource usage Gecode::IntArgs u; /// Limit for optional tasks int l; /// Offset int o; /// Get a reasonable maximal start time static int st(int c, int maxP, const Gecode::IntArgs& u) { double e = 0; for (int i=u.size(); i--; ) e += static_cast(maxP)*u[i]; return e / std::max(1,std::abs(c)); } public: /// Create and register test OptFlexCumulative(int c0, int minP, int maxP, const Gecode::IntArgs& u0, int o0, Gecode::IntPropLevel ipl0) : Test("Cumulative::Opt::Flex::"+str(o0)+"::"+ str(c0)+"::"+str(minP)+"::"+str(maxP)+"::"+str(u0)+ "::"+str(ipl0), (c0 >= 0) ? 3*u0.size() : 3*u0.size()+1, 0,std::max(maxP,st(c0,maxP,u0)), false,ipl0), c(c0), _minP(minP), _maxP(maxP), u(u0), l(std::max(maxP,st(c0,maxP,u0))/2), o(o0) { testsearch = false; testfix = false; contest = CTL_NONE; } /// Create and register initial assignment virtual Assignment* assignment(void) const { return new RandomMixAssignment((c >= 0) ? 2 * (arity / 3) : 2 * (arity / 3) + 1, dom, arity / 3, Gecode::IntSet(_minP, _maxP), 500, _rand); } /// Test whether \a x is solution virtual bool solution(const Assignment& x) const { int nn = (c >= 0) ? x.size() : x.size()-1; int n = nn / 3; int cmax = (c >= 0) ? c : x[2*n]; int pstart = (c >= 0) ? 2*n : 2*n+1; if (c < 0 && cmax > -c) return false; // Compute maximal time int t = 0; for (int i=0; i l) for (int t=0; t cmax) { delete [] used; return false; } // Compute resource usage (only internal) for (int i=0; i l) for (int t=1; t l && used[x[i]]+u[i] > cmax) { delete [] used; return false; } delete [] used; return true; } /// Post constraint on \a x virtual void post(Gecode::Space& home, Gecode::IntVarArray& x) { int nn = (c >= 0) ? x.size() : x.size()-1; int n=nn / 3; int pstart= (c >= 0) ? 2*n : 2*n+1; Gecode::IntVarArgs s(n); Gecode::IntVarArgs px(n); Gecode::IntVarArgs e(home,n, Gecode::Int::Limits::min, Gecode::Int::Limits::max); for (int i=n; i--;) { s[i] = expr(home, o+x[i]); px[i] = x[pstart+i]; rel(home, s[i]+px[i] == e[i]); rel(home, _minP <= px[i]); rel(home, _maxP >= px[i]); } Gecode::BoolVarArgs m(n); for (int i=0; i l)); if (c >= 0) { Gecode::cumulative(home, c, s, px, e, u, m, ipl); } else { Gecode::rel(home, x[2*n] <= -c); Gecode::cumulative(home, x[2*n], s, px, e, u, m, ipl); } } }; /// Help class to create and register tests class Create { public: /// Perform creation and registration Create(void) { using namespace Gecode; IntArgs p1({1,1,1,1}); IntArgs p2({2,2,2,2}); IntArgs p3({4,3,3,5}); IntArgs p4({4,0,3,5}); IntArgs p5({1,1,1}); IntArgs u1({1,1,1,1}); IntArgs u2({2,2,2,2}); IntArgs u3({2,3,4,5}); IntArgs u4({2,3,0,5}); IntArgs u5({1,3,2}); for (IntPropBasicAdvanced ipba; ipba(); ++ipba) { // Regression test: check correct detection of disjunctive case (void) new ManFixPCumulative(3,p5,u5,0,ipba.ipl()); for (int c=-7; c<8; c++) { int off = 0; for (int coff=0; coff<2; coff++) { (void) new ManFixPCumulative(c,p1,u1,off,ipba.ipl()); (void) new ManFixPCumulative(c,p1,u2,off,ipba.ipl()); (void) new ManFixPCumulative(c,p1,u3,off,ipba.ipl()); (void) new ManFixPCumulative(c,p1,u4,off,ipba.ipl()); (void) new ManFixPCumulative(c,p2,u1,off,ipba.ipl()); (void) new ManFixPCumulative(c,p2,u2,off,ipba.ipl()); (void) new ManFixPCumulative(c,p2,u3,off,ipba.ipl()); (void) new ManFixPCumulative(c,p2,u4,off,ipba.ipl()); (void) new ManFixPCumulative(c,p3,u1,off,ipba.ipl()); (void) new ManFixPCumulative(c,p3,u2,off,ipba.ipl()); (void) new ManFixPCumulative(c,p3,u3,off,ipba.ipl()); (void) new ManFixPCumulative(c,p3,u4,off,ipba.ipl()); (void) new ManFixPCumulative(c,p4,u1,off,ipba.ipl()); (void) new ManFixPCumulative(c,p4,u2,off,ipba.ipl()); (void) new ManFixPCumulative(c,p4,u3,off,ipba.ipl()); (void) new ManFixPCumulative(c,p4,u4,off,ipba.ipl()); (void) new ManFlexCumulative(c,0,1,u1,off,ipba.ipl()); (void) new ManFlexCumulative(c,0,1,u2,off,ipba.ipl()); (void) new ManFlexCumulative(c,0,1,u3,off,ipba.ipl()); (void) new ManFlexCumulative(c,0,1,u4,off,ipba.ipl()); (void) new ManFlexCumulative(c,0,2,u1,off,ipba.ipl()); (void) new ManFlexCumulative(c,0,2,u2,off,ipba.ipl()); (void) new ManFlexCumulative(c,0,2,u3,off,ipba.ipl()); (void) new ManFlexCumulative(c,0,2,u4,off,ipba.ipl()); (void) new ManFlexCumulative(c,3,5,u1,off,ipba.ipl()); (void) new ManFlexCumulative(c,3,5,u2,off,ipba.ipl()); (void) new ManFlexCumulative(c,3,5,u3,off,ipba.ipl()); (void) new ManFlexCumulative(c,3,5,u4,off,ipba.ipl()); (void) new OptFixPCumulative(c,p1,u1,off,ipba.ipl()); (void) new OptFixPCumulative(c,p1,u2,off,ipba.ipl()); (void) new OptFixPCumulative(c,p1,u3,off,ipba.ipl()); (void) new OptFixPCumulative(c,p1,u4,off,ipba.ipl()); (void) new OptFixPCumulative(c,p2,u1,off,ipba.ipl()); (void) new OptFixPCumulative(c,p2,u2,off,ipba.ipl()); (void) new OptFixPCumulative(c,p2,u3,off,ipba.ipl()); (void) new OptFixPCumulative(c,p2,u4,off,ipba.ipl()); (void) new OptFixPCumulative(c,p3,u1,off,ipba.ipl()); (void) new OptFixPCumulative(c,p3,u2,off,ipba.ipl()); (void) new OptFixPCumulative(c,p3,u3,off,ipba.ipl()); (void) new OptFixPCumulative(c,p3,u4,off,ipba.ipl()); (void) new OptFixPCumulative(c,p4,u1,off,ipba.ipl()); (void) new OptFixPCumulative(c,p4,u2,off,ipba.ipl()); (void) new OptFixPCumulative(c,p4,u3,off,ipba.ipl()); (void) new OptFixPCumulative(c,p4,u4,off,ipba.ipl()); (void) new OptFlexCumulative(c,0,1,u1,off,ipba.ipl()); (void) new OptFlexCumulative(c,0,1,u2,off,ipba.ipl()); (void) new OptFlexCumulative(c,0,1,u3,off,ipba.ipl()); (void) new OptFlexCumulative(c,0,1,u4,off,ipba.ipl()); (void) new OptFlexCumulative(c,0,2,u1,off,ipba.ipl()); (void) new OptFlexCumulative(c,0,2,u2,off,ipba.ipl()); (void) new OptFlexCumulative(c,0,2,u3,off,ipba.ipl()); (void) new OptFlexCumulative(c,0,2,u4,off,ipba.ipl()); (void) new OptFlexCumulative(c,3,5,u1,off,ipba.ipl()); (void) new OptFlexCumulative(c,3,5,u2,off,ipba.ipl()); (void) new OptFlexCumulative(c,3,5,u3,off,ipba.ipl()); (void) new OptFlexCumulative(c,3,5,u4,off,ipba.ipl()); off = Gecode::Int::Limits::min; } } } } }; Create c; //@} } }} // STATISTICS: test-int