divmod.hpp gecode/int/linear.hh /usr/include/gecode/int/arithmetic.hh Gecode Gecode::Int Gecode::Int::Arithmetic /*-*-mode:C++;c-basic-offset:2;indent-tabs-mode:nil-*-*/ /* *Mainauthors: *GuidoTack<tack@gecode.org> * *Copyright: *GuidoTack,2008 * *Lastmodified: *$Date:2013-02-1416:29:11+0100(Thu,14Feb2013)$by$Author:schulte$ *$Revision:13292$ * *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/linear.hh> namespaceGecode{namespaceInt{namespaceArithmetic{ /* *Positiveboundsconsistentdivision * */ template<classVA,classVB,classVC> forceinline DivPlusBnd<VA,VB,VC>::DivPlusBnd(Homehome,VAx0,VBx1,VCx2) :MixTernaryPropagator<VA,PC_INT_BND,VB,PC_INT_BND,VC,PC_INT_BND> (home,x0,x1,x2){} template<classVA,classVB,classVC> forceinline DivPlusBnd<VA,VB,VC>::DivPlusBnd(Space&home,boolshare, DivPlusBnd<VA,VB,VC>&p) :MixTernaryPropagator<VA,PC_INT_BND,VB,PC_INT_BND,VC,PC_INT_BND> (home,share,p){} template<classVA,classVB,classVC> Actor* DivPlusBnd<VA,VB,VC>::copy(Space&home,boolshare){ returnnew(home)DivPlusBnd<VA,VB,VC>(home,share,*this); } template<classVA,classVB,classVC> ExecStatus DivPlusBnd<VA,VB,VC>::propagate(Space&home,constModEventDelta&){ assert(pos(x0)&&pos(x1)&&!neg(x2)); boolmod; do{ mod=false; GECODE_ME_CHECK_MODIFIED(mod,x2.lq(home, floor_div_pp(x0.max(),x1.min()))); GECODE_ME_CHECK_MODIFIED(mod,x2.gq(home, floor_div_px(x0.min(),x1.max()))); GECODE_ME_CHECK_MODIFIED(mod,x0.le(home,mll(x1.max(),ill(x2.max())))); GECODE_ME_CHECK_MODIFIED(mod,x0.gq(home,mll(x1.min(),x2.min()))); if(x2.min()>0){ GECODE_ME_CHECK_MODIFIED(mod, x1.lq(home,floor_div_pp(x0.max(),x2.min()))); } GECODE_ME_CHECK_MODIFIED(mod,x1.gq(home,ceil_div_pp(ll(x0.min()), ill(x2.max())))); }while(mod); returnx0.assigned()&&x1.assigned()? home.ES_SUBSUMED(*this):ES_FIX; } template<classVA,classVB,classVC> forceinlineExecStatus DivPlusBnd<VA,VB,VC>::post(Homehome,VAx0,VBx1,VCx2){ GECODE_ME_CHECK(x0.gr(home,0)); GECODE_ME_CHECK(x1.gr(home,0)); GECODE_ME_CHECK(x2.gq(home,floor_div_pp(x0.min(),x1.max()))); (void)new(home)DivPlusBnd<VA,VB,VC>(home,x0,x1,x2); returnES_OK; } /* *Boundsconsistentdivision * */ template<classView> forceinline DivBnd<View>::DivBnd(Homehome,Viewx0,Viewx1,Viewx2) :TernaryPropagator<View,PC_INT_BND>(home,x0,x1,x2){} template<classView> forceinline DivBnd<View>::DivBnd(Space&home,boolshare,DivBnd<View>&p) :TernaryPropagator<View,PC_INT_BND>(home,share,p){} template<classView> Actor* DivBnd<View>::copy(Space&home,boolshare){ returnnew(home)DivBnd<View>(home,share,*this); } template<classView> ExecStatus DivBnd<View>::propagate(Space&home,constModEventDelta&){ if(pos(x1)){ if(pos(x2)||pos(x0))gotorewrite_ppp; if(neg(x2)||neg(x0))gotorewrite_npn; gotoprop_xpx; } if(neg(x1)){ if(neg(x2)||pos(x0))gotorewrite_pnn; if(pos(x2)||neg(x0))gotorewrite_nnp; gotoprop_xnx; } if(pos(x2)){ if(pos(x0))gotorewrite_ppp; if(neg(x0))gotorewrite_nnp; gotoprop_xxp; } if(neg(x2)){ if(pos(x0))gotorewrite_pnn; if(neg(x0))gotorewrite_npn; gotoprop_xxn; } assert(any(x1)&&any(x2)); GECODE_ME_CHECK(x0.lq(home,std::max(mll(x1.max(),ill(x2.max()))-1, mll(x1.min(),dll(x2.min()))-1))); GECODE_ME_CHECK(x0.gq(home,std::min(mll(x1.min(),ill(x2.max())), mll(x1.max(),dll(x2.min()))))); returnES_NOFIX; prop_xxp: assert(any(x0)&&any(x1)&&pos(x2)); GECODE_ME_CHECK(x0.le(home,mll(x1.max(),ill(x2.max())))); GECODE_ME_CHECK(x0.gq(home,mll(x1.min(),ill(x2.max())))); if(pos(x0))gotorewrite_ppp; if(neg(x0))gotorewrite_nnp; GECODE_ME_CHECK(x1.lq(home,floor_div_pp(x0.max(),x2.min()))); GECODE_ME_CHECK(x1.gq(home,ceil_div_xp(x0.min(),x2.min()))); if(x0.assigned()&&x1.assigned()) gotosubsumed; returnES_NOFIX; prop_xpx: assert(any(x0)&&pos(x1)&&any(x2)); GECODE_ME_CHECK(x0.le(home,mll(x1.max(),ill(x2.max())))); GECODE_ME_CHECK(x0.gq(home,mll(x1.max(),dll(x2.min())))); if(pos(x0))gotorewrite_ppp; if(neg(x0))gotorewrite_npn; GECODE_ME_CHECK(x2.lq(home,floor_div_xp(x0.max(),x1.min()))); GECODE_ME_CHECK(x2.gq(home,floor_div_xp(x0.min(),x1.min()))); if(x0.assigned()&&x1.assigned()) gotosubsumed; returnES_NOFIX; prop_xxn: assert(any(x0)&&any(x1)&&neg(x2)); GECODE_ME_CHECK(x0.lq(home,mll(x1.min(),dll(x2.min())))); GECODE_ME_CHECK(x0.gq(home,mll(x1.max(),dll(x2.min())))); if(pos(x0))gotorewrite_pnn; if(neg(x0))gotorewrite_npn; if(x2.max()!=-1) GECODE_ME_CHECK(x1.lq(home,ceil_div_xx(ll(x0.min()),ill(x2.max())))); if(x2.max()!=-1) GECODE_ME_CHECK(x1.gq(home,ceil_div_xx(ll(x0.max()),ill(x2.max())))); if(x0.assigned()&&x1.assigned()) gotosubsumed; returnES_NOFIX; prop_xnx: assert(any(x0)&&neg(x1)&&any(x2)); GECODE_ME_CHECK(x0.lq(home,mll(x1.min(),dll(x2.min())))); GECODE_ME_CHECK(x0.gq(home,mll(x1.min(),ill(x2.max())))); if(pos(x0))gotorewrite_pnn; if(neg(x0))gotorewrite_nnp; GECODE_ME_CHECK(x2.lq(home,floor_div_xx(x0.min(),x1.max()))); GECODE_ME_CHECK(x2.gq(home,floor_div_xx(x0.max(),x1.max()))); if(x0.assigned()&&x1.assigned()) gotosubsumed; returnES_NOFIX; rewrite_ppp: GECODE_REWRITE(*this,(DivPlusBnd<IntView,IntView,IntView> ::post(home(*this),x0,x1,x2))); rewrite_nnp: GECODE_REWRITE(*this,(DivPlusBnd<MinusView,MinusView,IntView> ::post(home(*this),MinusView(x0),MinusView(x1),x2))); rewrite_pnn: GECODE_REWRITE(*this,(DivPlusBnd<IntView,MinusView,MinusView> ::post(home(*this),x0,MinusView(x1),MinusView(x2)))); rewrite_npn: GECODE_REWRITE(*this,(DivPlusBnd<MinusView,IntView,MinusView> ::post(home(*this),MinusView(x0),x1,MinusView(x2)))); subsumed: assert(x0.assigned()&&x1.assigned()); intresult=std::abs(x0.val())/std::abs(x1.val()); if(x0.val()/x1.val()<0) result=-result; GECODE_ME_CHECK(x2.eq(home,result)); returnhome.ES_SUBSUMED(*this); } template<classView> ExecStatus DivBnd<View>::post(Homehome,Viewx0,Viewx1,Viewx2){ GECODE_ME_CHECK(x1.nq(home,0)); if(pos(x0)){ if(pos(x1)||pos(x2))gotopost_ppp; if(neg(x1)||neg(x2))gotopost_pnn; }elseif(neg(x0)){ if(neg(x1)||pos(x2))gotopost_nnp; if(pos(x1)||neg(x2))gotopost_npn; }elseif(pos(x1)){ if(pos(x2))gotopost_ppp; if(neg(x2))gotopost_npn; }elseif(neg(x1)){ if(pos(x2))gotopost_nnp; if(neg(x2))gotopost_pnn; } (void)new(home)DivBnd<View>(home,x0,x1,x2); returnES_OK; post_ppp: returnDivPlusBnd<IntView,IntView,IntView> ::post(home,x0,x1,x2); post_nnp: returnDivPlusBnd<MinusView,MinusView,IntView> ::post(home,MinusView(x0),MinusView(x1),x2); post_pnn: returnDivPlusBnd<IntView,MinusView,MinusView> ::post(home,x0,MinusView(x1),MinusView(x2)); post_npn: returnDivPlusBnd<MinusView,IntView,MinusView> ::post(home,MinusView(x0),x1,MinusView(x2)); } /* *Propagatorforx0!=0/\(x1!=0=>x0*x1>0)/\abs(x1)<abs(x0) * */ template<classView> forceinline DivMod<View>::DivMod(Homehome,Viewx0,Viewx1,Viewx2) :TernaryPropagator<View,PC_INT_BND>(home,x0,x1,x2){} template<classView> forceinlineExecStatus DivMod<View>::post(Homehome,Viewx0,Viewx1,Viewx2){ GECODE_ME_CHECK(x1.nq(home,0)); (void)new(home)DivMod<View>(home,x0,x1,x2); returnES_OK; } template<classView> forceinline DivMod<View>::DivMod(Space&home,boolshare,DivMod<View>&p) :TernaryPropagator<View,PC_INT_BND>(home,share,p){} template<classView> Actor* DivMod<View>::copy(Space&home,boolshare){ returnnew(home)DivMod<View>(home,share,*this); } template<classView> ExecStatus DivMod<View>::propagate(Space&home,constModEventDelta&){ boolsignIsSame; do{ signIsSame=true; //Thesignofx0andx2isthesame if(x0.min()>=0){ GECODE_ME_CHECK(x2.gq(home,0)); }elseif(x0.max()<=0){ GECODE_ME_CHECK(x2.lq(home,0)); }elseif(x2.min()>0){ GECODE_ME_CHECK(x0.gq(home,0)); }elseif(x2.max()<0){ GECODE_ME_CHECK(x0.lq(home,0)); }else{ signIsSame=false; } //abs(x2)islessthanabs(x1) intx1max=std::max(x1.max(),std::max(-x1.max(), std::max(x1.min(),-x1.min()))); GECODE_ME_CHECK(x2.le(home,x1max)); GECODE_ME_CHECK(x2.gr(home,-x1max)); intx2absmin=any(x2)?0:(pos(x2)?x2.min():-x2.max()); Iter::Ranges::Singletonsr(-x2absmin,x2absmin); GECODE_ME_CHECK(x1.minus_r(home,sr,false)); }while(!signIsSame&& (x0.min()>0||x0.max()<0||x2.min()>0||x2.max()<0)); if(signIsSame){ intx2max=std::max(x2.max(),std::max(-x2.max(), std::max(x2.min(),-x2.min()))); intx1absmin=any(x1)?0:(pos(x1)?x1.min():-x1.max()); if(x2max<x1absmin) returnhome.ES_SUBSUMED(*this); } returnES_FIX; } }}} //STATISTICS:int-prop