aboutsummaryrefslogtreecommitdiff
path: root/math/k_casinh.c
diff options
context:
space:
mode:
authorJoseph Myers <joseph@codesourcery.com>2013-03-30 13:31:53 +0000
committerJoseph Myers <joseph@codesourcery.com>2013-03-30 13:31:53 +0000
commitccc8cadf750fc3626fa418a92642b6127428f404 (patch)
tree51befecaa48f28eed844c0aba00913fb5c504f6f /math/k_casinh.c
parent0d1029de128ad2afd4a7b13a0805aea8fdb82e37 (diff)
downloadglibc-ccc8cadf750fc3626fa418a92642b6127428f404.zip
glibc-ccc8cadf750fc3626fa418a92642b6127428f404.tar.gz
glibc-ccc8cadf750fc3626fa418a92642b6127428f404.tar.bz2
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.c52
1 files changed, 52 insertions, 0 deletions
diff --git a/math/k_casinh.c b/math/k_casinh.c
index 4cb232a..145eb10 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;