max.hpp gecode/int/rel.hh /usr/include/gecode/int/arithmetic.hh Gecode Gecode::Int Gecode::Int::Arithmetic /*-*-mode:C++;c-basic-offset:2;indent-tabs-mode:nil-*-*/ /* *Mainauthors: *ChristianSchulte<schulte@gecode.org> * *Copyright: *ChristianSchulte,2004 * *Lastmodified: *$Date:2012-09-0717:31:22+0200(Fri,07Sep2012)$by$Author:schulte$ *$Revision:13068$ * *ThisfileispartofGecode,thegenericconstraint *developmentenvironment: *http://www.gecode.org * *Permissionisherebygranted,freeofcharge,toanypersonobtaining *acopyofthissoftwareandassociateddocumentationfiles(the *"Software"),todealintheSoftwarewithoutrestriction,including *withoutlimitationtherightstouse,copy,modify,merge,publish, *distribute,sublicense,and/orsellcopiesoftheSoftware,andto *permitpersonstowhomtheSoftwareisfurnishedtodoso,subjectto *thefollowingconditions: * *Theabovecopyrightnoticeandthispermissionnoticeshallbe *includedinallcopiesorsubstantialportionsoftheSoftware. * *THESOFTWAREISPROVIDED"ASIS",WITHOUTWARRANTYOFANYKIND, *EXPRESSORIMPLIED,INCLUDINGBUTNOTLIMITEDTOTHEWARRANTIESOF *MERCHANTABILITY,FITNESSFORAPARTICULARPURPOSEAND *NONINFRINGEMENT.INNOEVENTSHALLTHEAUTHORSORCOPYRIGHTHOLDERSBE *LIABLEFORANYCLAIM,DAMAGESOROTHERLIABILITY,WHETHERINANACTION *OFCONTRACT,TORTOROTHERWISE,ARISINGFROM,OUTOFORINCONNECTION *WITHTHESOFTWAREORTHEUSEOROTHERDEALINGSINTHESOFTWARE. * */ #include<gecode/int/rel.hh> namespaceGecode{namespaceInt{namespaceArithmetic{ /* *Ternaryboundsconsistentmaximum * */ template<classView> forceinlineExecStatus prop_max_bnd(Space&home,Viewx0,Viewx1,Viewx2){ boolmod=false; do{ mod=false; { ModEventme=x2.lq(home,std::max(x0.max(),x1.max())); if(me_failed(me))returnES_FAILED; mod|=me_modified(me); } { ModEventme=x2.gq(home,std::max(x0.min(),x1.min())); if(me_failed(me))returnES_FAILED; mod|=me_modified(me); } { ModEventme=x0.lq(home,x2.max()); if(me_failed(me))returnES_FAILED; mod|=me_modified(me); } { ModEventme=x1.lq(home,x2.max()); if(me_failed(me))returnES_FAILED; mod|=me_modified(me); } }while(mod); returnES_OK; } template<classView> forceinline MaxBnd<View>::MaxBnd(Homehome,Viewx0,Viewx1,Viewx2) :TernaryPropagator<View,PC_INT_BND>(home,x0,x1,x2){} template<classView> ExecStatus MaxBnd<View>::post(Homehome,Viewx0,Viewx1,Viewx2){ GECODE_ME_CHECK(x2.gq(home,std::max(x0.min(),x1.min()))); GECODE_ME_CHECK(x2.lq(home,std::max(x0.max(),x1.max()))); if(same(x0,x1)) returnRel::EqBnd<View,View>::post(home,x0,x2); if(same(x0,x2)) returnRel::Lq<View>::post(home,x1,x2); if(same(x1,x2)) returnRel::Lq<View>::post(home,x0,x2); (void)new(home)MaxBnd<View>(home,x0,x1,x2); returnES_OK; } template<classView> forceinline MaxBnd<View>::MaxBnd(Space&home,boolshare,MaxBnd<View>&p) :TernaryPropagator<View,PC_INT_BND>(home,share,p){} template<classView> forceinline MaxBnd<View>::MaxBnd(Space&home,boolshare,Propagator&p, Viewx0,Viewx1,Viewx2) :TernaryPropagator<View,PC_INT_BND>(home,share,p,x0,x1,x2){} template<classView> Actor* MaxBnd<View>::copy(Space&home,boolshare){ returnnew(home)MaxBnd<View>(home,share,*this); } template<classView> ExecStatus MaxBnd<View>::propagate(Space&home,constModEventDelta&){ GECODE_ES_CHECK(prop_max_bnd(home,x0,x1,x2)); if((x0.max()<=x1.min())||(x0.max()<x2.min())) GECODE_REWRITE(*this,(Rel::EqBnd<View,View>::post(home(*this),x1,x2))); if((x1.max()<=x0.min())||(x1.max()<x2.min())) GECODE_REWRITE(*this,(Rel::EqBnd<View,View>::post(home(*this),x0,x2))); returnx0.assigned()&&x1.assigned()&&x2.assigned()? home.ES_SUBSUMED(*this):ES_FIX; } /* *Naryboundsconsistentmaximum * */ template<classView> forceinline NaryMaxBnd<View>::NaryMaxBnd(Homehome,ViewArray<View>&x,Viewy) :NaryOnePropagator<View,PC_INT_BND>(home,x,y){} template<classView> ExecStatus NaryMaxBnd<View>::post(Homehome,ViewArray<View>&x,Viewy){ assert(x.size()>0); x.unique(home); if(x.size()==1) returnRel::EqBnd<View,View>::post(home,x[0],y); if(x.size()==2) returnMaxBnd<View>::post(home,x[0],x[1],y); intl=Int::Limits::min; intu=Int::Limits::min; for(inti=x.size();i--;){ l=std::max(l,x[i].min()); u=std::max(u,x[i].max()); } GECODE_ME_CHECK(y.gq(home,l)); GECODE_ME_CHECK(y.lq(home,u)); if(x.same(home,y)){ //Checkwhetheryoccursinx for(inti=x.size();i--;) GECODE_ES_CHECK(Rel::Lq<View>::post(home,x[i],y)); }else{ (void)new(home)NaryMaxBnd<View>(home,x,y); } returnES_OK; } template<classView> forceinline NaryMaxBnd<View>::NaryMaxBnd(Space&home,boolshare,NaryMaxBnd<View>&p) :NaryOnePropagator<View,PC_INT_BND>(home,share,p){} template<classView> Actor* NaryMaxBnd<View>::copy(Space&home,boolshare){ if(x.size()==1) returnnew(home)Rel::EqBnd<View,View>(home,share,*this,x[0],y); if(x.size()==2) returnnew(home)MaxBnd<View>(home,share,*this,x[0],x[1],y); returnnew(home)NaryMaxBnd<View>(home,share,*this); } enumMaxPropStatus{ MPS_ASSIGNED=1<<0, MPS_REMOVED=1<<1, MPS_NEW_BOUND=1<<2 }; template<classView> forceinlineExecStatus prop_nary_max_bnd(Space&home,Propagator&p, ViewArray<View>&x,Viewy,PropCondpc){ rerun: assert(x.size()>0); intmaxmax=x[x.size()-1].max(); intmaxmin=x[x.size()-1].min(); for(inti=x.size()-1;i--;){ maxmax=std::max(x[i].max(),maxmax); maxmin=std::max(x[i].min(),maxmin); } GECODE_ME_CHECK(y.lq(home,maxmax)); GECODE_ME_CHECK(y.gq(home,maxmin)); maxmin=y.min(); maxmax=y.max(); intstatus=MPS_ASSIGNED; for(inti=x.size();i--;){ ModEventme=x[i].lq(home,maxmax); if(me==ME_INT_FAILED) returnES_FAILED; if(me_modified(me)&&(x[i].max()!=maxmax)) status|=MPS_NEW_BOUND; if(x[i].max()<maxmin){ x.move_lst(i,home,p,pc); status|=MPS_REMOVED; }elseif(!x[i].assigned()) status&=~MPS_ASSIGNED; } if(x.size()==0) returnES_FAILED; if((status&MPS_REMOVED)!=0) gotorerun; if(((status&MPS_ASSIGNED)!=0)&&y.assigned()) returnhome.ES_SUBSUMED(p); return((status&MPS_NEW_BOUND)!=0)?ES_NOFIX:ES_FIX; } template<classView> ExecStatus NaryMaxBnd<View>::propagate(Space&home,constModEventDelta&){ ExecStatuses=prop_nary_max_bnd(home,*this,x,y,PC_INT_BND); GECODE_ES_CHECK(es); if(x.size()==1) GECODE_REWRITE(*this,(Rel::EqBnd<View,View>::post(home(*this),x[0],y))); returnes; } /* *Ternarydomainconsistentmaximum * */ template<classView> forceinline MaxDom<View>::MaxDom(Homehome,Viewx0,Viewx1,Viewx2) :TernaryPropagator<View,PC_INT_DOM>(home,x0,x1,x2){} template<classView> ExecStatus MaxDom<View>::post(Homehome,Viewx0,Viewx1,Viewx2){ GECODE_ME_CHECK(x2.gq(home,std::max(x0.min(),x1.min()))); GECODE_ME_CHECK(x2.lq(home,std::max(x0.max(),x1.max()))); if(same(x0,x1)) returnRel::EqDom<View,View>::post(home,x0,x2); if(same(x0,x2)) returnRel::Lq<View>::post(home,x1,x2); if(same(x1,x2)) returnRel::Lq<View>::post(home,x0,x2); (void)new(home)MaxDom<View>(home,x0,x1,x2); returnES_OK; } template<classView> forceinline MaxDom<View>::MaxDom(Space&home,boolshare,MaxDom<View>&p) :TernaryPropagator<View,PC_INT_DOM>(home,share,p){} template<classView> forceinline MaxDom<View>::MaxDom(Space&home,boolshare,Propagator&p, Viewx0,Viewx1,Viewx2) :TernaryPropagator<View,PC_INT_DOM>(home,share,p,x0,x1,x2){} template<classView> Actor* MaxDom<View>::copy(Space&home,boolshare){ returnnew(home)MaxDom<View>(home,share,*this); } template<classView> PropCost MaxDom<View>::cost(constSpace&,constModEventDelta&med)const{ returnPropCost::ternary((View::me(med)==ME_INT_DOM)? PropCost::HI:PropCost::LO); } template<classView> ExecStatus MaxDom<View>::propagate(Space&home,constModEventDelta&med){ if(View::me(med)!=ME_INT_DOM){ GECODE_ES_CHECK(prop_max_bnd(home,x0,x1,x2)); if((x0.max()<=x1.min())||(x0.max()<x2.min())) GECODE_REWRITE(*this,(Rel::EqDom<View,View>::post(home(*this),x1,x2))); if((x1.max()<=x0.min())||(x1.max()<x2.min())) GECODE_REWRITE(*this,(Rel::EqDom<View,View>::post(home(*this),x0,x2))); returnx0.assigned()&&x1.assigned()&&x2.assigned()? home.ES_SUBSUMED(*this): home.ES_NOFIX_PARTIAL(*this,View::med(ME_INT_DOM)); } ViewRanges<View>r0(x0),r1(x1); Iter::Ranges::Union<ViewRanges<View>,ViewRanges<View>>u(r0,r1); GECODE_ME_CHECK(x2.inter_r(home,u,false)); if(rtest_nq_dom(x0,x2)==RT_TRUE){ GECODE_ES_CHECK(Rel::Lq<View>::post(home,x0,x2)); GECODE_REWRITE(*this,(Rel::EqDom<View,View>::post(home(*this),x1,x2))); } if(rtest_nq_dom(x1,x2)==RT_TRUE){ GECODE_ES_CHECK(Rel::Lq<View>::post(home,x1,x2)); GECODE_REWRITE(*this,(Rel::EqDom<View,View>::post(home(*this),x0,x2))); } returnES_FIX; } /* *Narydomainconsistentmaximum * */ template<classView> forceinline NaryMaxDom<View>::NaryMaxDom(Homehome,ViewArray<View>&x,Viewy) :NaryOnePropagator<View,PC_INT_DOM>(home,x,y){} template<classView> ExecStatus NaryMaxDom<View>::post(Homehome,ViewArray<View>&x,Viewy){ assert(x.size()>0); x.unique(home); if(x.size()==1) returnRel::EqDom<View,View>::post(home,x[0],y); if(x.size()==2) returnMaxDom<View>::post(home,x[0],x[1],y); intl=Int::Limits::min; intu=Int::Limits::min; for(inti=x.size();i--;){ l=std::max(l,x[i].min()); u=std::max(u,x[i].max()); } GECODE_ME_CHECK(y.gq(home,l)); GECODE_ME_CHECK(y.lq(home,u)); if(x.same(home,y)){ //Checkwhetheryoccursinx for(inti=x.size();i--;) GECODE_ES_CHECK(Rel::Lq<View>::post(home,x[i],y)); }else{ (void)new(home)NaryMaxDom<View>(home,x,y); } returnES_OK; } template<classView> forceinline NaryMaxDom<View>::NaryMaxDom(Space&home,boolshare,NaryMaxDom<View>&p) :NaryOnePropagator<View,PC_INT_DOM>(home,share,p){} template<classView> Actor* NaryMaxDom<View>::copy(Space&home,boolshare){ if(x.size()==1) returnnew(home)Rel::EqDom<View,View>(home,share,*this,x[0],y); if(x.size()==2) returnnew(home)MaxDom<View>(home,share,*this,x[0],x[1],y); returnnew(home)NaryMaxDom<View>(home,share,*this); } template<classView> PropCost NaryMaxDom<View>::cost(constSpace&,constModEventDelta&med)const{ if(View::me(med)==ME_INT_DOM) returnPropCost::linear(PropCost::HI,y.size()); else returnPropCost::linear(PropCost::LO,x.size()); } template<classView> ExecStatus NaryMaxDom<View>::propagate(Space&home,constModEventDelta&med){ if(View::me(med)!=ME_INT_DOM){ ExecStatuses=prop_nary_max_bnd(home,*this,x,y,PC_INT_DOM); GECODE_ES_CHECK(es); return(es==ES_FIX)? home.ES_FIX_PARTIAL(*this,View::med(ME_INT_DOM)): home.ES_NOFIX_PARTIAL(*this,View::med(ME_INT_DOM)); } Regionr(home); ViewRanges<View>*i_x=r.alloc<ViewRanges<View>>(x.size()); for(inti=x.size();i--;){ ViewRanges<View>i_xi(x[i]);i_x[i]=i_xi; } Iter::Ranges::NaryUnionu(r,i_x,x.size()); GECODE_ME_CHECK(y.inter_r(home,u,false)); for(inti=x.size();i--;) if(rtest_nq_dom(x[i],y)==RT_TRUE){ GECODE_ES_CHECK(Rel::Lq<View>::post(home,x[i],y)); x.move_lst(i,home,*this,PC_INT_DOM); } assert(x.size()>0); if(x.size()==1) GECODE_REWRITE(*this,(Rel::EqDom<View,View>::post(home(*this),x[0],y))); returnES_FIX; } }}} //STATISTICS:int-prop