]> git.ipfire.org Git - thirdparty/gcc.git/commitdiff
Fix i386 FP_TRAPPING_EXCEPTIONS.
authorJoseph Myers <joseph@codesourcery.com>
Thu, 18 Sep 2014 12:04:43 +0000 (13:04 +0100)
committerJoseph Myers <jsm28@gcc.gnu.org>
Thu, 18 Sep 2014 12:04:43 +0000 (13:04 +0100)
The i386 sfp-machine.h defines FP_TRAPPING_EXCEPTIONS in a way that is
always wrong: it treats a set bit as indicating the exception is
trapping, when actually a set bit (both for 387 and SSE floating
point) indicates it is masked, and a clear bit indicates it is
trapping.  This patch fixes this bug.

Bootstrapped with no regressions on x86_64-unknown-linux-gnu.

libgcc:
* config/i386/sfp-machine.h (FP_TRAPPING_EXCEPTIONS): Treat clear
bits not set bits as indicating trapping exceptions.

gcc/testsuite:
* gcc.dg/torture/float128-exact-underflow.c: New test.

From-SVN: r215349

gcc/testsuite/ChangeLog
gcc/testsuite/gcc.dg/torture/float128-exact-underflow.c [new file with mode: 0644]
libgcc/ChangeLog
libgcc/config/i386/sfp-machine.h

index 1a340d2145bd6cd672d52e6d567ac1e453967554..9aab1c76fc498afd1b62ac598cfe773a5e43c73b 100644 (file)
@@ -1,3 +1,7 @@
+2014-09-18  Joseph Myers  <joseph@codesourcery.com>
+
+       * gcc.dg/torture/float128-exact-underflow.c: New test.
+
 2014-09-17  Jakub Jelinek  <jakub@redhat.com>
 
        PR debug/63284
diff --git a/gcc/testsuite/gcc.dg/torture/float128-exact-underflow.c b/gcc/testsuite/gcc.dg/torture/float128-exact-underflow.c
new file mode 100644 (file)
index 0000000..ea11f26
--- /dev/null
@@ -0,0 +1,41 @@
+/* Test that exact underflow in __float128 signals the underflow
+   exception if trapping is enabled, but does not raise the flag
+   otherwise.  */
+
+/* { dg-do run { target i?86-*-*gnu* x86_64-*-*gnu* } } */
+/* { dg-options "-D_GNU_SOURCE" } */
+/* { dg-require-effective-target fenv_exceptions } */
+
+#include <fenv.h>
+#include <setjmp.h>
+#include <signal.h>
+#include <stdlib.h>
+
+volatile sig_atomic_t caught_sigfpe;
+sigjmp_buf buf;
+
+static void
+handle_sigfpe (int sig)
+{
+  caught_sigfpe = 1;
+  siglongjmp (buf, 1);
+}
+
+int
+main (void)
+{
+  volatile __float128 a = 0x1p-16382q, b = 0x1p-2q;
+  volatile __float128 r;
+  r = a * b;
+  if (fetestexcept (FE_UNDERFLOW))
+    abort ();
+  if (r != 0x1p-16384q)
+    abort ();
+  feenableexcept (FE_UNDERFLOW);
+  signal (SIGFPE, handle_sigfpe);
+  if (sigsetjmp (buf, 1) == 0)
+    r = a * b;
+  if (!caught_sigfpe)
+    abort ();
+  exit (0);
+}
index 530597813c531c7abc7fba0d9f0c56e899e0aac6..0dbd425000ab5bddab0bbf7526d4509748b25886 100644 (file)
@@ -1,3 +1,8 @@
+2014-09-18  Joseph Myers  <joseph@codesourcery.com>
+
+       * config/i386/sfp-machine.h (FP_TRAPPING_EXCEPTIONS): Treat clear
+       bits not set bits as indicating trapping exceptions.
+
 2014-05-22  Release Manager
 
        * GCC 4.8.3 released.
index b19d94b6192d37ccfd92b28b0cb6ccf754f22bf1..27b42a108a0cdee85dee3ed0a611eb6921f70199 100644 (file)
@@ -59,7 +59,7 @@ void __sfp_handle_exceptions (int);
       __sfp_handle_exceptions (_fex);          \
   } while (0);
 
-#define FP_TRAPPING_EXCEPTIONS ((_fcw >> FP_EX_SHIFT) & FP_EX_ALL)
+#define FP_TRAPPING_EXCEPTIONS ((~_fcw >> FP_EX_SHIFT) & FP_EX_ALL)
 
 #define FP_ROUNDMODE           (_fcw & FP_RND_MASK)
 #endif