MATH-476 FastMath code contains 'magic' numbers

Extracted "splitter" value as a constant

git-svn-id: https://svn.apache.org/repos/asf/commons/proper/math/trunk@1061621 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Sebastian Bazley 2011-01-21 04:03:03 +00:00
parent 64a0eee333
commit d4005bbd6e
1 changed files with 58 additions and 51 deletions

View File

@ -161,6 +161,11 @@ public class FastMath {
1.2599210498948732, 1.2599210498948732,
1.5874010519681994 }; 1.5874010519681994 };
/**
* 0x40000000 - used to split a double into two parts, both with the low order bits cleared.
*/
private static final double HEX_40000000 = 1073741824.0;
// Initialize tables // Initialize tables
static { static {
int i; int i;
@ -245,13 +250,13 @@ public class FastMath {
double ya = hiPrec[0] + hiPrec[1]; double ya = hiPrec[0] + hiPrec[1];
double yb = -(ya - hiPrec[0] - hiPrec[1]); double yb = -(ya - hiPrec[0] - hiPrec[1]);
double temp = ya * 1073741824.0; double temp = ya * HEX_40000000;
double yaa = ya + temp - temp; double yaa = ya + temp - temp;
double yab = ya - yaa; double yab = ya - yaa;
// recip = 1/y // recip = 1/y
double recip = 1.0/ya; double recip = 1.0/ya;
temp = recip * 1073741824.0; temp = recip * HEX_40000000;
double recipa = recip + temp - temp; double recipa = recip + temp - temp;
double recipb = recip - recipa; double recipb = recip - recipa;
@ -309,13 +314,13 @@ public class FastMath {
double ya = hiPrec[0] + hiPrec[1]; double ya = hiPrec[0] + hiPrec[1];
double yb = -(ya - hiPrec[0] - hiPrec[1]); double yb = -(ya - hiPrec[0] - hiPrec[1]);
double temp = ya * 1073741824.0; double temp = ya * HEX_40000000;
double yaa = ya + temp - temp; double yaa = ya + temp - temp;
double yab = ya - yaa; double yab = ya - yaa;
// recip = 1/y // recip = 1/y
double recip = 1.0/ya; double recip = 1.0/ya;
temp = recip * 1073741824.0; temp = recip * HEX_40000000;
double recipa = recip + temp - temp; double recipa = recip + temp - temp;
double recipb = recip - recipa; double recipb = recip - recipa;
@ -350,11 +355,11 @@ public class FastMath {
double denomr = 1.0 / denom; double denomr = 1.0 / denom;
double denomb = -(denom - 1.0 - ya) + yb; double denomb = -(denom - 1.0 - ya) + yb;
double ratio = ya * denomr; double ratio = ya * denomr;
double temp = ratio * 1073741824.0; double temp = ratio * HEX_40000000;
double ra = ratio + temp - temp; double ra = ratio + temp - temp;
double rb = ratio - ra; double rb = ratio - ra;
temp = denom * 1073741824.0; temp = denom * HEX_40000000;
double za = denom + temp - temp; double za = denom + temp - temp;
double zb = denom - za; double zb = denom - za;
@ -434,13 +439,13 @@ public class FastMath {
db += -(temp - da - yb); db += -(temp - da - yb);
da = temp; da = temp;
temp = da * 1073741824.0; temp = da * HEX_40000000;
double daa = da + temp - temp; double daa = da + temp - temp;
double dab = da - daa; double dab = da - daa;
// ratio = na/da // ratio = na/da
double ratio = na/da; double ratio = na/da;
temp = ratio * 1073741824.0; temp = ratio * HEX_40000000;
double ratioa = ratio + temp - temp; double ratioa = ratio + temp - temp;
double ratiob = ratio - ratioa; double ratiob = ratio - ratioa;
@ -473,13 +478,13 @@ public class FastMath {
db += -(temp - da - yb); db += -(temp - da - yb);
da = temp; da = temp;
temp = da * 1073741824.0; temp = da * HEX_40000000;
double daa = da + temp - temp; double daa = da + temp - temp;
double dab = da - daa; double dab = da - daa;
// ratio = na/da // ratio = na/da
double ratio = na/da; double ratio = na/da;
temp = ratio * 1073741824.0; temp = ratio * HEX_40000000;
double ratioa = ratio + temp - temp; double ratioa = ratio + temp - temp;
double ratiob = ratio - ratioa; double ratiob = ratio - ratioa;
@ -814,7 +819,7 @@ public class FastMath {
tempB = -(temp - tempA - tempB); tempB = -(temp - tempA - tempB);
tempA = temp; tempA = temp;
temp = tempA * 1073741824.0; temp = tempA * HEX_40000000;
baseA = tempA + temp - temp; baseA = tempA + temp - temp;
baseB = tempB + (tempA - baseA); baseB = tempB + (tempA - baseA);
@ -835,7 +840,7 @@ public class FastMath {
zb = -(temp - za - zb); zb = -(temp - za - zb);
za = temp; za = temp;
temp = za * 1073741824.0; temp = za * HEX_40000000;
temp = za + temp - temp; temp = za + temp - temp;
zb += za - temp; zb += za - temp;
za = temp; za = temp;
@ -882,11 +887,11 @@ public class FastMath {
double denomr = 1.0 / denom; double denomr = 1.0 / denom;
double denomb = -(denom - 1.0 - ya) + yb; double denomb = -(denom - 1.0 - ya) + yb;
double ratio = ya * denomr; double ratio = ya * denomr;
temp = ratio * 1073741824.0; temp = ratio * HEX_40000000;
final double ra = ratio + temp - temp; final double ra = ratio + temp - temp;
double rb = ratio - ra; double rb = ratio - ra;
temp = denom * 1073741824.0; temp = denom * HEX_40000000;
za = denom + temp - temp; za = denom + temp - temp;
zb = denom - za; zb = denom - za;
@ -960,12 +965,12 @@ public class FastMath {
*/ */
private static void split(final double d, final double split[]) { private static void split(final double d, final double split[]) {
if (d < 8e298 && d > -8e298) { if (d < 8e298 && d > -8e298) {
final double a = d * 1073741824.0; final double a = d * HEX_40000000;
split[0] = (d + a) - a; split[0] = (d + a) - a;
split[1] = d - split[0]; split[1] = d - split[0];
} else { } else {
final double a = d * 9.31322574615478515625E-10; final double a = d * 9.31322574615478515625E-10;
split[0] = (d + a - d) * 1073741824.0; split[0] = (d + a - d) * HEX_40000000;
split[1] = d - split[0]; split[1] = d - split[0];
} }
} }
@ -979,12 +984,12 @@ public class FastMath {
final double d = -(c - a[0] - a[1]); final double d = -(c - a[0] - a[1]);
if (c < 8e298 && c > -8e298) { if (c < 8e298 && c > -8e298) {
double z = c * 1073741824.0; double z = c * HEX_40000000;
a[0] = (c + z) - z; a[0] = (c + z) - z;
a[1] = c - a[0] + d; a[1] = c - a[0] + d;
} else { } else {
double z = c * 9.31322574615478515625E-10; double z = c * 9.31322574615478515625E-10;
a[0] = (c + z - c) * 1073741824.0; a[0] = (c + z - c) * HEX_40000000;
a[1] = c - a[0] + d; a[1] = c - a[0] + d;
} }
} }
@ -1235,7 +1240,7 @@ public class FastMath {
/* Compute x - 1.0 and split it */ /* Compute x - 1.0 and split it */
double xa = x - 1.0; double xa = x - 1.0;
double xb = xa - x + 1.0; double xb = xa - x + 1.0;
double tmp = xa * 1073741824.0; double tmp = xa * HEX_40000000;
double aa = xa + tmp - tmp; double aa = xa + tmp - tmp;
double ab = xa - aa; double ab = xa - aa;
xa = aa; xa = aa;
@ -1249,7 +1254,7 @@ public class FastMath {
aa = ya * xa; aa = ya * xa;
ab = ya * xb + yb * xa + yb * xb; ab = ya * xb + yb * xa + yb * xb;
/* split, so now y = a */ /* split, so now y = a */
tmp = aa * 1073741824.0; tmp = aa * HEX_40000000;
ya = aa + tmp - tmp; ya = aa + tmp - tmp;
yb = aa - ya + ab; yb = aa - ya + ab;
@ -1257,7 +1262,7 @@ public class FastMath {
aa = ya + LN_QUICK_COEF[i][0]; aa = ya + LN_QUICK_COEF[i][0];
ab = yb + LN_QUICK_COEF[i][1]; ab = yb + LN_QUICK_COEF[i][1];
/* Split y = a */ /* Split y = a */
tmp = aa * 1073741824.0; tmp = aa * HEX_40000000;
ya = aa + tmp - tmp; ya = aa + tmp - tmp;
yb = aa - ya + ab; yb = aa - ya + ab;
} }
@ -1266,7 +1271,7 @@ public class FastMath {
aa = ya * xa; aa = ya * xa;
ab = ya * xb + yb * xa + yb * xb; ab = ya * xb + yb * xa + yb * xb;
/* split, so now y = a */ /* split, so now y = a */
tmp = aa * 1073741824.0; tmp = aa * HEX_40000000;
ya = aa + tmp - tmp; ya = aa + tmp - tmp;
yb = aa - ya + ab; yb = aa - ya + ab;
@ -1293,7 +1298,7 @@ public class FastMath {
if (hiPrec != null) { if (hiPrec != null) {
/* split epsilon -> x */ /* split epsilon -> x */
double tmp = epsilon * 1073741824.0; double tmp = epsilon * HEX_40000000;
double aa = epsilon + tmp - tmp; double aa = epsilon + tmp - tmp;
double ab = epsilon - aa; double ab = epsilon - aa;
double xa = aa; double xa = aa;
@ -1314,7 +1319,7 @@ public class FastMath {
aa = ya * xa; aa = ya * xa;
ab = ya * xb + yb * xa + yb * xb; ab = ya * xb + yb * xa + yb * xb;
/* split, so now y = a */ /* split, so now y = a */
tmp = aa * 1073741824.0; tmp = aa * HEX_40000000;
ya = aa + tmp - tmp; ya = aa + tmp - tmp;
yb = aa - ya + ab; yb = aa - ya + ab;
@ -1322,7 +1327,7 @@ public class FastMath {
aa = ya + LN_HI_PREC_COEF[i][0]; aa = ya + LN_HI_PREC_COEF[i][0];
ab = yb + LN_HI_PREC_COEF[i][1]; ab = yb + LN_HI_PREC_COEF[i][1];
/* Split y = a */ /* Split y = a */
tmp = aa * 1073741824.0; tmp = aa * HEX_40000000;
ya = aa + tmp - tmp; ya = aa + tmp - tmp;
yb = aa - ya + ab; yb = aa - ya + ab;
} }
@ -1454,7 +1459,7 @@ public class FastMath {
return lores; return lores;
} }
final double tmp = hiPrec[0] * 1073741824.0; final double tmp = hiPrec[0] * HEX_40000000;
final double lna = hiPrec[0] + tmp - tmp; final double lna = hiPrec[0] + tmp - tmp;
final double lnb = hiPrec[0] - lna + hiPrec[1]; final double lnb = hiPrec[0] - lna + hiPrec[1];
@ -1590,13 +1595,13 @@ public class FastMath {
double ya; double ya;
double yb; double yb;
if (y < 8e298 && y > -8e298) { if (y < 8e298 && y > -8e298) {
double tmp1 = y * 1073741824.0; double tmp1 = y * HEX_40000000;
ya = y + tmp1 - tmp1; ya = y + tmp1 - tmp1;
yb = y - ya; yb = y - ya;
} else { } else {
double tmp1 = y * 9.31322574615478515625E-10; double tmp1 = y * 9.31322574615478515625E-10;
double tmp2 = tmp1 * 9.31322574615478515625E-10; double tmp2 = tmp1 * 9.31322574615478515625E-10;
ya = (tmp1 + tmp2 - tmp1) * 1073741824.0 * 1073741824.0; ya = (tmp1 + tmp2 - tmp1) * HEX_40000000 * HEX_40000000;
yb = y - ya; yb = y - ya;
} }
@ -1610,7 +1615,7 @@ public class FastMath {
double lnb = lns[1]; double lnb = lns[1];
/* resplit lns */ /* resplit lns */
double tmp1 = lna * 1073741824.0; double tmp1 = lna * HEX_40000000;
double tmp2 = lna + tmp1 - tmp1; double tmp2 = lna + tmp1 - tmp1;
lnb += lna - tmp2; lnb += lna - tmp2;
lna = tmp2; lna = tmp2;
@ -1941,7 +1946,7 @@ public class FastMath {
final double cosEpsB = polyCosine(epsilon); final double cosEpsB = polyCosine(epsilon);
// Split epsilon xa + xb = x // Split epsilon xa + xb = x
final double temp = sinEpsA * 1073741824.0; final double temp = sinEpsA * HEX_40000000;
double temp2 = (sinEpsA + temp) - temp; double temp2 = (sinEpsA + temp) - temp;
sinEpsB += sinEpsA - temp2; sinEpsB += sinEpsA - temp2;
sinEpsA = temp2; sinEpsA = temp2;
@ -2085,7 +2090,7 @@ public class FastMath {
final double cosEpsB = polyCosine(epsilon); final double cosEpsB = polyCosine(epsilon);
// Split epsilon xa + xb = x // Split epsilon xa + xb = x
double temp = sinEpsA * 1073741824.0; double temp = sinEpsA * HEX_40000000;
double temp2 = (sinEpsA + temp) - temp; double temp2 = (sinEpsA + temp) - temp;
sinEpsB += sinEpsA - temp2; sinEpsB += sinEpsA - temp2;
sinEpsA = temp2; sinEpsA = temp2;
@ -2177,11 +2182,11 @@ public class FastMath {
double est = sina/cosa; double est = sina/cosa;
/* Split the estimate to get more accurate read on division rounding */ /* Split the estimate to get more accurate read on division rounding */
temp = est * 1073741824.0; temp = est * HEX_40000000;
double esta = (est + temp) - temp; double esta = (est + temp) - temp;
double estb = est - esta; double estb = est - esta;
temp = cosa * 1073741824.0; temp = cosa * HEX_40000000;
double cosaa = (cosa + temp) - temp; double cosaa = (cosa + temp) - temp;
double cosab = cosa - cosaa; double cosab = cosa - cosaa;
@ -2765,7 +2770,7 @@ public class FastMath {
epsA = temp; epsA = temp;
/* Compute eps = eps / (1.0 + xa*tangent) */ /* Compute eps = eps / (1.0 + xa*tangent) */
temp = xa * 1073741824.0; temp = xa * HEX_40000000;
double ya = xa + temp - temp; double ya = xa + temp - temp;
double yb = xb + xa - ya; double yb = xb + xa - ya;
xa = ya; xa = ya;
@ -2791,11 +2796,11 @@ public class FastMath {
zb += xb * TANGENT_TABLE_B[idx]; zb += xb * TANGENT_TABLE_B[idx];
ya = epsA / za; ya = epsA / za;
temp = ya * 1073741824.0; temp = ya * HEX_40000000;
final double yaa = (ya + temp) - temp; final double yaa = (ya + temp) - temp;
final double yab = ya - yaa; final double yab = ya - yaa;
temp = za * 1073741824.0; temp = za * HEX_40000000;
final double zaa = (za + temp) - temp; final double zaa = (za + temp) - temp;
final double zab = za - zaa; final double zab = za - zaa;
@ -2974,13 +2979,13 @@ public class FastMath {
} }
// Split y // Split y
double temp = x * 1073741824.0; double temp = x * HEX_40000000;
final double xa = x + temp - temp; final double xa = x + temp - temp;
final double xb = x - xa; final double xb = x - xa;
// Compute ratio r = x/y // Compute ratio r = x/y
final double r = y/x; final double r = y/x;
temp = r * 1073741824.0; temp = r * HEX_40000000;
double ra = r + temp - temp; double ra = r + temp - temp;
double rb = r - ra; double rb = r - ra;
@ -3024,7 +3029,7 @@ public class FastMath {
/* Compute asin(x) = atan(x/sqrt(1-x*x)) */ /* Compute asin(x) = atan(x/sqrt(1-x*x)) */
/* Split x */ /* Split x */
double temp = x * 1073741824.0; double temp = x * HEX_40000000;
final double xa = x + temp - temp; final double xa = x + temp - temp;
final double xb = x - xa; final double xb = x - xa;
@ -3046,7 +3051,7 @@ public class FastMath {
/* Square root */ /* Square root */
double y; double y;
y = sqrt(za); y = sqrt(za);
temp = y * 1073741824.0; temp = y * HEX_40000000;
ya = y + temp - temp; ya = y + temp - temp;
yb = y - ya; yb = y - ya;
@ -3058,7 +3063,7 @@ public class FastMath {
// Compute ratio r = x/y // Compute ratio r = x/y
double r = x/y; double r = x/y;
temp = r * 1073741824.0; temp = r * HEX_40000000;
double ra = r + temp - temp; double ra = r + temp - temp;
double rb = r - ra; double rb = r - ra;
@ -3100,7 +3105,7 @@ public class FastMath {
/* Compute acos(x) = atan(sqrt(1-x*x)/x) */ /* Compute acos(x) = atan(sqrt(1-x*x)/x) */
/* Split x */ /* Split x */
double temp = x * 1073741824.0; double temp = x * HEX_40000000;
final double xa = x + temp - temp; final double xa = x + temp - temp;
final double xb = x - xa; final double xb = x - xa;
@ -3121,7 +3126,7 @@ public class FastMath {
/* Square root */ /* Square root */
double y = sqrt(za); double y = sqrt(za);
temp = y * 1073741824.0; temp = y * HEX_40000000;
ya = y + temp - temp; ya = y + temp - temp;
yb = y - ya; yb = y - ya;
@ -3141,8 +3146,8 @@ public class FastMath {
return Math.PI/2; // so return the appropriate value return Math.PI/2; // so return the appropriate value
} }
if (abs(r) < Double.MAX_VALUE/1073741824.0){ // is it safe to split r ? if (abs(r) < Double.MAX_VALUE/HEX_40000000){ // is it safe to split r ?
temp = r * 1073741824.0; temp = r * HEX_40000000;
} else { } else {
temp = 0.0; temp = 0.0;
} }
@ -3214,13 +3219,13 @@ public class FastMath {
est += (xs - est*est*est) / (3*est*est); est += (xs - est*est*est) / (3*est*est);
// Do one round of Newton's method in extended precision to get the last bit right. // Do one round of Newton's method in extended precision to get the last bit right.
double temp = est * 1073741824.0; double temp = est * HEX_40000000;
double ya = est + temp - temp; double ya = est + temp - temp;
double yb = est - ya; double yb = est - ya;
double za = ya * ya; double za = ya * ya;
double zb = ya * yb * 2.0 + yb * yb; double zb = ya * yb * 2.0 + yb * yb;
temp = za * 1073741824.0; temp = za * HEX_40000000;
double temp2 = za + temp - temp; double temp2 = za + temp - temp;
zb += za - temp2; zb += za - temp2;
za = temp2; za = temp2;
@ -3255,12 +3260,13 @@ public class FastMath {
return x; return x;
} }
// These are PI/180 split into high and low order bits
final double facta = 0.01745329052209854; final double facta = 0.01745329052209854;
final double factb = 1.997844754509471E-9; final double factb = 1.997844754509471E-9;
double temp = 0; double temp = 0;
if (abs(x) < Double.MAX_VALUE/1073741824.0) { // prevent overflow to infinity if (abs(x) < Double.MAX_VALUE/HEX_40000000) { // prevent overflow to infinity
temp = x * 1073741824.0; temp = x * HEX_40000000;
} }
double xa = x + temp - temp; double xa = x + temp - temp;
double xb = x - xa; double xb = x - xa;
@ -3283,12 +3289,13 @@ public class FastMath {
return x; return x;
} }
// These are 180/PI split into high and low order bits
final double facta = 57.2957763671875; final double facta = 57.2957763671875;
final double factb = 3.145894820876798E-6; final double factb = 3.145894820876798E-6;
double temp = 0; double temp = 0;
if (abs(x) < Double.MAX_VALUE/1073741824.0) { // prevent overflow to infinity if (abs(x) < Double.MAX_VALUE/HEX_40000000) { // prevent overflow to infinity
temp = x * 1073741824.0; temp = x * HEX_40000000;
} }
double xa = x + temp - temp; double xa = x + temp - temp;
double xb = x - xa; double xb = x - xa;