55 lines
1.2 KiB
C++
55 lines
1.2 KiB
C++
|
|
#include "config.h"
|
|
|
|
#include "fpu_ctrl.h"
|
|
|
|
#ifdef HAVE_INTRIN_H
|
|
#include <intrin.h>
|
|
#endif
|
|
#ifdef HAVE_SSE_INTRINSICS
|
|
#include <xmmintrin.h>
|
|
#endif
|
|
|
|
#include "cpu_caps.h"
|
|
|
|
|
|
FPUCtl::FPUCtl()
|
|
{
|
|
#if defined(HAVE_SSE_INTRINSICS)
|
|
this->sse_state = _mm_getcsr();
|
|
unsigned int sseState = this->sse_state;
|
|
sseState |= 0x8000; /* set flush-to-zero */
|
|
sseState |= 0x0040; /* set denormals-are-zero */
|
|
_mm_setcsr(sseState);
|
|
|
|
#elif defined(__GNUC__) && defined(HAVE_SSE)
|
|
|
|
if((CPUCapFlags&CPU_CAP_SSE))
|
|
{
|
|
__asm__ __volatile__("stmxcsr %0" : "=m" (*&this->sse_state));
|
|
unsigned int sseState = this->sse_state;
|
|
sseState |= 0x8000; /* set flush-to-zero */
|
|
if((CPUCapFlags&CPU_CAP_SSE2))
|
|
sseState |= 0x0040; /* set denormals-are-zero */
|
|
__asm__ __volatile__("ldmxcsr %0" : : "m" (*&sseState));
|
|
}
|
|
#endif
|
|
|
|
this->in_mode = true;
|
|
}
|
|
|
|
void FPUCtl::leave()
|
|
{
|
|
if(!this->in_mode) return;
|
|
|
|
#if defined(HAVE_SSE_INTRINSICS)
|
|
_mm_setcsr(this->sse_state);
|
|
|
|
#elif defined(__GNUC__) && defined(HAVE_SSE)
|
|
|
|
if((CPUCapFlags&CPU_CAP_SSE))
|
|
__asm__ __volatile__("ldmxcsr %0" : : "m" (*&this->sse_state));
|
|
#endif
|
|
this->in_mode = false;
|
|
}
|