]> git.ipfire.org Git - thirdparty/gcc.git/commitdiff
ibm-ldouble.c (__gcc_qdiv): Scale up arguments in case of small numerator and finite...
authorJoseph Myers <joseph@codesourcery.com>
Fri, 3 Jan 2014 02:05:44 +0000 (02:05 +0000)
committerJoseph Myers <jsm28@gcc.gnu.org>
Fri, 3 Jan 2014 02:05:44 +0000 (02:05 +0000)
libgcc:
* config/rs6000/ibm-ldouble.c (__gcc_qdiv): Scale up arguments in
case of small numerator and finite nonzero result.

gcc/testsuite:
* gcc.target/powerpc/rs6000-ldouble-3.c: New test.

From-SVN: r206310

gcc/testsuite/ChangeLog
gcc/testsuite/gcc.target/powerpc/rs6000-ldouble-3.c [new file with mode: 0644]
libgcc/ChangeLog
libgcc/config/rs6000/ibm-ldouble.c

index 359421a406ddcdc7f0b36e860a766b3affacde02..ebc453d590123b3e384e323148559b10137f2277 100644 (file)
@@ -1,3 +1,7 @@
+2014-01-02  Joseph Myers  <joseph@codesourcery.com>
+
+       * gcc.target/powerpc/rs6000-ldouble-3.c: New test.
+
 2014-01-02  Marc Glisse  <marc.glisse@inria.fr>
 
        PR c++/59641
diff --git a/gcc/testsuite/gcc.target/powerpc/rs6000-ldouble-3.c b/gcc/testsuite/gcc.target/powerpc/rs6000-ldouble-3.c
new file mode 100644 (file)
index 0000000..1c78052
--- /dev/null
@@ -0,0 +1,21 @@
+/* Test accuracy of long double division (glibc bug 15396).  */
+/* { dg-do run { target powerpc*-*-linux* powerpc*-*-darwin* powerpc*-*-aix* rs6000-*-* } } */
+/* { dg-options "-mlong-double-128" } */
+
+extern void exit (int);
+extern void abort (void);
+
+volatile long double a = 0x1p-1024L;
+volatile long double b = 0x3p-53L;
+volatile long double r;
+volatile long double expected = 0x1.55555555555555555555555555p-973L;
+
+int
+main (void)
+{
+  r = a / b;
+  /* Allow error up to 2ulp.  */
+  if (__builtin_fabsl (r - expected) > 0x1p-1073L)
+    abort ();
+  exit (0);
+}
index 0c326525a92034c86e9ac13522ac3475ca3c8230..7fc4bd08449c5d3a32d0a42af997fcab00f95aa5 100644 (file)
@@ -1,3 +1,8 @@
+2014-01-02  Joseph Myers  <joseph@codesourcery.com>
+
+       * config/rs6000/ibm-ldouble.c (__gcc_qdiv): Scale up arguments in
+       case of small numerator and finite nonzero result.
+
 2014-01-02  Richard Sandiford  <rdsandiford@googlemail.com>
 
        Update copyright years
index 089de7d5352cffa218feecf4f27fba27f06f7e65..abcd3c5143526d4a682dace19ad59c7023b0dd03 100644 (file)
@@ -190,7 +190,16 @@ __gcc_qdiv (double a, double b, double c, double d)
       || nonfinite (t))
     return t;
 
-  /* Finite nonzero result requires corrections to the highest order term.  */
+  /* Finite nonzero result requires corrections to the highest order
+     term.  These corrections require the low part of c * t to be
+     exactly represented in double.  */
+  if (fabs (a) <= 0x1p-969)
+    {
+      a *= 0x1p106;
+      b *= 0x1p106;
+      c *= 0x1p106;
+      d *= 0x1p106;
+    }
 
   s = c * t;                    /* (s,sigma) = c*t exactly.  */
   w = -(-b + d * t);   /* Written to get fnmsub for speed, but not