aboutsummaryrefslogtreecommitdiff
path: root/math/k_casinhf.c
diff options
context:
space:
mode:
authorJoseph Myers <joseph@codesourcery.com>2013-03-27 14:38:44 +0000
committerJoseph Myers <joseph@codesourcery.com>2013-03-27 14:38:44 +0000
commit3a7182a14becbf3149419bdbaff4388b636c9f31 (patch)
tree16616a6c0915ec7099fcd318204f8e5741571b29 /math/k_casinhf.c
parent28831a9a671b109e49d0a4ddce00a08d734bfc97 (diff)
downloadglibc-3a7182a14becbf3149419bdbaff4388b636c9f31.zip
glibc-3a7182a14becbf3149419bdbaff4388b636c9f31.tar.gz
glibc-3a7182a14becbf3149419bdbaff4388b636c9f31.tar.bz2
Fix casinh inaccuracy near i, imaginary part > 1 (bug 15307).
Diffstat (limited to 'math/k_casinhf.c')
-rw-r--r--math/k_casinhf.c33
1 files changed, 33 insertions, 0 deletions
diff --git a/math/k_casinhf.c b/math/k_casinhf.c
index 4d1a92f..b368313 100644
--- a/math/k_casinhf.c
+++ b/math/k_casinhf.c
@@ -77,6 +77,39 @@ __kernel_casinhf (__complex__ float x, int adj)
else
__imag__ res = __ieee754_atan2f (s, rx);
}
+ else if (ix > 1.0f && ix < 1.5f && rx < 0.5f)
+ {
+ if (rx < FLT_EPSILON * FLT_EPSILON)
+ {
+ float ix2m1 = (ix + 1.0f) * (ix - 1.0f);
+ float s = __ieee754_sqrtf (ix2m1);
+
+ __real__ res = __log1pf (2.0f * (ix2m1 + ix * s)) / 2.0f;
+ if (adj)
+ __imag__ res = __ieee754_atan2f (rx, __copysignf (s, __imag__ x));
+ else
+ __imag__ res = __ieee754_atan2f (s, rx);
+ }
+ else
+ {
+ float ix2m1 = (ix + 1.0f) * (ix - 1.0f);
+ float rx2 = rx * rx;
+ float f = rx2 * (2.0f + rx2 + 2.0f * ix * ix);
+ float d = __ieee754_sqrtf (ix2m1 * ix2m1 + f);
+ float dp = d + ix2m1;
+ float dm = f / dp;
+ float r1 = __ieee754_sqrtf ((dm + rx2) / 2.0f);
+ float r2 = rx * ix / r1;
+
+ __real__ res
+ = __log1pf (rx2 + dp + 2.0f * (rx * r1 + ix * r2)) / 2.0f;
+ if (adj)
+ __imag__ res = __ieee754_atan2f (rx + r1, __copysignf (ix + r2,
+ __imag__ x));
+ else
+ __imag__ res = __ieee754_atan2f (ix + r2, rx + r1);
+ }
+ }
else if (ix == 1.0f && rx < 0.5f)
{
if (rx < FLT_EPSILON / 8.0f)