155#ifdef WORDS_BIGENDIAN
156#define IEEE_BIG_ENDIAN
158#define IEEE_LITTLE_ENDIAN
163#undef IEEE_BIG_ENDIAN
164#undef IEEE_LITTLE_ENDIAN
167#if defined(__arm__) && !defined(__VFP_FP__)
168#define IEEE_BIG_ENDIAN
169#undef IEEE_LITTLE_ENDIAN
180#if (INT_MAX >> 30) && !(INT_MAX >> 31)
182#define ULong unsigned int
183#elif (LONG_MAX >> 30) && !(LONG_MAX >> 31)
185#define ULong unsigned long int
187#error No 32bit integer
190#if defined(HAVE_LONG_LONG) && (HAVE_LONG_LONG)
191#define Llong LONG_LONG
198#define Bug(x) {fprintf(stderr, "%s\n", (x)); exit(EXIT_FAILURE);}
203#define ISDIGIT(c) isdigit(c)
213#if defined(HAVE_STDCKDINT_H) || !defined(__has_include)
214#elif __has_include(<stdckdint.h>)
215# define HAVE_STDCKDINT_H 1
217#ifdef HAVE_STDCKDINT_H
218# include <stdckdint.h>
223ckd_add(
int *result,
int x,
int y)
226 if (y < INT_MIN - x)
return 1;
229 if (y > INT_MAX - x)
return 1;
237extern void *MALLOC(
size_t);
242extern void FREE(
void*);
247#define NO_SANITIZE(x, y) y
251#undef Avoid_Underflow
252#ifdef IEEE_BIG_ENDIAN
255#ifdef IEEE_LITTLE_ENDIAN
263#define DBL_MAX_10_EXP 308
264#define DBL_MAX_EXP 1024
270#define DBL_MAX_10_EXP 75
271#define DBL_MAX_EXP 63
273#define DBL_MAX 7.2370055773322621e+75
278#define DBL_MAX_10_EXP 38
279#define DBL_MAX_EXP 127
281#define DBL_MAX 1.7014118346046923e+38
285#define LONG_MAX 2147483647
302static const char hexdigit[] =
"0123456789abcdef0123456789ABCDEF";
305#if defined(IEEE_LITTLE_ENDIAN) + defined(IEEE_BIG_ENDIAN) + defined(VAX) + defined(IBM) != 1
306Exactly one of IEEE_LITTLE_ENDIAN, IEEE_BIG_ENDIAN, VAX, or IBM should be defined.
309typedef union {
double d; ULong L[2]; }
U;
314# ifdef IEEE_LITTLE_ENDIAN
315# define word0(x) (((ULong *)&(x))[1])
316# define word1(x) (((ULong *)&(x))[0])
318# define word0(x) (((ULong *)&(x))[0])
319# define word1(x) (((ULong *)&(x))[1])
323# ifdef IEEE_LITTLE_ENDIAN
324# define word0(x) ((x).L[1])
325# define word1(x) ((x).L[0])
327# define word0(x) ((x).L[0])
328# define word1(x) ((x).L[1])
330# define dval(x) ((x).d)
337#if defined(IEEE_LITTLE_ENDIAN) + defined(VAX) + defined(__arm__)
338#define Storeinc(a,b,c) (((unsigned short *)(a))[1] = (unsigned short)(b), \
339((unsigned short *)(a))[0] = (unsigned short)(c), (a)++)
341#define Storeinc(a,b,c) (((unsigned short *)(a))[0] = (unsigned short)(b), \
342((unsigned short *)(a))[1] = (unsigned short)(c), (a)++)
354#define Exp_msk1 0x100000
355#define Exp_msk11 0x100000
356#define Exp_mask 0x7ff00000
360#define Exp_1 0x3ff00000
361#define Exp_11 0x3ff00000
363#define Frac_mask 0xfffff
364#define Frac_mask1 0xfffff
367#define Bndry_mask 0xfffff
368#define Bndry_mask1 0xfffff
370#define Sign_bit 0x80000000
377#define Avoid_Underflow
379#undef Sudden_Underflow
385#define Flt_Rounds FLT_ROUNDS
391#ifdef Honor_FLT_ROUNDS
392#define Rounding rounding
393#undef Check_FLT_ROUNDS
394#define Check_FLT_ROUNDS
396#define Rounding Flt_Rounds
400#undef Check_FLT_ROUNDS
401#undef Honor_FLT_ROUNDS
403#undef Sudden_Underflow
404#define Sudden_Underflow
410#define Exp_msk1 0x1000000
411#define Exp_msk11 0x1000000
412#define Exp_mask 0x7f000000
415#define Exp_1 0x41000000
416#define Exp_11 0x41000000
418#define Frac_mask 0xffffff
419#define Frac_mask1 0xffffff
422#define Bndry_mask 0xefffff
423#define Bndry_mask1 0xffffff
425#define Sign_bit 0x80000000
427#define Tiny0 0x100000
437#define Exp_msk11 0x800000
438#define Exp_mask 0x7f80
441#define Exp_1 0x40800000
444#define Frac_mask 0x7fffff
445#define Frac_mask1 0xffff007f
448#define Bndry_mask 0xffff007f
449#define Bndry_mask1 0xffff007f
451#define Sign_bit 0x8000
465#define rounded_product(a,b) ((a) = rnd_prod((a), (b)))
466#define rounded_quotient(a,b) ((a) = rnd_quot((a), (b)))
467extern double rnd_prod(
double,
double), rnd_quot(
double,
double);
469#define rounded_product(a,b) ((a) *= (b))
470#define rounded_quotient(a,b) ((a) /= (b))
473#define Big0 (Frac_mask1 | Exp_msk1*(DBL_MAX_EXP+Bias-1))
474#define Big1 0xffffffff
480#define FFFFFFFF 0xffffffffUL
494#define Llong long long
497#define ULLong unsigned Llong
501#define MULTIPLE_THREADS 1
503#ifndef MULTIPLE_THREADS
504#define ACQUIRE_DTOA_LOCK(n)
505#define FREE_DTOA_LOCK(n)
507#define ACQUIRE_DTOA_LOCK(n)
508#define FREE_DTOA_LOCK(n)
511#ifndef ATOMIC_PTR_CAS
512#define ATOMIC_PTR_CAS(var, old, new) ((var) = (new), (void *)(old))
518#define UNLIKELY(x) (x)
521#define ASSUME(x) (void)(x)
528 int k, maxwds, sign, wds;
541 rv = (
Bigint *)MALLOC(
sizeof(
Bigint) + (x-1)*
sizeof(ULong));
542 if (!rv)
return NULL;
545 rv->sign = rv->wds = 0;
556#define Bfree(v) Bclear(&(v))
558#define Bcopy(x,y) memcpy((char *)&(x)->sign, (char *)&(y)->sign, \
559(y)->wds*sizeof(Long) + 2*sizeof(int))
562multadd(
Bigint *b,
int m,
int a)
582 y = *x * (ULLong)m + carry;
584 *x++ = (ULong)(y & FFFFFFFF);
588 y = (xi & 0xffff) * m + carry;
589 z = (xi >> 16) * m + (y >> 16);
591 *x++ = (z << 16) + (y & 0xffff);
600 if (wds >= b->maxwds) {
610 b->x[wds++] = (ULong)carry;
617s2b(
const char *s,
int nd0,
int nd, ULong y9)
624 for (k = 0, y = 1; x > y; y <<= 1, k++) ;
633 b->x[0] = y9 & 0xffff;
634 b->wds = (b->x[1] = y9 >> 16) ? 2 : 1;
641 b = multadd(b, 10, *s++ -
'0');
648 for (; i < nd; i++) {
649 b = multadd(b, 10, *s++ -
'0');
656hi0bits(
register ULong x)
660 if (!(x & 0xffff0000)) {
664 if (!(x & 0xff000000)) {
668 if (!(x & 0xf0000000)) {
672 if (!(x & 0xc0000000)) {
676 if (!(x & 0x80000000)) {
678 if (!(x & 0x40000000))
688 register ULong x = *y;
739#define Bzero_p(b) (!(b)->x[0] && (b)->wds <= 1)
746 ULong *x, *xa, *xae, *xb, *xbe, *xc, *xc0;
757 if (Bzero_p(a) || Bzero_p(b)) {
765 if (a->wds < b->wds) {
778 for (x = c->x, xa = x + wc; x < xa; x++)
786 for (; xb < xbe; xc0++) {
787 if ((y = *xb++) != 0) {
792 z = *x++ * (ULLong)y + *xc + carry;
794 *xc++ = (ULong)(z & FFFFFFFF);
801 for (; xb < xbe; xb++, xc0++) {
802 if ((y = *xb & 0xffff) != 0) {
807 z = (*x & 0xffff) * y + (*xc & 0xffff) + carry;
809 z2 = (*x++ >> 16) * y + (*xc >> 16) + carry;
815 if ((y = *xb >> 16) != 0) {
821 z = (*x & 0xffff) * y + (*xc >> 16) + carry;
824 z2 = (*x++ >> 16) * y + (*xc & 0xffff) + carry;
831 for (; xb < xbe; xc0++) {
837 z = *x++ * y + *xc + carry;
846 for (xc0 = c->x, xc = xc0 + wc; wc > 0 && !*--xc; --wc) ;
858 static const int p05[3] = { 5, 25, 125 };
860 if ((i = k & 3) != 0) {
861 b = multadd(b, p05[i-1], 0);
865#define b_cache(var, addr, new_expr) \
866 if ((var = addr) != 0) {} else { \
868 ACQUIRE_DTOA_LOCK(1); \
869 if (!(var = addr) && (var = (new_expr)) != 0) { \
871 tmp = ATOMIC_PTR_CAS(addr, NULL, var); \
874 if (UNLIKELY(tmp)) { \
887 b_cache(p5, p5s, i2b(625));
897 b_cache(p51, p5->next, mult(p5, p5));
908 ULong *x, *x1, *xe, z;
910 if (!k || Bzero_p(b))
return b;
919 for (i = b->maxwds; n1 > i; i <<= 1)
927 for (i = 0; i < n; i++)
947 *x1++ = *x << k & 0xffff | z;
966 ULong *xa, *xa0, *xb, *xb0;
972 if (i > 1 && !a->x[i-1])
973 Bug(
"cmp called with a->x[a->wds-1] == 0");
974 if (j > 1 && !b->x[j-1])
975 Bug(
"cmp called with b->x[b->wds-1] == 0");
985 return *xa < *xb ? -1 : 1;
998 ULong *xa, *xae, *xb, *xbe, *xc;
1011 if (!c)
return NULL;
1025 if (!c)
return NULL;
1037 y = (ULLong)*xa++ - *xb++ - borrow;
1038 borrow = y >> 32 & (ULong)1;
1039 *xc++ = (ULong)(y & FFFFFFFF);
1043 borrow = y >> 32 & (ULong)1;
1044 *xc++ = (ULong)(y & FFFFFFFF);
1049 y = (*xa & 0xffff) - (*xb & 0xffff) - borrow;
1050 borrow = (y & 0x10000) >> 16;
1051 z = (*xa++ >> 16) - (*xb++ >> 16) - borrow;
1052 borrow = (z & 0x10000) >> 16;
1056 y = (*xa & 0xffff) - borrow;
1057 borrow = (y & 0x10000) >> 16;
1058 z = (*xa++ >> 16) - borrow;
1059 borrow = (z & 0x10000) >> 16;
1064 y = *xa++ - *xb++ - borrow;
1065 borrow = (y & 0x10000) >> 16;
1070 borrow = (y & 0x10000) >> 16;
1088 L = (word0(x) & Exp_mask) - (P-1)*Exp_msk1;
1089#ifndef Avoid_Underflow
1090#ifndef Sudden_Underflow
1099#ifndef Avoid_Underflow
1100#ifndef Sudden_Underflow
1103 L = -L >> Exp_shift;
1104 if (L < Exp_shift) {
1105 word0(a) = 0x80000 >> L;
1111 word1(a) = L >= 31 ? 1 : 1 << 31 - L;
1122 ULong *xa, *xa0, w, y, z;
1136 if (!y) Bug(
"zero y in b2d");
1142 d0 = Exp_1 | y >> (Ebits - k);
1143 w = xa > xa0 ? *--xa : 0;
1144 d1 = y << ((32-Ebits) + k) | w >> (Ebits - k);
1147 z = xa > xa0 ? *--xa : 0;
1149 d0 = Exp_1 | y << k | z >> (32 - k);
1150 y = xa > xa0 ? *--xa : 0;
1151 d1 = z << k | y >> (32 - k);
1158 if (k < Ebits + 16) {
1159 z = xa > xa0 ? *--xa : 0;
1160 d0 = Exp_1 | y << k - Ebits | z >> Ebits + 16 - k;
1161 w = xa > xa0 ? *--xa : 0;
1162 y = xa > xa0 ? *--xa : 0;
1163 d1 = z << k + 16 - Ebits | w << k - Ebits | y >> 16 + Ebits - k;
1166 z = xa > xa0 ? *--xa : 0;
1167 w = xa > xa0 ? *--xa : 0;
1169 d0 = Exp_1 | y << k + 16 | z << k | w >> 16 - k;
1170 y = xa > xa0 ? *--xa : 0;
1171 d1 = w << k + 16 | y << k;
1175 word0(d) = d0 >> 16 | d0 << 16;
1176 word1(d) = d1 >> 16 | d1 << 16;
1185d2b(
double d_,
int *e,
int *bits)
1191#ifndef Sudden_Underflow
1199 d0 = word0(d) >> 16 | word0(d) << 16;
1200 d1 = word1(d) >> 16 | word1(d) << 16;
1211 if (!b)
return NULL;
1216#ifdef Sudden_Underflow
1217 de = (int)(d0 >> Exp_shift);
1222 if ((de = (
int)(d0 >> Exp_shift)) != 0)
1226 if ((y = d1) != 0) {
1227 if ((k = lo0bits(&y)) != 0) {
1228 x[0] = y | z << (32 - k);
1233#ifndef Sudden_Underflow
1236 b->wds = (x[1] = z) ? 2 : 1;
1241 Bug(
"Zero passed to d2b");
1245#ifndef Sudden_Underflow
1253 if (k = lo0bits(&y))
1255 x[0] = y | z << 32 - k & 0xffff;
1256 x[1] = z >> k - 16 & 0xffff;
1262 x[1] = y >> 16 | z << 16 - k & 0xffff;
1263 x[2] = z >> k & 0xffff;
1278 Bug(
"Zero passed to d2b");
1296#ifndef Sudden_Underflow
1300 *e = (de - Bias - (P-1) << 2) + k;
1301 *bits = 4*P + 8 - k - hi0bits(word0(d) & Frac_mask);
1303 *e = de - Bias - (P-1) + k;
1306#ifndef Sudden_Underflow
1309 *e = de - Bias - (P-1) + 1 + k;
1311 *bits = 32*i - hi0bits(x[i-1]);
1313 *bits = (i+2)*16 - hi0bits(x[i]);
1328 dval(da) = b2d(a, &ka);
1329 dval(db) = b2d(b, &kb);
1331 k = ka - kb + 32*(a->wds - b->wds);
1333 k = ka - kb + 16*(a->wds - b->wds);
1337 word0(da) += (k >> 2)*Exp_msk1;
1343 word0(db) += (k >> 2)*Exp_msk1;
1349 word0(da) += k*Exp_msk1;
1352 word0(db) += k*Exp_msk1;
1355 return dval(da) / dval(db);
1360 1e0, 1e1, 1e2, 1e3, 1e4, 1e5, 1e6, 1e7, 1e8, 1e9,
1361 1e10, 1e11, 1e12, 1e13, 1e14, 1e15, 1e16, 1e17, 1e18, 1e19,
1370bigtens[] = { 1e16, 1e32, 1e64, 1e128, 1e256 };
1371static const double tinytens[] = { 1e-16, 1e-32, 1e-64, 1e-128,
1372#ifdef Avoid_Underflow
1373 9007199254740992.*9007199254740992.e-256
1381#define Scale_Bit 0x10
1385bigtens[] = { 1e16, 1e32, 1e64 };
1386static const double tinytens[] = { 1e-16, 1e-32, 1e-64 };
1389bigtens[] = { 1e16, 1e32 };
1390static const double tinytens[] = { 1e-16, 1e-32 };
1402#define NAN_WORD0 0x7ff80000
1410match(
const char **sp,
char *t)
1413 const char *s = *sp;
1416 if ((c = *++s) >=
'A' && c <=
'Z')
1427hexnan(
double *rvp,
const char **sp)
1431 int havedig, udx0, xshift;
1434 havedig = xshift = 0;
1437 while (c = *(
const unsigned char*)++s) {
1438 if (c >=
'0' && c <=
'9')
1440 else if (c >=
'a' && c <=
'f')
1442 else if (c >=
'A' && c <=
'F')
1444 else if (c <=
' ') {
1445 if (udx0 && havedig) {
1451 else if ( c ==
')' && havedig) {
1464 x[0] = (x[0] << 4) | (x[1] >> 28);
1465 x[1] = (x[1] << 4) | c;
1467 if ((x[0] &= 0xfffff) || x[1]) {
1468 word0(*rvp) = Exp_mask | x[0];
1475NO_SANITIZE(
"unsigned-integer-overflow",
double strtod(
const char *s00,
char **se));
1477strtod(
const char *s00,
char **se)
1479#ifdef Avoid_Underflow
1482 int bb2, bb5, bbe, bd2, bd5, bbbits, bs2, c, dsign,
1483 e, e1, esign, i, j, k, nd, nd0, nf, nz, nz0, sign;
1484 const char *s, *s0, *s1;
1489 Bigint *bb, *bb1, *bd, *bd0, *bs, *delta;
1491 int inexact, oldinexact;
1493#ifdef Honor_FLT_ROUNDS
1501 sign = nz0 = nz = 0;
1526 if (s[1] ==
'x' || s[1] ==
'X') {
1532 if (!*++s || (!(s1 = strchr(hexdigit, *s)) && *s !=
'.'))
goto ret0;
1534 while (*++s ==
'0');
1536 s1 = strchr(hexdigit, *s);
1540 adj += aadj * ((s1 - hexdigit) & 15);
1543 }
while (*++s && (s1 = strchr(hexdigit, *s)));
1546 if ((*s ==
'.') && *++s && (s1 = strchr(hexdigit, *s))) {
1553 for (; *s && (s1 = strchr(hexdigit, *s)); ++s) {
1554 adj += aadj * ((s1 - hexdigit) & 15);
1555 if ((aadj /= 16) == 0.0) {
1556 while (*++s && strchr(hexdigit, *s));
1562 if (*s ==
'P' || *s ==
'p') {
1563 dsign = 0x2C - *++s;
1564 if (abs(dsign) == 1) s++;
1569 if (c <
'0' ||
'9' < c)
goto ret0;
1576 if (nd + dsign * nd0 > 2095) {
1577 while (
'0' <= c && c <=
'9') c = *++s;
1580 }
while (
'0' <= c && c <=
'9');
1583 dval(rv) = ldexp(adj, nd0);
1587 while (*++s ==
'0') ;
1593 for (nd = nf = 0; (c = *s) >=
'0' && c <=
'9'; nd++, s++)
1596 else if (nd < DBL_DIG + 2)
1600 s1 = localeconv()->decimal_point;
1623 for (; c ==
'0'; c = *++s)
1625 if (c >
'0' && c <=
'9') {
1633 for (; c >=
'0' && c <=
'9'; c = *++s) {
1636 if (nd > DBL_DIG * 4) {
1641 for (i = 1; i < nz; i++)
1644 else if (nd <= DBL_DIG + 2)
1648 else if (nd <= DBL_DIG + 2)
1656 if (c ==
'e' || c ==
'E') {
1657 if (!nd && !nz && !nz0) {
1668 if (c >=
'0' && c <=
'9') {
1671 if (c >
'0' && c <=
'9') {
1674 while ((c = *++s) >=
'0' && c <=
'9')
1676 if (s - s1 > 8 || L > 19999)
1699 if (match(&s,
"nf")) {
1701 if (!match(&s,
"inity"))
1703 word0(rv) = 0x7ff00000;
1710 if (match(&s,
"an")) {
1711 word0(rv) = NAN_WORD0;
1712 word1(rv) = NAN_WORD1;
1736 k = nd < DBL_DIG + 2 ? nd : DBL_DIG + 2;
1741 oldinexact = get_inexact();
1743 dval(rv) = tens[k - 9] * dval(rv) + z;
1745 bd0 = bb = bd = bs = delta = 0;
1748#ifndef Honor_FLT_ROUNDS
1756 if (e <= Ten_pmax) {
1758 goto vax_ovfl_check;
1760#ifdef Honor_FLT_ROUNDS
1763 dval(rv) = -dval(rv);
1767 rounded_product(dval(rv), tens[e]);
1772 if (e <= Ten_pmax + i) {
1776#ifdef Honor_FLT_ROUNDS
1779 dval(rv) = -dval(rv);
1784 dval(rv) *= tens[i];
1790 word0(rv) -= P*Exp_msk1;
1791 rounded_product(dval(rv), tens[e]);
1792 if ((word0(rv) & Exp_mask)
1793 > Exp_msk1*(DBL_MAX_EXP+Bias-1-P))
1795 word0(rv) += P*Exp_msk1;
1797 rounded_product(dval(rv), tens[e]);
1802#ifndef Inaccurate_Divide
1803 else if (e >= -Ten_pmax) {
1804#ifdef Honor_FLT_ROUNDS
1807 dval(rv) = -dval(rv);
1811 rounded_quotient(dval(rv), tens[-e]);
1822 oldinexact = get_inexact();
1824#ifdef Avoid_Underflow
1827#ifdef Honor_FLT_ROUNDS
1828 if ((rounding = Flt_Rounds) >= 2) {
1830 rounding = rounding == 2 ? 0 : 2;
1841 if ((i = e1 & 15) != 0)
1842 dval(rv) *= tens[i];
1844 if (e1 > DBL_MAX_10_EXP) {
1851#ifdef Honor_FLT_ROUNDS
1859 word0(rv) = Exp_mask;
1863 word0(rv) = Exp_mask;
1869 dval(rv0) *= dval(rv0);
1880 for (j = 0; e1 > 1; j++, e1 >>= 1)
1882 dval(rv) *= bigtens[j];
1884 word0(rv) -= P*Exp_msk1;
1885 dval(rv) *= bigtens[j];
1886 if ((z = word0(rv) & Exp_mask)
1887 > Exp_msk1*(DBL_MAX_EXP+Bias-P))
1889 if (z > Exp_msk1*(DBL_MAX_EXP+Bias-1-P)) {
1896 word0(rv) += P*Exp_msk1;
1901 if ((i = e1 & 15) != 0)
1902 dval(rv) /= tens[i];
1904 if (e1 >= 1 << n_bigtens)
1906#ifdef Avoid_Underflow
1909 for (j = 0; e1 > 0; j++, e1 >>= 1)
1911 dval(rv) *= tinytens[j];
1912 if (scale && (j = 2*P + 1 - ((word0(rv) & Exp_mask)
1913 >> Exp_shift)) > 0) {
1918 word0(rv) = (P+2)*Exp_msk1;
1920 word0(rv) &= 0xffffffff << (j-32);
1923 word1(rv) &= 0xffffffff << j;
1926 for (j = 0; e1 > 1; j++, e1 >>= 1)
1928 dval(rv) *= tinytens[j];
1930 dval(rv0) = dval(rv);
1931 dval(rv) *= tinytens[j];
1933 dval(rv) = 2.*dval(rv0);
1934 dval(rv) *= tinytens[j];
1946#ifndef Avoid_Underflow
1961 bd0 = s2b(s0, nd0, nd, y);
1965 bd = Balloc(bd0->k);
1966 if (!bd)
goto retfree;
1968 bb = d2b(dval(rv), &bbe, &bbbits);
1969 if (!bb)
goto retfree;
1971 if (!bs)
goto retfree;
1986#ifdef Honor_FLT_ROUNDS
1990#ifdef Avoid_Underflow
1998#ifdef Sudden_Underflow
2000 j = 1 + 4*P - 3 - bbbits + ((bbe + bbbits - 1) & 3);
2015#ifdef Avoid_Underflow
2018 i = bb2 < bd2 ? bb2 : bd2;
2027 bs = pow5mult(bs, bb5);
2028 if (!bs)
goto retfree;
2032 if (!bb)
goto retfree;
2035 bb = lshift(bb, bb2);
2036 if (!bb)
goto retfree;
2039 bd = pow5mult(bd, bd5);
2040 if (!bd)
goto retfree;
2043 bd = lshift(bd, bd2);
2044 if (!bd)
goto retfree;
2047 bs = lshift(bs, bs2);
2048 if (!bs)
goto retfree;
2050 delta = diff(bb, bd);
2051 if (!delta)
goto retfree;
2052 dsign = delta->sign;
2055#ifdef Honor_FLT_ROUNDS
2056 if (rounding != 1) {
2059 if (!delta->x[0] && delta->wds <= 1) {
2075 && !(word0(rv) & Frac_mask)) {
2076 y = word0(rv) & Exp_mask;
2077#ifdef Avoid_Underflow
2078 if (!scale || y > 2*P*Exp_msk1)
2083 delta = lshift(delta,Log2P);
2084 if (!delta)
goto nomem;
2085 if (cmp(delta, bs) <= 0)
2090#ifdef Avoid_Underflow
2091 if (scale && (y = word0(rv) & Exp_mask)
2093 word0(adj) += (2*P+1)*Exp_msk1 - y;
2095#ifdef Sudden_Underflow
2096 if ((word0(rv) & Exp_mask) <=
2098 word0(rv) += P*Exp_msk1;
2099 dval(rv) += adj*ulp(dval(rv));
2100 word0(rv) -= P*Exp_msk1;
2105 dval(rv) += adj*ulp(dval(rv));
2109 adj = ratio(delta, bs);
2112 if (adj <= 0x7ffffffe) {
2116 if (!((rounding>>1) ^ dsign))
2121#ifdef Avoid_Underflow
2122 if (scale && (y = word0(rv) & Exp_mask) <= 2*P*Exp_msk1)
2123 word0(adj) += (2*P+1)*Exp_msk1 - y;
2125#ifdef Sudden_Underflow
2126 if ((word0(rv) & Exp_mask) <= P*Exp_msk1) {
2127 word0(rv) += P*Exp_msk1;
2128 adj *= ulp(dval(rv));
2133 word0(rv) -= P*Exp_msk1;
2138 adj *= ulp(dval(rv));
2151 if (dsign || word1(rv) || word0(rv) & Bndry_mask
2153#ifdef Avoid_Underflow
2154 || (word0(rv) & Exp_mask) <= (2*P+1)*Exp_msk1
2156 || (word0(rv) & Exp_mask) <= Exp_msk1
2161 if (!delta->x[0] && delta->wds <= 1)
2166 if (!delta->x[0] && delta->wds <= 1) {
2173 delta = lshift(delta,Log2P);
2174 if (!delta)
goto retfree;
2175 if (cmp(delta, bs) > 0)
2182 if ((word0(rv) & Bndry_mask1) == Bndry_mask1
2184#ifdef Avoid_Underflow
2185 (scale && (y = word0(rv) & Exp_mask) <= 2*P*Exp_msk1)
2186 ? (0xffffffff & (0xffffffff << (2*P+1-(y>>Exp_shift)))) :
2190 word0(rv) = (word0(rv) & Exp_mask)
2197#ifdef Avoid_Underflow
2203 else if (!(word0(rv) & Bndry_mask) && !word1(rv)) {
2206#ifdef Sudden_Underflow
2207 L = word0(rv) & Exp_mask;
2211#ifdef Avoid_Underflow
2212 if (L <= (scale ? (2*P+1)*Exp_msk1 : Exp_msk1))
2220#ifdef Avoid_Underflow
2222 L = word0(rv) & Exp_mask;
2223 if (L <= (2*P+1)*Exp_msk1) {
2224 if (L > (P+2)*Exp_msk1)
2233 L = (word0(rv) & Exp_mask) - Exp_msk1;
2235 word0(rv) = L | Bndry_mask1;
2236 word1(rv) = 0xffffffff;
2244 if (!(word1(rv) & LSB))
2248 dval(rv) += ulp(dval(rv));
2251 dval(rv) -= ulp(dval(rv));
2252#ifndef Sudden_Underflow
2257#ifdef Avoid_Underflow
2263 if ((aadj = ratio(delta, bs)) <= 2.) {
2265 aadj = dval(aadj1) = 1.;
2266 else if (word1(rv) || word0(rv) & Bndry_mask) {
2267#ifndef Sudden_Underflow
2268 if (word1(rv) == Tiny1 && !word0(rv))
2278 if (aadj < 2./FLT_RADIX)
2279 aadj = 1./FLT_RADIX;
2282 dval(aadj1) = -aadj;
2287 dval(aadj1) = dsign ? aadj : -aadj;
2288#ifdef Check_FLT_ROUNDS
2298 if (Flt_Rounds == 0)
2302 y = word0(rv) & Exp_mask;
2306 if (y == Exp_msk1*(DBL_MAX_EXP+Bias-1)) {
2307 dval(rv0) = dval(rv);
2308 word0(rv) -= P*Exp_msk1;
2309 adj = dval(aadj1) * ulp(dval(rv));
2311 if ((word0(rv) & Exp_mask) >=
2312 Exp_msk1*(DBL_MAX_EXP+Bias-P)) {
2313 if (word0(rv0) == Big0 && word1(rv0) == Big1)
2320 word0(rv) += P*Exp_msk1;
2323#ifdef Avoid_Underflow
2324 if (scale && y <= 2*P*Exp_msk1) {
2325 if (aadj <= 0x7fffffff) {
2326 if ((z = (
int)aadj) <= 0)
2329 dval(aadj1) = dsign ? aadj : -aadj;
2331 word0(aadj1) += (2*P+1)*Exp_msk1 - y;
2333 adj = dval(aadj1) * ulp(dval(rv));
2336#ifdef Sudden_Underflow
2337 if ((word0(rv) & Exp_mask) <= P*Exp_msk1) {
2338 dval(rv0) = dval(rv);
2339 word0(rv) += P*Exp_msk1;
2340 adj = dval(aadj1) * ulp(dval(rv));
2343 if ((word0(rv) & Exp_mask) < P*Exp_msk1)
2345 if ((word0(rv) & Exp_mask) <= P*Exp_msk1)
2348 if (word0(rv0) == Tiny0 && word1(rv0) == Tiny1)
2355 word0(rv) -= P*Exp_msk1;
2358 adj = dval(aadj1) * ulp(dval(rv));
2369 if (y <= (P-1)*Exp_msk1 && aadj > 1.) {
2370 dval(aadj1) = (double)(
int)(aadj + 0.5);
2372 dval(aadj1) = -dval(aadj1);
2374 adj = dval(aadj1) * ulp(dval(rv));
2379 z = word0(rv) & Exp_mask;
2381#ifdef Avoid_Underflow
2389 if (dsign || word1(rv) || word0(rv) & Bndry_mask) {
2390 if (aadj < .4999999 || aadj > .5000001)
2393 else if (aadj < .4999999/FLT_RADIX)
2406 word0(rv0) = Exp_1 + (70 << Exp_shift);
2411 else if (!oldinexact)
2414#ifdef Avoid_Underflow
2416 word0(rv0) = Exp_1 - 2*P*Exp_msk1;
2418 dval(rv) *= dval(rv0);
2421 if (word0(rv) == 0 && word1(rv) == 0)
2427 if (inexact && !(word0(rv) & Exp_mask)) {
2430 dval(rv0) *= dval(rv0);
2442 return sign ? -dval(rv) : dval(rv);
2445NO_SANITIZE(
"unsigned-integer-overflow",
static int quorem(
Bigint *b,
Bigint *S));
2450 ULong *bx, *bxe, q, *sx, *sxe;
2452 ULLong borrow, carry, y, ys;
2454 ULong borrow, carry, y, ys;
2463 Bug(
"oversize b in quorem");
2471 q = *bxe / (*sxe + 1);
2474 Bug(
"oversized quotient in quorem");
2481 ys = *sx++ * (ULLong)q + carry;
2483 y = *bx - (ys & FFFFFFFF) - borrow;
2484 borrow = y >> 32 & (ULong)1;
2485 *bx++ = (ULong)(y & FFFFFFFF);
2489 ys = (si & 0xffff) * q + carry;
2490 zs = (si >> 16) * q + (ys >> 16);
2492 y = (*bx & 0xffff) - (ys & 0xffff) - borrow;
2493 borrow = (y & 0x10000) >> 16;
2494 z = (*bx >> 16) - (zs & 0xffff) - borrow;
2495 borrow = (z & 0x10000) >> 16;
2498 ys = *sx++ * q + carry;
2500 y = *bx - (ys & 0xffff) - borrow;
2501 borrow = (y & 0x10000) >> 16;
2505 }
while (sx <= sxe);
2508 while (--bxe > bx && !*bxe)
2513 if (cmp(b, S) >= 0) {
2523 y = *bx - (ys & FFFFFFFF) - borrow;
2524 borrow = y >> 32 & (ULong)1;
2525 *bx++ = (ULong)(y & FFFFFFFF);
2529 ys = (si & 0xffff) + carry;
2530 zs = (si >> 16) + (ys >> 16);
2532 y = (*bx & 0xffff) - (ys & 0xffff) - borrow;
2533 borrow = (y & 0x10000) >> 16;
2534 z = (*bx >> 16) - (zs & 0xffff) - borrow;
2535 borrow = (z & 0x10000) >> 16;
2540 y = *bx - (ys & 0xffff) - borrow;
2541 borrow = (y & 0x10000) >> 16;
2545 }
while (sx <= sxe);
2549 while (--bxe > bx && !*bxe)
2557#ifndef MULTIPLE_THREADS
2558static char *dtoa_result;
2561#ifndef MULTIPLE_THREADS
2565 return dtoa_result = MALLOC(i);
2568#define rv_alloc(i) MALLOC(i)
2572nrv_alloc(
const char *s,
char **rve,
size_t n)
2576 t = rv = rv_alloc(n);
2577 if (!rv)
return NULL;
2578 while ((*t = *s++) != 0) t++;
2584#define rv_strdup(s, rve) nrv_alloc((s), (rve), strlen(s)+1)
2586#ifndef MULTIPLE_THREADS
2600static const char INFSTR[] =
"Infinity";
2601static const char NANSTR[] =
"NaN";
2602static const char ZEROSTR[] =
"0";
2639dtoa(
double d_,
int mode,
int ndigits,
int *decpt,
int *sign,
char **rve)
2675 int bbits, b2, b5, be, dig, i, ieps, ilim, ilim0, ilim1,
2676 j, j1, k, k0, k_check, leftright, m2, m5, s2, s5,
2677 spec_case, try_quick, half = 0;
2679#ifndef Sudden_Underflow
2683 Bigint *b, *b1, *delta, *mlo = 0, *mhi = 0, *S;
2687#ifdef Honor_FLT_ROUNDS
2691 int inexact, oldinexact;
2696#ifndef MULTIPLE_THREADS
2698 freedtoa(dtoa_result);
2703 if (word0(d) & Sign_bit) {
2706 word0(d) &= ~Sign_bit;
2711#if defined(IEEE_Arith) + defined(VAX)
2713 if ((word0(d) & Exp_mask) == Exp_mask)
2715 if (word0(d) == 0x8000)
2721 if (!word1(d) && !(word0(d) & 0xfffff))
2722 return rv_strdup(INFSTR, rve);
2724 return rv_strdup(NANSTR, rve);
2732 return rv_strdup(ZEROSTR, rve);
2736 try_quick = oldinexact = get_inexact();
2739#ifdef Honor_FLT_ROUNDS
2740 if ((rounding = Flt_Rounds) >= 2) {
2742 rounding = rounding == 2 ? 0 : 2;
2749 b = d2b(dval(d), &be, &bbits);
2750 if (!b)
return NULL;
2751#ifdef Sudden_Underflow
2752 i = (int)(word0(d) >> Exp_shift1 & (Exp_mask>>Exp_shift1));
2754 if ((i = (
int)(word0(d) >> Exp_shift1 & (Exp_mask>>Exp_shift1))) != 0) {
2757 word0(d2) &= Frac_mask1;
2758 word0(d2) |= Exp_11;
2760 if (j = 11 - hi0bits(word0(d2) & Frac_mask))
2791#ifndef Sudden_Underflow
2797 i = bbits + be + (Bias + (P-1) - 1);
2798 x = i > 32 ? word0(d) << (64 - i) | word1(d) >> (i - 32)
2799 : word1(d) << (32 - i);
2801 word0(d2) -= 31*Exp_msk1;
2802 i -= (Bias + (P-1) - 1) + 1;
2806 ds = (dval(d2)-1.5)*0.289529654602168 + 0.1760912590558 + i*0.301029995663981;
2808 if (ds < 0. && ds != k)
2811 if (k >= 0 && k <= Ten_pmax) {
2812 if (dval(d) < tens[k])
2835 if (mode < 0 || mode > 9)
2839#ifdef Check_FLT_ROUNDS
2840 try_quick = Rounding == 1;
2864 ilim = ilim1 = i = ndigits;
2870 if (ckd_add(&i, ndigits, k + 1)) {
2879 s = s0 = rv_alloc(i+1);
2885#ifdef Honor_FLT_ROUNDS
2886 if (mode > 1 && rounding != 1)
2890 if (ilim >= 0 && ilim <= Quick_max && try_quick) {
2905 dval(d) /= bigtens[n_bigtens-1];
2908 for (; j; j >>= 1, i++)
2915 else if ((j1 = -k) != 0) {
2916 dval(d) *= tens[j1 & 0xf];
2917 for (j = j1 >> 4; j; j >>= 1, i++)
2920 dval(d) *= bigtens[i];
2923 if (k_check && dval(d) < 1. && ilim > 0) {
2931 dval(eps) = ieps*dval(d) + 7.;
2932 word0(eps) -= (P-1)*Exp_msk1;
2936 if (dval(d) > dval(eps))
2938 if (dval(d) < -dval(eps))
2947 dval(eps) = 0.5/tens[ilim-1] - dval(eps);
2951 *s++ =
'0' + (int)L;
2952 if (dval(d) < dval(eps))
2954 if (1. - dval(d) < dval(eps))
2965 dval(eps) *= tens[ilim-1];
2966 for (i = 1;; i++, dval(d) *= 10.) {
2967 L = (Long)(dval(d));
2968 if (!(dval(d) -= L))
2970 *s++ =
'0' + (int)L;
2972 if (dval(d) > 0.5 + dval(eps))
2974 else if (dval(d) < 0.5 - dval(eps)) {
2975 while (*--s ==
'0') ;
2980 if ((*(s-1) -
'0') & 1) {
2998 if (be >= 0 && k <= Int_max) {
3001 if (ndigits < 0 && ilim <= 0) {
3003 if (ilim < 0 || dval(d) <= 5*ds)
3007 for (i = 1;; i++, dval(d) *= 10.) {
3008 L = (Long)(dval(d) / ds);
3010#ifdef Check_FLT_ROUNDS
3017 *s++ =
'0' + (int)L;
3025#ifdef Honor_FLT_ROUNDS
3029 case 2:
goto bump_up;
3033 if (dval(d) > ds || (dval(d) == ds && (L & 1))) {
3053#ifndef Sudden_Underflow
3054 denorm ? be + (Bias + (P-1) - 1 + 1) :
3057 1 + 4*P - 3 - bbits + ((bbits + be - 1) & 3);
3064 if (!mhi)
goto nomem;
3066 if (m2 > 0 && s2 > 0) {
3067 i = m2 < s2 ? m2 : s2;
3075 mhi = pow5mult(mhi, m5);
3076 if (!mhi)
goto nomem;
3082 if ((j = b5 - m5) != 0) {
3088 b = pow5mult(b, b5);
3095 S = pow5mult(S, s5);
3102 if ((mode < 2 || leftright)
3103#ifdef Honor_FLT_ROUNDS
3107 if (!word1(d) && !(word0(d) & Bndry_mask)
3108#ifndef Sudden_Underflow
3109 && word0(d) & (Exp_mask & ~Exp_msk1)
3127 if ((i = ((s5 ? 32 - hi0bits(S->x[S->wds-1]) : 1) + s2) & 0x1f) != 0)
3130 if ((i = ((s5 ? 32 - hi0bits(S->x[S->wds-1]) : 1) + s2) & 0xf) != 0)
3156 b = multadd(b, 10, 0);
3159 mhi = multadd(mhi, 10, 0);
3160 if (!mhi)
goto nomem;
3165 if (ilim <= 0 && (mode == 3 || mode == 5)) {
3166 if (ilim < 0 || cmp(b,S = multadd(S,5,0)) <= 0) {
3179 mhi = lshift(mhi, m2);
3180 if (!mhi)
goto nomem;
3189 mhi = Balloc(mhi->k);
3190 if (!mhi)
goto nomem;
3192 mhi = lshift(mhi, Log2P);
3193 if (!mhi)
goto nomem;
3197 dig = quorem(b,S) +
'0';
3202 delta = diff(S, mhi);
3203 if (!delta)
goto nomem;
3204 j1 = delta->sign ? 1 : cmp(b, delta);
3207 if (j1 == 0 && mode != 1 && !(word1(d) & 1)
3208#ifdef Honor_FLT_ROUNDS
3217 else if (!b->x[0] && b->wds <= 1)
3224 if (j < 0 || (j == 0 && mode != 1
3229 if (!b->x[0] && b->wds <= 1) {
3235#ifdef Honor_FLT_ROUNDS
3238 case 0:
goto accept_dig;
3239 case 2:
goto keep_dig;
3246 if ((j1 > 0 || (j1 == 0 && (dig & 1))) && dig++ ==
'9')
3254#ifdef Honor_FLT_ROUNDS
3266#ifdef Honor_FLT_ROUNDS
3272 b = multadd(b, 10, 0);
3275 mlo = mhi = multadd(mhi, 10, 0);
3276 if (!mlo)
goto nomem;
3279 mlo = multadd(mlo, 10, 0);
3280 if (!mlo)
goto nomem;
3281 mhi = multadd(mhi, 10, 0);
3282 if (!mhi)
goto nomem;
3288 *s++ = dig = quorem(b,S) +
'0';
3289 if (!b->x[0] && b->wds <= 1) {
3297 b = multadd(b, 10, 0);
3303#ifdef Honor_FLT_ROUNDS
3305 case 0:
goto trimzeros;
3306 case 2:
goto roundoff;
3312 if (j > 0 || (j == 0 && (dig & 1))) {
3320 if (!half || (*s -
'0') & 1)
3324 while (*--s ==
'0') ;
3330 if (mlo && mlo != mhi)
3338 word0(d) = Exp_1 + (70 << Exp_shift);
3343 else if (!oldinexact)
3355 if (mlo && mlo != mhi)
3390#define DBL_MANH_SIZE 20
3391#define DBL_MANL_SIZE 32
3392#define DBL_ADJ (DBL_MAX_EXP - 2)
3393#define SIGFIGS ((DBL_MANT_DIG + 3) / 4 + 1)
3394#define dexp_get(u) ((int)(word0(u) >> Exp_shift) & ~Exp_msk1)
3395#define dexp_set(u,v) (word0(u) = (((int)(word0(u)) & ~Exp_mask) | ((v) << Exp_shift)))
3396#define dmanh_get(u) ((uint32_t)(word0(u) & Frac_mask))
3397#define dmanl_get(u) ((uint32_t)word1(u))
3425hdtoa(
double d,
const char *xdigs,
int ndigits,
int *decpt,
int *sign,
char **rve)
3430 uint32_t manh, manl;
3433 if (word0(u) & Sign_bit) {
3436 word0(u) &= ~Sign_bit;
3443 return rv_strdup(INFSTR, rve);
3445 else if (isnan(d)) {
3447 return rv_strdup(NANSTR, rve);
3449 else if (d == 0.0) {
3451 return rv_strdup(ZEROSTR, rve);
3453 else if (dexp_get(u)) {
3454 *decpt = dexp_get(u) - DBL_ADJ;
3457 u.d *= 5.363123171977039e+154 ;
3458 *decpt = dexp_get(u) - (514 + DBL_ADJ);
3468 bufsize = (ndigits > 0) ? ndigits : SIGFIGS;
3469 s0 = rv_alloc(bufsize+1);
3470 if (!s0)
return NULL;
3473 if (SIGFIGS > ndigits && ndigits > 0) {
3475 int offset = 4 * ndigits + DBL_MAX_EXP - 4 - DBL_MANT_DIG;
3476 dexp_set(u, offset);
3479 *decpt += dexp_get(u) - offset;
3482 manh = dmanh_get(u);
3483 manl = dmanl_get(u);
3485 for (s = s0 + 1; s < s0 + bufsize; s++) {
3486 *s = xdigs[(manh >> (DBL_MANH_SIZE - 4)) & 0xf];
3487 manh = (manh << 4) | (manl >> (DBL_MANL_SIZE - 4));
3493 for (ndigits = SIGFIGS; s0[ndigits - 1] ==
'0'; ndigits--)
#define ISDIGIT
Old name of rb_isdigit.
#define strtod(s, e)
Just another name of ruby_strtod.
#define errno
Ractor-aware version of errno.