Annotation of OpenXM_contrib/gmp/mpn/pa64/udiv_qrnnd.c, Revision 1.1.1.1
1.1 maekawa 1: /*
2: Copyright (C) 1999, 2000 Free Software Foundation, Inc.
3:
4: This file is part of the GNU MP Library.
5:
6: The GNU MP Library is free software; you can redistribute it and/or modify
7: it under the terms of the GNU Lesser General Public License as published by
8: the Free Software Foundation; either version 2.1 of the License, or (at your
9: option) any later version.
10:
11: The GNU MP Library is distributed in the hope that it will be useful, but
12: WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
13: or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
14: License for more details.
15:
16: You should have received a copy of the GNU Lesser General Public License
17: along with the GNU MP Library; see the file COPYING.LIB. If not, write to
18: the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston,
19: MA 02111-1307, USA.
20: */
21:
22: #include "gmp.h"
23: #include "gmp-impl.h"
24: #include "longlong.h"
25:
26: #define TWO64 18446744073709551616.0
27:
28: mp_limb_t
29: #if __STDC__
30: __MPN(udiv_qrnnd) (mp_limb_t n1, mp_limb_t n0, mp_limb_t d, mp_limb_t *r)
31: #else
32: __MPN(udiv_qrnnd) (n1, n0, d, r)
33: mp_limb_t n1;
34: mp_limb_t n0;
35: mp_limb_t d;
36: mp_limb_t *r;
37: #endif
38: {
39: mp_limb_t q1, q2, q;
40: mp_limb_t p1, p0;
41: double di, dq;
42:
43: di = 1.0 / d;
44:
45: /* Generate upper 53 bits of quotient. Be careful here; the `double'
46: quotient may be rounded to 2^64 which we cannot safely convert back
47: to a 64-bit integer. */
48: dq = (TWO64 * (double) n1 + (double) n0) * di;
49: if (dq >= TWO64)
50: q1 = 0xfffffffffffff800LL;
51: else
52: q1 = (mp_limb_t) dq;
53:
54: /* Multiply back in order to compare the product to the dividend. */
55: umul_ppmm (p1, p0, q1, d);
56:
57: /* Was the 53-bit quotient greater that our sought quotient? Test the
58: sign of the partial remainder to find out. */
59: if (n1 < p1 || (n1 == p1 && n0 < p0))
60: {
61: /* 53-bit quotient too large. Partial remainder is negative.
62: Compute the absolute value of the remainder in n1,,n0. */
63: n1 = p1 - (n1 + (p0 < n0));
64: n0 = p0 - n0;
65:
66: /* Now use the partial remainder as new dividend to compute more bits of
67: quotient. This is an adjustment for the one we got previously. */
68: q2 = (mp_limb_t) ((TWO64 * (double) n1 + (double) n0) * di);
69: umul_ppmm (p1, p0, q2, d);
70:
71: q = q1 - q2;
72: if (n1 < p1 || (n1 == p1 && n0 <= p0))
73: {
74: n0 = p0 - n0;
75: }
76: else
77: {
78: n0 = p0 - n0;
79: n0 += d;
80: q--;
81: }
82: }
83: else
84: {
85: n1 = n1 - (p1 + (n0 < p0));
86: n0 = n0 - p0;
87:
88: q2 = (mp_limb_t) ((TWO64 * (double) n1 + (double) n0) * di);
89: umul_ppmm (p1, p0, q2, d);
90:
91: q = q1 + q2;
92: if (n1 < p1 || (n1 == p1 && n0 < p0))
93: {
94: n0 = n0 - p0;
95: n0 += d;
96: q--;
97: }
98: else
99: {
100: n0 = n0 - p0;
101: if (n0 >= d)
102: {
103: n0 -= d;
104: q++;
105: }
106: }
107: }
108:
109: *r = n0;
110: return q;
111: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>