diff options
author | Joseph Myers <joseph@codesourcery.com> | 2013-03-30 13:31:53 +0000 |
---|---|---|
committer | Joseph Myers <joseph@codesourcery.com> | 2013-03-30 13:31:53 +0000 |
commit | ccc8cadf750fc3626fa418a92642b6127428f404 (patch) | |
tree | 51befecaa48f28eed844c0aba00913fb5c504f6f /math/k_casinh.c | |
parent | 0d1029de128ad2afd4a7b13a0805aea8fdb82e37 (diff) | |
download | glibc-ccc8cadf750fc3626fa418a92642b6127428f404.tar.gz glibc-ccc8cadf750fc3626fa418a92642b6127428f404.tar.xz glibc-ccc8cadf750fc3626fa418a92642b6127428f404.zip |
Fix casinh inaccuracy for imaginary part < 1.0, real part small (bug 10357).
Diffstat (limited to 'math/k_casinh.c')
-rw-r--r-- | math/k_casinh.c | 52 |
1 files changed, 52 insertions, 0 deletions
diff --git a/math/k_casinh.c b/math/k_casinh.c index 4cb232a17a..145eb10110 100644 --- a/math/k_casinh.c +++ b/math/k_casinh.c @@ -134,6 +134,58 @@ __kernel_casinh (__complex__ double x, int adj) __imag__ res = __ieee754_atan2 (1.0 + s2, rx + s1); } } + else if (ix < 1.0 && rx < 0.5) + { + if (ix >= DBL_EPSILON) + { + if (rx < DBL_EPSILON * DBL_EPSILON) + { + double onemix2 = (1.0 + ix) * (1.0 - ix); + double s = __ieee754_sqrt (onemix2); + + __real__ res = __log1p (2.0 * rx / s) / 2.0; + if (adj) + __imag__ res = __ieee754_atan2 (s, __imag__ x); + else + __imag__ res = __ieee754_atan2 (ix, s); + } + else + { + double onemix2 = (1.0 + ix) * (1.0 - ix); + double rx2 = rx * rx; + double f = rx2 * (2.0 + rx2 + 2.0 * ix * ix); + double d = __ieee754_sqrt (onemix2 * onemix2 + f); + double dp = d + onemix2; + double dm = f / dp; + double r1 = __ieee754_sqrt ((dp + rx2) / 2.0); + double r2 = rx * ix / r1; + + __real__ res + = __log1p (rx2 + dm + 2.0 * (rx * r1 + ix * r2)) / 2.0; + if (adj) + __imag__ res = __ieee754_atan2 (rx + r1, + __copysign (ix + r2, + __imag__ x)); + else + __imag__ res = __ieee754_atan2 (ix + r2, rx + r1); + } + } + else + { + double s = __ieee754_hypot (1.0, rx); + + __real__ res = __log1p (2.0 * rx * (rx + s)) / 2.0; + if (adj) + __imag__ res = __ieee754_atan2 (s, __imag__ x); + else + __imag__ res = __ieee754_atan2 (ix, s); + } + if (__real__ res < DBL_MIN) + { + volatile double force_underflow = __real__ res * __real__ res; + (void) force_underflow; + } + } else { __real__ y = (rx - ix) * (rx + ix) + 1.0; |