2012-03-07 07:41:14 +08:00
|
|
|
#ifndef X86_64_MATH_PRIVATE_H
|
|
|
|
#define X86_64_MATH_PRIVATE_H 1
|
2007-04-17 04:41:42 +08:00
|
|
|
|
2009-08-25 05:52:49 +08:00
|
|
|
/* We can do a few things better on x86-64. */
|
|
|
|
|
2012-01-29 03:48:46 +08:00
|
|
|
#if defined __AVX__ || defined SSE2AVX
|
2011-10-25 20:17:57 +08:00
|
|
|
# define MOVD "vmovd"
|
|
|
|
#else
|
|
|
|
# define MOVD "movd"
|
|
|
|
#endif
|
|
|
|
|
2009-08-25 09:05:48 +08:00
|
|
|
/* Direct movement of float into integer register. */
|
2011-10-25 20:17:57 +08:00
|
|
|
#define EXTRACT_WORDS64(i, d) \
|
|
|
|
do { \
|
2012-03-20 06:09:58 +08:00
|
|
|
int64_t i_; \
|
2011-10-25 20:17:57 +08:00
|
|
|
asm (MOVD " %1, %0" : "=rm" (i_) : "x" ((double) (d))); \
|
|
|
|
(i) = i_; \
|
|
|
|
} while (0)
|
2009-08-25 09:05:48 +08:00
|
|
|
|
|
|
|
/* And the reverse. */
|
2011-10-25 20:17:57 +08:00
|
|
|
#define INSERT_WORDS64(d, i) \
|
|
|
|
do { \
|
2012-03-20 06:09:58 +08:00
|
|
|
int64_t i_ = i; \
|
2011-10-25 20:17:57 +08:00
|
|
|
double d__; \
|
|
|
|
asm (MOVD " %1, %0" : "=x" (d__) : "rm" (i_)); \
|
|
|
|
d = d__; \
|
|
|
|
} while (0)
|
2009-08-25 09:05:48 +08:00
|
|
|
|
2009-08-25 05:52:49 +08:00
|
|
|
/* Direct movement of float into integer register. */
|
2011-10-25 20:17:57 +08:00
|
|
|
#define GET_FLOAT_WORD(i, d) \
|
|
|
|
do { \
|
|
|
|
int i_; \
|
|
|
|
asm (MOVD " %1, %0" : "=rm" (i_) : "x" ((float) (d))); \
|
|
|
|
(i) = i_; \
|
|
|
|
} while (0)
|
2009-08-25 05:52:49 +08:00
|
|
|
|
|
|
|
/* And the reverse. */
|
2011-10-25 20:17:57 +08:00
|
|
|
#define SET_FLOAT_WORD(f, i) \
|
|
|
|
do { \
|
|
|
|
int i_ = i; \
|
|
|
|
float f__; \
|
|
|
|
asm (MOVD " %1, %0" : "=x" (f__) : "rm" (i_)); \
|
|
|
|
f = f__; \
|
|
|
|
} while (0)
|
2009-08-25 05:52:49 +08:00
|
|
|
|
2012-03-19 06:58:00 +08:00
|
|
|
#include <sysdeps/i386/fpu/fenv_private.h>
|
2012-03-10 04:38:23 +08:00
|
|
|
#include_next <math_private.h>
|
2011-10-12 23:27:51 +08:00
|
|
|
|
2012-03-07 07:41:14 +08:00
|
|
|
extern __always_inline double
|
|
|
|
__ieee754_sqrt (double d)
|
|
|
|
{
|
|
|
|
double res;
|
2012-01-29 03:48:46 +08:00
|
|
|
#if defined __AVX__ || defined SSE2AVX
|
2012-03-07 07:41:14 +08:00
|
|
|
asm ("vsqrtsd %1, %0, %0" : "=x" (res) : "xm" (d));
|
2011-10-25 20:17:57 +08:00
|
|
|
#else
|
2012-03-07 07:41:14 +08:00
|
|
|
asm ("sqrtsd %1, %0" : "=x" (res) : "xm" (d));
|
2011-10-25 20:17:57 +08:00
|
|
|
#endif
|
2012-03-07 07:41:14 +08:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern __always_inline float
|
|
|
|
__ieee754_sqrtf (float d)
|
|
|
|
{
|
|
|
|
float res;
|
|
|
|
#if defined __AVX__ || defined SSE2AVX
|
|
|
|
asm ("vsqrtss %1, %0, %0" : "=x" (res) : "xm" (d));
|
|
|
|
#else
|
|
|
|
asm ("sqrtss %1, %0" : "=x" (res) : "xm" (d));
|
|
|
|
#endif
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern __always_inline long double
|
|
|
|
__ieee754_sqrtl (long double d)
|
|
|
|
{
|
|
|
|
long double res;
|
|
|
|
asm ("fsqrt" : "=t" (res) : "0" (d));
|
|
|
|
return res;
|
|
|
|
}
|
2011-10-17 23:23:40 +08:00
|
|
|
|
|
|
|
#ifdef __SSE4_1__
|
2012-03-07 07:58:51 +08:00
|
|
|
extern __always_inline double
|
|
|
|
__rint (double d)
|
|
|
|
{
|
|
|
|
double res;
|
|
|
|
# if defined __AVX__ || defined SSE2AVX
|
|
|
|
asm ("vroundsd $4, %1, %0, %0" : "=x" (res) : "xm" (d));
|
|
|
|
# else
|
|
|
|
asm ("roundsd $4, %1, %0" : "=x" (res) : "xm" (d));
|
2011-10-17 23:23:40 +08:00
|
|
|
# endif
|
2012-03-07 07:58:51 +08:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern __always_inline float
|
|
|
|
__rintf (float d)
|
|
|
|
{
|
|
|
|
float res;
|
|
|
|
# if defined __AVX__ || defined SSE2AVX
|
|
|
|
asm ("vroundss $4, %1, %0, %0" : "=x" (res) : "xm" (d));
|
|
|
|
# else
|
|
|
|
asm ("roundss $4, %1, %0" : "=x" (res) : "xm" (d));
|
2011-10-17 23:23:40 +08:00
|
|
|
# endif
|
2012-03-07 07:58:51 +08:00
|
|
|
return res;
|
|
|
|
}
|
2011-10-17 23:23:40 +08:00
|
|
|
|
2012-03-07 07:58:51 +08:00
|
|
|
extern __always_inline double
|
|
|
|
__floor (double d)
|
|
|
|
{
|
|
|
|
double res;
|
|
|
|
# if defined __AVX__ || defined SSE2AVX
|
|
|
|
asm ("vroundsd $1, %1, %0, %0" : "=x" (res) : "xm" (d));
|
|
|
|
# else
|
|
|
|
asm ("roundsd $1, %1, %0" : "=x" (res) : "xm" (d));
|
2011-10-17 23:23:40 +08:00
|
|
|
# endif
|
2012-03-07 07:58:51 +08:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern __always_inline float
|
|
|
|
__floorf (float d)
|
|
|
|
{
|
|
|
|
float res;
|
|
|
|
# if defined __AVX__ || defined SSE2AVX
|
|
|
|
asm ("vroundss $1, %1, %0, %0" : "=x" (res) : "xm" (d));
|
|
|
|
# else
|
|
|
|
asm ("roundss $1, %1, %0" : "=x" (res) : "xm" (d));
|
2011-10-25 20:17:57 +08:00
|
|
|
# endif
|
2012-03-07 07:58:51 +08:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
#endif /* __SSE4_1__ */
|
2011-10-18 21:00:46 +08:00
|
|
|
|
2012-03-07 07:41:14 +08:00
|
|
|
#endif /* X86_64_MATH_PRIVATE_H */
|