nroot.hpp gecode/int/rel.hh climits algorithm /usr/include/gecode/int/arithmetic.hh Gecode::Int::Arithmetic::RangesMapPow Gecode::Int::Arithmetic::RangesMapNroot Gecode Gecode::Int Gecode::Int::Arithmetic /*-*-mode:C++;c-basic-offset:2;indent-tabs-mode:nil-*-*/ /* *Mainauthors: *ChristianSchulte<schulte@gecode.org> * *Copyright: *ChristianSchulte,2012 * *Lastmodified: *$Date:2013-08-2916:05:54+0200(Thu,29Aug2013)$by$Author:schulte$ *$Revision:13993$ * *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> #include<climits> #include<algorithm> namespaceGecode{namespaceInt{namespaceArithmetic{ /* *Positiveboundsconsistentnthroot * */ template<classOps,boolminus> forceinlineExecStatus prop_nroot_plus_bnd(Space&home,IntViewx0,IntViewx1,constOps&ops){ if(minus){ boolmod; do{ mod=false; { ModEventme=x1.gq(home,-ops.cnroot(-x0.min())); if(me_failed(me))returnES_FAILED; mod|=me_modified(me); } { ModEventme=x1.lq(home,-ops.cnroot(-x0.max())); if(me_failed(me))returnES_FAILED; mod|=me_modified(me); } { ModEventme=x0.gq(home,-ops.tpow(-x1.min())); if(me_failed(me))returnES_FAILED; mod|=me_modified(me); } { ModEventme=x0.lq(home,-(ops.tpow(-x1.max()-1)+1)); if(me_failed(me))returnES_FAILED; mod|=me_modified(me); } }while(mod); }else{ boolmod; do{ mod=false; { ModEventme=x1.lq(home,ops.fnroot(x0.max())); if(me_failed(me))returnES_FAILED; mod|=me_modified(me); } { ModEventme=x1.gq(home,ops.fnroot(x0.min())); if(me_failed(me))returnES_FAILED; mod|=me_modified(me); } { ModEventme=x0.le(home,ops.tpow(x1.max()+1)); if(me_failed(me))returnES_FAILED; mod|=me_modified(me); } { ModEventme=x0.gq(home,ops.tpow(x1.min())); if(me_failed(me))returnES_FAILED; mod|=me_modified(me); } }while(mod); } returnES_OK; } template<classOps,boolminus> forceinline NrootPlusBnd<Ops,minus>::NrootPlusBnd(Homehome,IntViewx0,IntViewx1, constOps&o) :BinaryPropagator<IntView,PC_INT_BND>(home,x0,x1), ops(o){} template<classOps,boolminus> forceinlineExecStatus NrootPlusBnd<Ops,minus>::post(Homehome,IntViewx0,IntViewx1,Opsops){ if(minus){ GECODE_ME_CHECK(x0.lq(home,0)); GECODE_ME_CHECK(x1.lq(home,0)); }else{ GECODE_ME_CHECK(x0.gq(home,0)); GECODE_ME_CHECK(x1.gq(home,0)); } (void)new(home)NrootPlusBnd<Ops,minus>(home,x0,x1,ops); returnES_OK; } template<classOps,boolminus> forceinline NrootPlusBnd<Ops,minus>::NrootPlusBnd(Space&home,boolshare, NrootPlusBnd<Ops,minus>&p) :BinaryPropagator<IntView,PC_INT_BND>(home,share,p), ops(p.ops){} template<classOps,boolminus> Actor* NrootPlusBnd<Ops,minus>::copy(Space&home,boolshare){ returnnew(home)NrootPlusBnd<Ops,minus>(home,share,*this); } template<classOps,boolminus> ExecStatus NrootPlusBnd<Ops,minus>::propagate(Space&home,constModEventDelta&){ GECODE_ES_CHECK((prop_nroot_plus_bnd<Ops,minus>(home,x0,x1,ops))); returnx1.assigned()?home.ES_SUBSUMED(*this):ES_FIX; } /* *Boundsconsistentnthroot * */ template<classOps> forceinlineExecStatus prop_nroot_bnd(Space&home,IntViewx0,IntViewx1,constOps&ops){ assert((x0.min()<0)&&(x0.max()>0)); assert((x1.min()<0)&&(x1.max()>0)); GECODE_ME_CHECK(x1.lq(home,ops.fnroot(x0.max()))); GECODE_ME_CHECK(x1.gq(home,-ops.cnroot(-x0.min()))); GECODE_ME_CHECK(x0.le(home,ops.tpow(x1.max()+1))); GECODE_ME_CHECK(x0.gq(home,ops.tpow(x1.min()-1)+1)); returnES_OK; } template<classOps> forceinline NrootBnd<Ops>::NrootBnd(Homehome,IntViewx0,IntViewx1,constOps&o) :BinaryPropagator<IntView,PC_INT_BND>(home,x0,x1), ops(o){} template<classOps> forceinlineExecStatus NrootBnd<Ops>::post(Homehome,IntViewx0,IntViewx1,Opsops){ if(static_cast<unsignedint>(ops.exp())>=sizeof(int)*CHAR_BIT){ //Theintegerlimitsallowonly-2,-1,0,1forx1 GECODE_ME_CHECK(x1.lq(home,1)); GECODE_ME_CHECK(x1.gq(home,-2)); //Justrewritetovaluesthatcanbehandeledwithoutoverflow ops.exp(ops.even()?30:31); } if(ops.exp()==0){ GECODE_ME_CHECK(x1.eq(home,1)); returnES_OK; }elseif(ops.exp()==1){ returnRel::EqBnd<IntView,IntView>::post(home,x0,x1); } if(same(x0,x1)){ assert(ops.exp()>1); GECODE_ME_CHECK(x0.lq(home,1)); GECODE_ME_CHECK(x0.gq(home,ops.even()?0:-2)); returnES_OK; } //Limitsvaluessuchthatnooverflowcanoccur GECODE_ME_CHECK(x1.lq(home,ops.fnroot(Limits::max))); GECODE_ME_CHECK(x1.gq(home,-ops.cnroot(-Limits::min))); if(ops.even()){ GECODE_ME_CHECK(x0.gq(home,0)); GECODE_ME_CHECK(x1.gq(home,0)); } if((x0.min()>=0)||(x1.min()>=0)) returnNrootPlusBnd<Ops,false>::post(home,x0,x1,ops); if((x0.max()<=0)||(x1.max()<=0)) returnNrootPlusBnd<Ops,true>::post(home,x0,x1,ops); assert((x0.min()<0)&&(x0.max()>0)); assert((x1.min()<0)&&(x1.max()>0)); GECODE_ES_CHECK(prop_nroot_bnd<Ops>(home,x0,x1,ops)); (void)new(home)NrootBnd(home,x0,x1,ops); returnES_OK; } template<classOps> forceinline NrootBnd<Ops>::NrootBnd(Space&home,boolshare,NrootBnd<Ops>&p) :BinaryPropagator<IntView,PC_INT_BND>(home,share,p), ops(p.ops){} template<classOps> Actor* NrootBnd<Ops>::copy(Space&home,boolshare){ returnnew(home)NrootBnd<Ops>(home,share,*this); } template<classOps> ExecStatus NrootBnd<Ops>::propagate(Space&home,constModEventDelta&){ assert(!ops.even()); if((x0.min()>=0)||(x1.min()>=0)) GECODE_REWRITE(*this,(NrootPlusBnd<Ops,false>::post(home,x0,x1,ops))); if((x0.max()<=0)||(x1.max()<=0)) GECODE_REWRITE(*this,(NrootPlusBnd<Ops,true>::post(home,x0,x1,ops))); GECODE_ES_CHECK(prop_nroot_bnd(home,x0,x1,ops)); returnx0.assigned()&&x1.assigned()?home.ES_SUBSUMED(*this):ES_NOFIX; } /* *Domainconsistentnthroot * */ template<classOps> classRangesMapPow{ protected: Opsops; public: forceinlineRangesMapPow(constOps&o):ops(o){} forceinlineintmin(intx)const{ returnops.tpow(x); } forceinlineintmax(intx)const{ returnops.tpow(x+1)-1; } }; template<classOps> classRangesMapNroot{ protected: Opsops; public: forceinlineRangesMapNroot(constOps&o):ops(o){} forceinlineintmin(intx)const{ return(x<0)?-ops.cnroot(-x):ops.fnroot(x); } forceinlineintmax(intx)const{ return(x<0)?-ops.cnroot(-x):ops.fnroot(x); } }; template<classOps,boolminus> forceinline NrootPlusDom<Ops,minus>::NrootPlusDom(Homehome,IntViewx0,IntViewx1, constOps&o) :BinaryPropagator<IntView,PC_INT_DOM>(home,x0,x1), ops(o){} template<classOps,boolminus> forceinlineExecStatus NrootPlusDom<Ops,minus>::post(Homehome,IntViewx0,IntViewx1,Opsops){ if(minus){ GECODE_ME_CHECK(x0.lq(home,0)); GECODE_ME_CHECK(x1.lq(home,0)); }else{ GECODE_ME_CHECK(x0.gq(home,0)); GECODE_ME_CHECK(x1.gq(home,0)); } GECODE_ES_CHECK((prop_nroot_plus_bnd<Ops,minus>(home,x0,x1,ops))); (void)new(home)NrootPlusDom<Ops,minus>(home,x0,x1,ops); returnES_OK; } template<classOps,boolminus> forceinline NrootPlusDom<Ops,minus>::NrootPlusDom(Space&home,boolshare, NrootPlusDom<Ops,minus>&p) :BinaryPropagator<IntView,PC_INT_DOM>(home,share,p), ops(p.ops){} template<classOps,boolminus> Actor* NrootPlusDom<Ops,minus>::copy(Space&home,boolshare){ returnnew(home)NrootPlusDom<Ops,minus>(home,share,*this); } template<classOps,boolminus> PropCost NrootPlusDom<Ops,minus>::cost(constSpace&,constModEventDelta&med)const{ if(IntView::me(med)==ME_INT_VAL) returnPropCost::unary(PropCost::LO); elseif(IntView::me(med)==ME_INT_DOM) returnPropCost::binary(PropCost::HI); else returnPropCost::binary(PropCost::LO); } template<classOps,boolminus> ExecStatus NrootPlusDom<Ops,minus>::propagate(Space&home,constModEventDelta&med){ if(IntView::me(med)!=ME_INT_DOM){ GECODE_ES_CHECK((prop_nroot_plus_bnd<Ops,minus>(home,x0,x1,ops))); returnx1.assigned()?home.ES_SUBSUMED(*this) :home.ES_NOFIX_PARTIAL(*this,IntView::med(ME_INT_DOM)); } { ViewRanges<IntView>r(x0); RangesMapNroot<Ops>rmn(ops); Iter::Ranges::Map<ViewRanges<IntView>,RangesMapNroot<Ops>,false> m(r,rmn); GECODE_ME_CHECK(x1.inter_r(home,m,false)); } { ViewRanges<IntView>r(x1); RangesMapPow<Ops>rmp(ops); Iter::Ranges::Map<ViewRanges<IntView>,RangesMapPow<Ops>,true> m(r,rmp); GECODE_ME_CHECK(x0.inter_r(home,m,false)); } returnx1.assigned()?home.ES_SUBSUMED(*this):ES_FIX; } template<classOps> forceinline NrootDom<Ops>::NrootDom(Homehome,IntViewx0,IntViewx1,constOps&o) :BinaryPropagator<IntView,PC_INT_DOM>(home,x0,x1), ops(o){} template<classOps> forceinlineExecStatus NrootDom<Ops>::post(Homehome,IntViewx0,IntViewx1,Opsops){ if(static_cast<unsignedint>(ops.exp())>=sizeof(int)*CHAR_BIT){ //Theintegerlimitsallowonly-2,-1,0,1forx1 GECODE_ME_CHECK(x1.lq(home,1)); GECODE_ME_CHECK(x1.gq(home,-2)); //Justrewritetovaluesthatcanbehandeledwithoutoverflow ops.exp(ops.even()?30:31); } if(ops.exp()==0){ GECODE_ME_CHECK(x1.eq(home,1)); returnES_OK; }elseif(ops.exp()==1){ returnRel::EqDom<IntView,IntView>::post(home,x0,x1); } if(same(x0,x1)){ assert(ops.exp()>1); GECODE_ME_CHECK(x0.lq(home,1)); GECODE_ME_CHECK(x0.gq(home,ops.even()?0:-2)); returnES_OK; } //Limitsvaluessuchthatnooverflowcanoccur GECODE_ME_CHECK(x1.lq(home,ops.fnroot(Limits::max))); GECODE_ME_CHECK(x1.gq(home,-ops.cnroot(-Limits::min))); if(ops.even()){ GECODE_ME_CHECK(x0.gq(home,0)); GECODE_ME_CHECK(x1.gq(home,0)); } if((x0.min()>=0)||(x1.min()>=0)) returnNrootPlusDom<Ops,false>::post(home,x0,x1,ops); if((x0.max()<=0)||(x1.max()<=0)) returnNrootPlusDom<Ops,true>::post(home,x0,x1,ops); assert((x0.min()<0)&&(x0.max()>0)); assert((x1.min()<0)&&(x1.max()>0)); GECODE_ES_CHECK(prop_nroot_bnd<Ops>(home,x0,x1,ops)); (void)new(home)NrootDom(home,x0,x1,ops); returnES_OK; } template<classOps> forceinline NrootDom<Ops>::NrootDom(Space&home,boolshare,NrootDom<Ops>&p) :BinaryPropagator<IntView,PC_INT_DOM>(home,share,p), ops(p.ops){} template<classOps> Actor* NrootDom<Ops>::copy(Space&home,boolshare){ returnnew(home)NrootDom<Ops>(home,share,*this); } template<classOps> PropCost NrootDom<Ops>::cost(constSpace&,constModEventDelta&med)const{ if(IntView::me(med)==ME_INT_VAL) returnPropCost::unary(PropCost::LO); elseif(IntView::me(med)==ME_INT_DOM) returnPropCost::binary(PropCost::HI); else returnPropCost::binary(PropCost::LO); } template<classOps> ExecStatus NrootDom<Ops>::propagate(Space&home,constModEventDelta&med){ assert(!ops.even()); if((x0.min()>=0)||(x1.min()>=0)) GECODE_REWRITE(*this,(NrootPlusDom<Ops,false>::post(home,x0,x1,ops))); if((x0.max()<=0)||(x1.max()<=0)) GECODE_REWRITE(*this,(NrootPlusDom<Ops,true>::post(home,x0,x1,ops))); if(IntView::me(med)!=ME_INT_DOM){ GECODE_ES_CHECK(prop_nroot_bnd<Ops>(home,x0,x1,ops)); returnx0.assigned()&&x1.assigned()?home.ES_SUBSUMED(*this) :home.ES_NOFIX_PARTIAL(*this,IntView::med(ME_INT_DOM)); } { ViewRanges<IntView>r(x0); RangesMapNroot<Ops>rmn(ops); Iter::Ranges::Map<ViewRanges<IntView>,RangesMapNroot<Ops>,false> m(r,rmn); GECODE_ME_CHECK(x1.inter_r(home,m,false)); } { ViewRanges<IntView>r(x1); RangesMapPow<Ops>rmp(ops); Iter::Ranges::Map<ViewRanges<IntView>,RangesMapPow<Ops>,true> m(r,rmp); GECODE_ME_CHECK(x0.inter_r(home,m,false)); } returnx0.assigned()&&x1.assigned()?home.ES_SUBSUMED(*this):ES_FIX; } }}} //STATISTICS:int-prop