Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/lib/libc/compat/gen Teach libc's compat ldexp stub to raise ...



details:   https://anonhg.NetBSD.org/src/rev/e70d0ddaf288
branches:  trunk
changeset: 972243:e70d0ddaf288
user:      riastradh <riastradh%NetBSD.org@localhost>
date:      Thu May 21 05:56:31 2020 +0000

description:
Teach libc's compat ldexp stub to raise fp exceptions.

This ldexp stub will shadow the ldexp weak alias for scalbn in libm,
which is unfortunate but hard to fix properly without chasing the
mythical libc bump beast.  With the change here, we should raise all
the same exceptions that libm's scalbn does -- overflow, underflow,
inexact-result, and (for signalling NaN only) invalid-operation.
This in turn should correct the missing overflow/underflow exceptions
of our portable C fma, and perhaps other routines.

XXX pullup

diffstat:

 lib/libc/compat/gen/compat_ldexp_ieee754.c |  57 ++++++++++++++++++-----------
 1 files changed, 35 insertions(+), 22 deletions(-)

diffs (108 lines):

diff -r 0f56c98c1262 -r e70d0ddaf288 lib/libc/compat/gen/compat_ldexp_ieee754.c
--- a/lib/libc/compat/gen/compat_ldexp_ieee754.c        Thu May 21 05:41:40 2020 +0000
+++ b/lib/libc/compat/gen/compat_ldexp_ieee754.c        Thu May 21 05:56:31 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: compat_ldexp_ieee754.c,v 1.7 2016/08/27 09:35:13 christos Exp $ */
+/* $NetBSD: compat_ldexp_ieee754.c,v 1.8 2020/05/21 05:56:31 riastradh Exp $ */
 
 /*-
  * Copyright (c) 1999 The NetBSD Foundation, Inc.
@@ -31,14 +31,34 @@
 
 #include <sys/cdefs.h>
 #if defined(LIBC_SCCS) && !defined(lint)
-__RCSID("$NetBSD: compat_ldexp_ieee754.c,v 1.7 2016/08/27 09:35:13 christos Exp $");
+__RCSID("$NetBSD: compat_ldexp_ieee754.c,v 1.8 2020/05/21 05:56:31 riastradh Exp $");
 #endif /* LIBC_SCCS and not lint */
 
 #include <sys/types.h>
+
 #include <machine/ieee.h>
+
 #include <errno.h>
+#include <float.h>
+#include <math.h>
 
-double ldexp(double, int);
+static volatile const double tiny = DBL_MIN, huge = DBL_MAX;
+
+static double
+underflow(double val)
+{
+
+       errno = ERANGE;
+       return (val < 0 ? -tiny*tiny : tiny*tiny);
+}
+
+static double
+overflow(double val)
+{
+
+       errno = ERANGE;
+       return (val < 0 ? -huge*huge : huge*huge);
+}
 
 /*
  * Multiply the given value by 2^expon.
@@ -53,10 +73,13 @@
        oldexp = u.dblu_dbl.dbl_exp;
 
        /*
-        * If input is zero, Inf or NaN, just return it.
+        * If input is zero, Inf or NaN, just return it, but raise
+        * invalid exception if it is a signalling NaN: adding any of
+        * these inputs to itself gives itself as output; arithmetic on
+        * a signalling NaN additionally raises invalid-operation.
         */
        if (u.dblu_d == 0.0 || oldexp == DBL_EXP_INFNAN)
-               return (val);
+               return (val + val);
 
        if (oldexp == 0) {
                /*
@@ -68,17 +91,13 @@
                         * Optimization: if the scaling can be done in a single
                         * multiply, or underflows, just do it now.
                         */
-                       if (expon <= -DBL_FRACBITS) {
-                               errno = ERANGE;
-                               return (val < 0.0 ? -0.0 : 0.0);
-                       }
+                       if (expon <= -DBL_FRACBITS)
+                               return underflow(val);
                        mul.dblu_d = 0.0;
                        mul.dblu_dbl.dbl_exp = expon + DBL_EXP_BIAS;
                        u.dblu_d *= mul.dblu_d;
-                       if (u.dblu_d == 0.0) {
-                               errno = ERANGE;
-                               return (val < 0.0 ? -0.0 : 0.0);
-                       }
+                       if (u.dblu_d == 0.0)
+                               return underflow(val);
                        return (u.dblu_d);
                } else {
                        /*
@@ -105,20 +124,14 @@
                /*
                 * The result overflowed; return +/-Inf.
                 */
-               u.dblu_dbl.dbl_exp = DBL_EXP_INFNAN;
-               u.dblu_dbl.dbl_frach = 0;
-               u.dblu_dbl.dbl_fracl = 0;
-               errno = ERANGE;
-               return (u.dblu_d);
+               return overflow(val);
        } else if (newexp <= 0) {
                /*
                 * The output number is either denormal or underflows (see
                 * comments in machine/ieee.h).
                 */
-               if (newexp <= -DBL_FRACBITS) {
-                       errno = ERANGE;
-                       return (val < 0.0 ? -0.0 : 0.0);
-               }
+               if (newexp <= -DBL_FRACBITS)
+                       return underflow(val);
                /*
                 * Denormalize the result.  We do this with a multiply.  If
                 * expon is very large, it won't fit in a double, so we have



Home | Main Index | Thread Index | Old Index