1 1.1 mrg /* mpz_fdiv_qr_ui -- Division rounding the quotient towards -infinity. 2 1.1 mrg The remainder gets the same sign as the denominator. 3 1.1 mrg 4 1.1.1.4 mrg Copyright 1994-1996, 1999, 2001, 2002, 2004, 2012, 2015 Free Software 5 1.1.1.4 mrg Foundation, Inc. 6 1.1 mrg 7 1.1 mrg This file is part of the GNU MP Library. 8 1.1 mrg 9 1.1 mrg The GNU MP Library is free software; you can redistribute it and/or modify 10 1.1.1.3 mrg it under the terms of either: 11 1.1.1.3 mrg 12 1.1.1.3 mrg * the GNU Lesser General Public License as published by the Free 13 1.1.1.3 mrg Software Foundation; either version 3 of the License, or (at your 14 1.1.1.3 mrg option) any later version. 15 1.1.1.3 mrg 16 1.1.1.3 mrg or 17 1.1.1.3 mrg 18 1.1.1.3 mrg * the GNU General Public License as published by the Free Software 19 1.1.1.3 mrg Foundation; either version 2 of the License, or (at your option) any 20 1.1.1.3 mrg later version. 21 1.1.1.3 mrg 22 1.1.1.3 mrg or both in parallel, as here. 23 1.1 mrg 24 1.1 mrg The GNU MP Library is distributed in the hope that it will be useful, but 25 1.1 mrg WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY 26 1.1.1.3 mrg or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 27 1.1.1.3 mrg for more details. 28 1.1 mrg 29 1.1.1.3 mrg You should have received copies of the GNU General Public License and the 30 1.1.1.3 mrg GNU Lesser General Public License along with the GNU MP Library. If not, 31 1.1.1.3 mrg see https://www.gnu.org/licenses/. */ 32 1.1 mrg 33 1.1 mrg #include "gmp-impl.h" 34 1.1 mrg 35 1.1 mrg unsigned long int 36 1.1 mrg mpz_fdiv_qr_ui (mpz_ptr quot, mpz_ptr rem, mpz_srcptr dividend, unsigned long int divisor) 37 1.1 mrg { 38 1.1 mrg mp_size_t ns, nn, qn; 39 1.1 mrg mp_ptr np, qp; 40 1.1 mrg mp_limb_t rl; 41 1.1 mrg 42 1.1.1.2 mrg if (UNLIKELY (divisor == 0)) 43 1.1 mrg DIVIDE_BY_ZERO; 44 1.1 mrg 45 1.1 mrg ns = SIZ(dividend); 46 1.1 mrg if (ns == 0) 47 1.1 mrg { 48 1.1 mrg SIZ(quot) = 0; 49 1.1 mrg SIZ(rem) = 0; 50 1.1 mrg return 0; 51 1.1 mrg } 52 1.1 mrg 53 1.1 mrg nn = ABS(ns); 54 1.1.1.2 mrg qp = MPZ_REALLOC (quot, nn); 55 1.1 mrg np = PTR(dividend); 56 1.1 mrg 57 1.1 mrg #if BITS_PER_ULONG > GMP_NUMB_BITS /* avoid warnings about shift amount */ 58 1.1 mrg if (divisor > GMP_NUMB_MAX) 59 1.1 mrg { 60 1.1 mrg mp_limb_t dp[2]; 61 1.1 mrg mp_ptr rp; 62 1.1 mrg mp_size_t rn; 63 1.1 mrg 64 1.1.1.4 mrg rp = MPZ_REALLOC (rem, 2); 65 1.1 mrg 66 1.1 mrg if (nn == 1) /* tdiv_qr requirements; tested above for 0 */ 67 1.1 mrg { 68 1.1 mrg qp[0] = 0; 69 1.1 mrg qn = 1; /* a white lie, fixed below */ 70 1.1 mrg rl = np[0]; 71 1.1 mrg rp[0] = rl; 72 1.1 mrg } 73 1.1 mrg else 74 1.1 mrg { 75 1.1 mrg dp[0] = divisor & GMP_NUMB_MASK; 76 1.1 mrg dp[1] = divisor >> GMP_NUMB_BITS; 77 1.1 mrg mpn_tdiv_qr (qp, rp, (mp_size_t) 0, np, nn, dp, (mp_size_t) 2); 78 1.1 mrg rl = rp[0] + (rp[1] << GMP_NUMB_BITS); 79 1.1 mrg qn = nn - 2 + 1; 80 1.1 mrg } 81 1.1 mrg 82 1.1 mrg if (rl != 0 && ns < 0) 83 1.1 mrg { 84 1.1 mrg mpn_incr_u (qp, (mp_limb_t) 1); 85 1.1 mrg rl = divisor - rl; 86 1.1 mrg rp[0] = rl & GMP_NUMB_MASK; 87 1.1 mrg rp[1] = rl >> GMP_NUMB_BITS; 88 1.1 mrg } 89 1.1 mrg 90 1.1 mrg qn -= qp[qn - 1] == 0; qn -= qn != 0 && qp[qn - 1] == 0; 91 1.1 mrg rn = 1 + (rl > GMP_NUMB_MAX); rn -= (rp[rn - 1] == 0); 92 1.1 mrg SIZ(rem) = rn; 93 1.1 mrg } 94 1.1 mrg else 95 1.1 mrg #endif 96 1.1 mrg { 97 1.1 mrg rl = mpn_divrem_1 (qp, (mp_size_t) 0, np, nn, (mp_limb_t) divisor); 98 1.1 mrg if (rl == 0) 99 1.1 mrg SIZ(rem) = 0; 100 1.1 mrg else 101 1.1 mrg { 102 1.1 mrg if (ns < 0) 103 1.1 mrg { 104 1.1 mrg mpn_incr_u (qp, (mp_limb_t) 1); 105 1.1 mrg rl = divisor - rl; 106 1.1 mrg } 107 1.1 mrg 108 1.1.1.4 mrg MPZ_NEWALLOC (rem, 1)[0] = rl; 109 1.1 mrg SIZ(rem) = rl != 0; 110 1.1 mrg } 111 1.1 mrg qn = nn - (qp[nn - 1] == 0); 112 1.1 mrg } 113 1.1 mrg 114 1.1 mrg SIZ(quot) = ns >= 0 ? qn : -qn; 115 1.1 mrg return rl; 116 1.1 mrg } 117