2002-06-13 10:20:48 +00:00
|
|
|
#include <fenv.h>
|
2006-07-03 10:32:58 +00:00
|
|
|
#include "cpu_features.h"
|
|
|
|
|
2002-06-13 10:20:48 +00:00
|
|
|
/* 7.6.3.2
|
|
|
|
The fesetround function establishes the rounding direction
|
|
|
|
represented by its argument round. If the argument is not equal
|
|
|
|
to the value of a rounding direction macro, the rounding direction
|
|
|
|
is not changed. */
|
|
|
|
|
|
|
|
int fesetround (int mode)
|
|
|
|
{
|
|
|
|
unsigned short _cw;
|
|
|
|
if ((mode & ~(FE_TONEAREST | FE_DOWNWARD | FE_UPWARD | FE_TOWARDZERO))
|
|
|
|
!= 0)
|
|
|
|
return -1;
|
|
|
|
__asm__ volatile ("fnstcw %0;": "=m" (_cw));
|
|
|
|
_cw &= ~(FE_TONEAREST | FE_DOWNWARD | FE_UPWARD | FE_TOWARDZERO);
|
|
|
|
_cw |= mode;
|
|
|
|
__asm__ volatile ("fldcw %0;" : : "m" (_cw));
|
2006-07-03 10:32:58 +00:00
|
|
|
|
|
|
|
if (__HAS_SSE)
|
|
|
|
{
|
2007-03-05 07:42:36 +00:00
|
|
|
unsigned int _mxcsr;
|
|
|
|
__asm__ volatile ("stmxcsr %0" : "=m" (_mxcsr));
|
|
|
|
_mxcsr &= ~ 0x6000;
|
|
|
|
_mxcsr |= (mode << __MXCSR_ROUND_FLAG_SHIFT);
|
|
|
|
__asm__ volatile ("ldmxcsr %0" : : "m" (_mxcsr));
|
2006-07-03 10:32:58 +00:00
|
|
|
}
|
|
|
|
|
2002-06-13 10:20:48 +00:00
|
|
|
return 0;
|
|
|
|
}
|