2007-04-16 20:41:42 +00:00
|
|
|
#ifndef _MATH_PRIVATE_H
|
|
|
|
|
|
|
|
#define math_opt_barrier(x) \
|
|
|
|
({ __typeof(x) __x; \
|
|
|
|
if (sizeof (x) <= sizeof (double)) \
|
|
|
|
__asm ("" : "=x" (__x) : "0" (x)); \
|
|
|
|
else \
|
|
|
|
__asm ("" : "=t" (__x) : "0" (x)); \
|
|
|
|
__x; })
|
|
|
|
#define math_force_eval(x) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
if (sizeof (x) <= sizeof (double)) \
|
|
|
|
__asm __volatile ("" : : "x" (x)); \
|
|
|
|
else \
|
|
|
|
__asm __volatile ("" : : "f" (x)); \
|
|
|
|
} \
|
|
|
|
while (0)
|
|
|
|
|
|
|
|
#include <math/math_private.h>
|
2009-08-24 14:52:49 -07:00
|
|
|
|
|
|
|
/* We can do a few things better on x86-64. */
|
|
|
|
|
2009-08-24 18:05:48 -07:00
|
|
|
/* Direct movement of float into integer register. */
|
|
|
|
#undef EXTRACT_WORDS64
|
|
|
|
#define EXTRACT_WORDS64(i,d) \
|
|
|
|
do { \
|
|
|
|
long int i_; \
|
|
|
|
asm ("movd %1, %0" : "=rm" (i_) : "x" (d)); \
|
|
|
|
(i) = i_; \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
/* And the reverse. */
|
|
|
|
#undef INSERT_WORDS64
|
|
|
|
#define INSERT_WORDS64(d,i) \
|
|
|
|
do { \
|
|
|
|
long int i_ = i; \
|
|
|
|
asm ("movd %1, %0" : "=x" (d) : "rm" (i_)); \
|
|
|
|
} while (0)
|
|
|
|
|
2009-08-24 14:52:49 -07:00
|
|
|
/* Direct movement of float into integer register. */
|
|
|
|
#undef GET_FLOAT_WORD
|
|
|
|
#define GET_FLOAT_WORD(i,d) \
|
|
|
|
do { \
|
|
|
|
int i_; \
|
|
|
|
asm ("movd %1, %0" : "=rm" (i_) : "x" (d)); \
|
|
|
|
(i) = i_; \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
/* And the reverse. */
|
|
|
|
#undef SET_FLOAT_WORD
|
|
|
|
#define SET_FLOAT_WORD(d,i) \
|
|
|
|
do { \
|
|
|
|
int i_ = i; \
|
|
|
|
asm ("movd %1, %0" : "=x" (d) : "rm" (i_)); \
|
|
|
|
} while (0)
|
|
|
|
|
2007-04-16 20:41:42 +00:00
|
|
|
#endif
|
2011-10-08 10:18:26 -04:00
|
|
|
|
|
|
|
#define __isnan(d) \
|
2011-10-12 11:27:51 -04:00
|
|
|
({ long int __di; EXTRACT_WORDS64 (__di, (double) d); \
|
2011-10-08 10:18:26 -04:00
|
|
|
(__di & 0x7fffffffffffffffl) > 0x7ff0000000000000l; })
|
|
|
|
#define __isnanf(d) \
|
2011-10-12 11:27:51 -04:00
|
|
|
({ int __di; GET_FLOAT_WORD (__di, (float) d); \
|
2011-10-08 10:18:26 -04:00
|
|
|
(__di & 0x7fffffff) > 0x7f800000; })
|
|
|
|
|
|
|
|
#define __isinf_ns(d) \
|
2011-10-12 11:27:51 -04:00
|
|
|
({ long int __di; EXTRACT_WORDS64 (__di, (double) d); \
|
2011-10-08 10:18:26 -04:00
|
|
|
(__di & 0x7fffffffffffffffl) == 0x7ff0000000000000l; })
|
|
|
|
#define __isinf_nsf(d) \
|
2011-10-12 11:27:51 -04:00
|
|
|
({ int __di; GET_FLOAT_WORD (__di, (float) d); \
|
2011-10-08 10:18:26 -04:00
|
|
|
(__di & 0x7fffffff) == 0x7f800000; })
|
|
|
|
|
|
|
|
#define __finite(d) \
|
2011-10-12 11:27:51 -04:00
|
|
|
({ long int __di; EXTRACT_WORDS64 (__di, (double) d); \
|
2011-10-08 10:18:26 -04:00
|
|
|
(__di & 0x7fffffffffffffffl) < 0x7ff0000000000000l; })
|
|
|
|
#define __finitef(d) \
|
2011-10-12 11:27:51 -04:00
|
|
|
({ int __di; GET_FLOAT_WORD (__di, (float) d); \
|
2011-10-08 10:18:26 -04:00
|
|
|
(__di & 0x7fffffff) < 0x7f800000; })
|
2011-10-12 11:27:51 -04:00
|
|
|
|
|
|
|
#define __ieee754_sqrt(d) \
|
|
|
|
({ double __res; \
|
|
|
|
asm ("sqrtsd %1, %0" : "=x" (__res) : "xm" ((double) d)); \
|
|
|
|
__res; })
|
|
|
|
#define __ieee754_sqrtf(d) \
|
|
|
|
({ float __res; \
|
|
|
|
asm ("sqrtss %1, %0" : "=x" (__res) : "xm" ((float) d)); \
|
|
|
|
__res; })
|
|
|
|
#define __ieee754_sqrtl(d) \
|
|
|
|
({ long double __res; \
|
|
|
|
asm ("fsqrt" : "=t" (__res) : "0" ((long double) d)); \
|
|
|
|
__res; })
|
2011-10-17 11:23:40 -04:00
|
|
|
|
|
|
|
#ifdef __SSE4_1__
|
|
|
|
# ifndef __rint
|
|
|
|
# define __rint(d) \
|
|
|
|
({ double __res; \
|
|
|
|
asm ("roundsd $4, %1, %0" : "=x" (__res) : "x" ((double) d)); \
|
|
|
|
__res; })
|
|
|
|
# endif
|
|
|
|
# ifndef __rintf
|
|
|
|
# define __rintf(d) \
|
|
|
|
({ float __res; \
|
|
|
|
asm ("roundss $4, %1, %0" : "=x" (__res) : "x" ((float) d)); \
|
|
|
|
__res; })
|
|
|
|
# endif
|
|
|
|
|
|
|
|
# ifndef __floor
|
|
|
|
# define __floor(d) \
|
|
|
|
({ double __res; \
|
|
|
|
asm ("roundsd $1, %1, %0" : "=x" (__res) : "x" ((double) d)); \
|
|
|
|
__res; })
|
|
|
|
# endif
|
|
|
|
# ifndef __floorf
|
|
|
|
# define __floorf(d) \
|
|
|
|
({ float __res; \
|
|
|
|
asm ("roundss $1, %1, %0" : "=x" (__res) : "x" ((float) d)); \
|
|
|
|
__res; })
|
|
|
|
# endif
|
|
|
|
#endif
|