/* -*- mode: C++; c-basic-offset: 2; indent-tabs-mode: nil -*- */ /* * Main authors: * Christian Schulte * Vincent Barichard * * Copyright: * Christian Schulte, 2004 * Vincent Barichard, 2012 * * 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. * */ namespace Gecode { namespace Float { namespace Arithmetic { /// Test whether \a x is postive template forceinline bool pos(const View& x) { return x.min() >= 0.0; } /// Test whether \a x is negative template forceinline bool neg(const View& x) { return x.max() <= 0.0; } /// Test whether \a x is neither positive nor negative template forceinline bool any(const View& x) { return (x.min() <= 0.0) && (x.max() >= 0.0); } /* * Propagator for x * y = x * */ template forceinline MultZeroOne::MultZeroOne(Home home, View x0, View x1) : BinaryPropagator(home,x0,x1) {} template forceinline ExecStatus MultZeroOne::post(Home home, View x0, View x1) { switch (rtest_eq(x0,0.0)) { case RT_FALSE: GECODE_ME_CHECK(x1.eq(home,1.0)); break; case RT_TRUE: break; case RT_MAYBE: switch (rtest_eq(x1,1.0)) { case RT_FALSE: GECODE_ME_CHECK(x0.eq(home,0.0)); break; case RT_TRUE: break; case RT_MAYBE: (void) new (home) MultZeroOne(home,x0,x1); break; default: GECODE_NEVER; } break; default: GECODE_NEVER; } return ES_OK; } template forceinline MultZeroOne::MultZeroOne(Space& home, MultZeroOne& p) : BinaryPropagator(home,p) {} template Actor* MultZeroOne::copy(Space& home) { return new (home) MultZeroOne(home,*this); } template ExecStatus MultZeroOne::propagate(Space& home, const ModEventDelta&) { switch (rtest_eq(x0,0.0)) { case RT_FALSE: GECODE_ME_CHECK(x1.eq(home,1.0)); break; case RT_TRUE: break; case RT_MAYBE: switch (rtest_eq(x1,1.0)) { case RT_FALSE: GECODE_ME_CHECK(x0.eq(home,0.0)); break; case RT_TRUE: break; case RT_MAYBE: return ES_FIX; default: GECODE_NEVER; } break; default: GECODE_NEVER; } return home.ES_SUBSUMED(*this); } /* * Positive bounds consistent multiplication * */ template forceinline MultPlus::MultPlus(Home home, VA x0, VB x1, VC x2) : MixTernaryPropagator (home,x0,x1,x2) {} template forceinline MultPlus::MultPlus(Space& home, MultPlus& p) : MixTernaryPropagator (home,p) {} template Actor* MultPlus::copy(Space& home) { return new (home) MultPlus(home,*this); } template ExecStatus MultPlus::propagate(Space& home, const ModEventDelta&) { if (x1.min() != 0.0) GECODE_ME_CHECK(x0.eq(home,x2.val() / x1.val())); if (x0.min() != 0.0) GECODE_ME_CHECK(x1.eq(home,x2.val() / x0.val())); GECODE_ME_CHECK(x2.eq(home,x0.val() * x1.val())); if (x0.assigned() && x1.assigned() && x2.assigned()) return home.ES_SUBSUMED(*this); return ES_NOFIX; } template forceinline ExecStatus MultPlus::post(Home home, VA x0, VB x1, VC x2) { GECODE_ME_CHECK(x0.gq(home,0.0)); GECODE_ME_CHECK(x1.gq(home,0.0)); Rounding r; GECODE_ME_CHECK(x2.gq(home,r.mul_down(x0.min(),x1.min()))); (void) new (home) MultPlus(home,x0,x1,x2); return ES_OK; } /* * Bounds consistent multiplication * */ template forceinline Mult::Mult(Home home, View x0, View x1, View x2) : TernaryPropagator(home,x0,x1,x2) {} template forceinline Mult::Mult(Space& home, Mult& p) : TernaryPropagator(home,p) {} template Actor* Mult::copy(Space& home) { return new (home) Mult(home,*this); } template ExecStatus Mult::propagate(Space& home, const ModEventDelta&) { Rounding r; if (pos(x0)) { if (pos(x1) || pos(x2)) goto rewrite_ppp; if (neg(x1) || neg(x2)) goto rewrite_pnn; goto prop_pxx; } if (neg(x0)) { if (neg(x1) || pos(x2)) goto rewrite_nnp; if (pos(x1) || neg(x2)) goto rewrite_npn; goto prop_nxx; } if (pos(x1)) { if (pos(x2)) goto rewrite_ppp; if (neg(x2)) goto rewrite_npn; goto prop_xpx; } if (neg(x1)) { if (pos(x2)) goto rewrite_nnp; if (neg(x2)) goto rewrite_pnn; goto prop_xnx; } assert(any(x0) && any(x1)); GECODE_ME_CHECK(x2.lq(home,std::max(r.mul_up(x0.max(),x1.max()), r.mul_up(x0.min(),x1.min())))); GECODE_ME_CHECK(x2.gq(home,std::min(r.mul_down(x0.min(),x1.max()), r.mul_down(x0.max(),x1.min())))); if (pos(x2)) { if (r.div_up(x2.min(),x1.min()) < x0.min()) GECODE_ME_CHECK(x0.gq(home,0)); if (r.div_up(x2.min(),x0.min()) < x1.min()) GECODE_ME_CHECK(x1.gq(home,0)); } if (neg(x2)) { if (r.div_up(x2.max(),x1.max()) < x0.min()) GECODE_ME_CHECK(x0.gq(home,0)); if (r.div_up(x2.max(),x0.max()) < x1.min()) GECODE_ME_CHECK(x1.gq(home,0)); } if (x0.assigned()) { assert((x0.val() == 0.0) && (x2.val() == 0.0)); return home.ES_SUBSUMED(*this); } if (x1.assigned()) { assert((x1.val() == 0.0) && (x2.val() == 0.0)); return home.ES_SUBSUMED(*this); } return ES_NOFIX; prop_xpx: std::swap(x0,x1); prop_pxx: assert(pos(x0) && any(x1) && any(x2)); GECODE_ME_CHECK(x2.lq(home,r.mul_up(x0.max(),x1.max()))); GECODE_ME_CHECK(x2.gq(home,r.mul_down(x0.max(),x1.min()))); if (pos(x2)) goto rewrite_ppp; if (neg(x2)) goto rewrite_pnn; GECODE_ME_CHECK(x1.lq(home,r.div_up(x2.max(),x0.min()))); GECODE_ME_CHECK(x1.gq(home,r.div_down(x2.min(),x0.min()))); if (x0.assigned() && x1.assigned()) { GECODE_ME_CHECK(x2.eq(home,x0.val()*x1.val())); return home.ES_SUBSUMED(*this); } return ES_NOFIX; prop_xnx: std::swap(x0,x1); prop_nxx: assert(neg(x0) && any(x1) && any(x2)); GECODE_ME_CHECK(x2.lq(home,r.mul_up(x0.min(),x1.min()))); GECODE_ME_CHECK(x2.gq(home,r.mul_down(x0.min(),x1.max()))); if (pos(x2)) goto rewrite_nnp; if (neg(x2)) goto rewrite_npn; if (x0.max() != 0.0) { GECODE_ME_CHECK(x1.lq(home,r.div_up(x2.min(),x0.max()))); GECODE_ME_CHECK(x1.gq(home,r.div_down(x2.max(),x0.max()))); } if (x0.assigned() && x1.assigned()) { GECODE_ME_CHECK(x2.eq(home,x0.val()*x1.val())); return home.ES_SUBSUMED(*this); } return ES_NOFIX; rewrite_ppp: GECODE_REWRITE(*this,(MultPlus ::post(home(*this),x0,x1,x2))); rewrite_nnp: GECODE_REWRITE(*this,(MultPlus ::post(home(*this),MinusView(x0),MinusView(x1),x2))); rewrite_pnn: std::swap(x0,x1); rewrite_npn: GECODE_REWRITE(*this,(MultPlus ::post(home(*this),MinusView(x0),x1,MinusView(x2)))); } template ExecStatus Mult::post(Home home, View x0, View x1, View x2) { if (x0 == x1) return Sqr::post(home,x0,x2); if (x0 == x2) return MultZeroOne::post(home,x0,x1); if (x1 == x2) return MultZeroOne::post(home,x1,x0); if (pos(x0)) { if (pos(x1) || pos(x2)) goto post_ppp; if (neg(x1) || neg(x2)) goto post_pnn; } else if (neg(x0)) { if (neg(x1) || pos(x2)) goto post_nnp; if (pos(x1) || neg(x2)) goto post_npn; } else if (pos(x1)) { if (pos(x2)) goto post_ppp; if (neg(x2)) goto post_npn; } else if (neg(x1)) { if (pos(x2)) goto post_nnp; if (neg(x2)) goto post_pnn; } { GECODE_ME_CHECK(x2.eq(home,x0.val()*x1.val())); (void) new (home) Mult(home,x0,x1,x2); } return ES_OK; post_ppp: return MultPlus::post(home,x0,x1,x2); post_nnp: return MultPlus::post(home, MinusView(x0),MinusView(x1),x2); post_pnn: std::swap(x0,x1); post_npn: return MultPlus::post(home, MinusView(x0),x1,MinusView(x2)); } }}} // STATISTICS: float-prop