summaryrefslogtreecommitdiffstats
path: root/lib/msun/src/math_private.h
diff options
context:
space:
mode:
authorkargl <kargl@FreeBSD.org>2012-07-23 19:13:55 +0000
committerkargl <kargl@FreeBSD.org>2012-07-23 19:13:55 +0000
commitb4194dfd5548262d72f4d521bb5895aa65dff172 (patch)
treed7a34dad173a32ea918010c172031c83f4305093 /lib/msun/src/math_private.h
parent32ec1fa0f59002b046c9dad4e046eb05da10e8cc (diff)
downloadFreeBSD-src-b4194dfd5548262d72f4d521bb5895aa65dff172.zip
FreeBSD-src-b4194dfd5548262d72f4d521bb5895aa65dff172.tar.gz
Compute the exponential of x for Intel 80-bit format and IEEE 128-bit
format. These implementations are based on PTP Tang, "Table-driven implementation of the exponential function in IEEE floating-point arithmetic," ACM Trans. Math. Soft., 15, 144-157 (1989). PR: standards/152415 Submitted by: kargl Reviewed by: bde, das Approved by: das (mentor)
Diffstat (limited to 'lib/msun/src/math_private.h')
-rw-r--r--lib/msun/src/math_private.h41
1 files changed, 41 insertions, 0 deletions
diff --git a/lib/msun/src/math_private.h b/lib/msun/src/math_private.h
index 79280e3..94e7507 100644
--- a/lib/msun/src/math_private.h
+++ b/lib/msun/src/math_private.h
@@ -207,6 +207,13 @@ do { \
(d) = se_u.e; \
} while (0)
+/* Long double constants are broken on i386. This workaround is OK always. */
+#define LD80C(m, ex, s, v) { \
+ /* .e = v, */ /* overwritten */ \
+ .xbits.man = __CONCAT(m, ULL), \
+ .xbits.expsign = (0x3fff + (ex)) | ((s) ? 0x8000 : 0), \
+}
+
#ifdef FLT_EVAL_METHOD
/*
* Attempt to get strict C99 semantics for assignment with non-C99 compilers.
@@ -225,8 +232,30 @@ do { \
} \
} while (0)
#endif
+#endif /* FLT_EVAL_METHOD */
+
+/* Support switching the mode to FP_PE if necessary. */
+#if defined(__i386__) && !defined(NO_FPSETPREC)
+#define ENTERI() \
+ long double __retval; \
+ fp_prec_t __oprec; \
+ \
+ if ((__oprec = fpgetprec()) != FP_PE) \
+ fpsetprec(FP_PE);
+#define RETURNI(x) do { \
+ __retval = (x); \
+ if (__oprec != FP_PE) \
+ fpsetprec(__oprec); \
+ RETURNF(__retval); \
+} while (0)
+#else
+#define ENTERI(x)
+#define RETURNI(x) RETURNF(x)
#endif
+/* Default return statement if hack*_t() is not used. */
+#define RETURNF(v) return (v)
+
/*
* Common routine to process the arguments to nan(), nanf(), and nanl().
*/
@@ -323,6 +352,18 @@ irint(double x)
#define HAVE_EFFICIENT_IRINT
#endif
+#if defined(__amd64__) || defined(__i386__)
+static __inline int
+irintl(long double x)
+{
+ int n;
+
+ asm("fistl %0" : "=m" (n) : "t" (x));
+ return (n);
+}
+#define HAVE_EFFICIENT_IRINTL
+#endif
+
#endif /* __GNUCLIKE_ASM */
/*
OpenPOWER on IntegriCloud