diff options
author | kargl <kargl@FreeBSD.org> | 2012-07-23 19:13:55 +0000 |
---|---|---|
committer | kargl <kargl@FreeBSD.org> | 2012-07-23 19:13:55 +0000 |
commit | b4194dfd5548262d72f4d521bb5895aa65dff172 (patch) | |
tree | d7a34dad173a32ea918010c172031c83f4305093 /lib/msun/src/math_private.h | |
parent | 32ec1fa0f59002b046c9dad4e046eb05da10e8cc (diff) | |
download | FreeBSD-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.h | 41 |
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 */ /* |