xref: /haiku/src/system/libroot/posix/musl/math/remquo.c (revision 1deede7388b04dbeec5af85cae7164735ea9e70d)
1 #include <math.h>
2 #include <stdint.h>
3 
4 double remquo(double x, double y, int *quo)
5 {
6 	union {double f; uint64_t i;} ux = {x}, uy = {y};
7 	int ex = ux.i>>52 & 0x7ff;
8 	int ey = uy.i>>52 & 0x7ff;
9 	int sx = ux.i>>63;
10 	int sy = uy.i>>63;
11 	uint32_t q;
12 	uint64_t i;
13 	uint64_t uxi = ux.i;
14 
15 	*quo = 0;
16 	if (uy.i<<1 == 0 || isnan(y) || ex == 0x7ff)
17 		return (x*y)/(x*y);
18 	if (ux.i<<1 == 0)
19 		return x;
20 
21 	/* normalize x and y */
22 	if (!ex) {
23 		for (i = uxi<<12; i>>63 == 0; ex--, i <<= 1);
24 		uxi <<= -ex + 1;
25 	} else {
26 		uxi &= -1ULL >> 12;
27 		uxi |= 1ULL << 52;
28 	}
29 	if (!ey) {
30 		for (i = uy.i<<12; i>>63 == 0; ey--, i <<= 1);
31 		uy.i <<= -ey + 1;
32 	} else {
33 		uy.i &= -1ULL >> 12;
34 		uy.i |= 1ULL << 52;
35 	}
36 
37 	q = 0;
38 	if (ex < ey) {
39 		if (ex+1 == ey)
40 			goto end;
41 		return x;
42 	}
43 
44 	/* x mod y */
45 	for (; ex > ey; ex--) {
46 		i = uxi - uy.i;
47 		if (i >> 63 == 0) {
48 			uxi = i;
49 			q++;
50 		}
51 		uxi <<= 1;
52 		q <<= 1;
53 	}
54 	i = uxi - uy.i;
55 	if (i >> 63 == 0) {
56 		uxi = i;
57 		q++;
58 	}
59 	if (uxi == 0)
60 		ex = -60;
61 	else
62 		for (; uxi>>52 == 0; uxi <<= 1, ex--);
63 end:
64 	/* scale result and decide between |x| and |x|-|y| */
65 	if (ex > 0) {
66 		uxi -= 1ULL << 52;
67 		uxi |= (uint64_t)ex << 52;
68 	} else {
69 		uxi >>= -ex + 1;
70 	}
71 	ux.i = uxi;
72 	x = ux.f;
73 	if (sy)
74 		y = -y;
75 	if (ex == ey || (ex+1 == ey && (2*x > y || (2*x == y && q%2)))) {
76 		x -= y;
77 		q++;
78 	}
79 	q &= 0x7fffffff;
80 	*quo = sx^sy ? -(int)q : (int)q;
81 	return sx ? -x : x;
82 }
83