Blame sysdeps/i386/fpu/fesetenv.c

Packit 6c4009
/* Install given floating-point environment.
Packit 6c4009
   Copyright (C) 1997-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
   Contributed by Ulrich Drepper <drepper@cygnus.com>, 1997.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <fenv.h>
Packit 6c4009
#include <fpu_control.h>
Packit 6c4009
#include <assert.h>
Packit 6c4009
#include <unistd.h>
Packit 6c4009
#include <ldsodefs.h>
Packit 6c4009
#include <dl-procinfo.h>
Packit 6c4009
Packit 6c4009
Packit 6c4009
/* All exceptions, including the x86-specific "denormal operand"
Packit 6c4009
   exception.  */
Packit 6c4009
#define FE_ALL_EXCEPT_X86 (FE_ALL_EXCEPT | __FE_DENORM)
Packit 6c4009
Packit 6c4009
Packit 6c4009
int
Packit 6c4009
__fesetenv (const fenv_t *envp)
Packit 6c4009
{
Packit 6c4009
  fenv_t temp;
Packit 6c4009
Packit 6c4009
  /* The memory block used by fstenv/fldenv has a size of 28 bytes.  */
Packit 6c4009
  assert (sizeof (fenv_t) == 28);
Packit 6c4009
Packit 6c4009
  /* Install the environment specified by ENVP.  But there are a few
Packit 6c4009
     values which we do not want to come from the saved environment.
Packit 6c4009
     Therefore, we get the current environment and replace the values
Packit 6c4009
     we want to use from the environment specified by the parameter.  */
Packit 6c4009
  __asm__ ("fnstenv %0" : "=m" (*&temp));
Packit 6c4009
Packit 6c4009
  if (envp == FE_DFL_ENV)
Packit 6c4009
    {
Packit 6c4009
      temp.__control_word |= FE_ALL_EXCEPT_X86;
Packit 6c4009
      temp.__control_word &= ~FE_TOWARDZERO;
Packit 6c4009
      temp.__control_word |= _FPU_EXTENDED;
Packit 6c4009
      temp.__status_word &= ~FE_ALL_EXCEPT_X86;
Packit 6c4009
    }
Packit 6c4009
  else if (envp == FE_NOMASK_ENV)
Packit 6c4009
    {
Packit 6c4009
      temp.__control_word &= ~(FE_ALL_EXCEPT | FE_TOWARDZERO);
Packit 6c4009
      /* Keep the "denormal operand" exception masked.  */
Packit 6c4009
      temp.__control_word |= __FE_DENORM;
Packit 6c4009
      temp.__control_word |= _FPU_EXTENDED;
Packit 6c4009
      temp.__status_word &= ~FE_ALL_EXCEPT_X86;
Packit 6c4009
    }
Packit 6c4009
  else
Packit 6c4009
    {
Packit 6c4009
      temp.__control_word &= ~(FE_ALL_EXCEPT_X86
Packit 6c4009
			       | FE_TOWARDZERO
Packit 6c4009
			       | _FPU_EXTENDED);
Packit 6c4009
      temp.__control_word |= (envp->__control_word
Packit 6c4009
			      & (FE_ALL_EXCEPT_X86
Packit 6c4009
				 | FE_TOWARDZERO
Packit 6c4009
				 | _FPU_EXTENDED));
Packit 6c4009
      temp.__status_word &= ~FE_ALL_EXCEPT_X86;
Packit 6c4009
      temp.__status_word |= envp->__status_word & FE_ALL_EXCEPT_X86;
Packit 6c4009
    }
Packit 6c4009
  temp.__eip = 0;
Packit 6c4009
  temp.__cs_selector = 0;
Packit 6c4009
  temp.__opcode = 0;
Packit 6c4009
  temp.__data_offset = 0;
Packit 6c4009
  temp.__data_selector = 0;
Packit 6c4009
Packit 6c4009
  __asm__ ("fldenv %0" : : "m" (temp));
Packit 6c4009
Packit Service a086f9
  if (CPU_FEATURE_USABLE (SSE))
Packit 6c4009
    {
Packit 6c4009
      unsigned int mxcsr;
Packit 6c4009
      __asm__ ("stmxcsr %0" : "=m" (mxcsr));
Packit 6c4009
Packit 6c4009
      if (envp == FE_DFL_ENV)
Packit 6c4009
	{
Packit 6c4009
	  /* Clear SSE exceptions.  */
Packit 6c4009
	  mxcsr &= ~FE_ALL_EXCEPT_X86;
Packit 6c4009
	  /* Set mask for SSE MXCSR.  */
Packit 6c4009
	  mxcsr |= (FE_ALL_EXCEPT_X86 << 7);
Packit 6c4009
	  /* Set rounding to FE_TONEAREST.  */
Packit 6c4009
	  mxcsr &= ~0x6000;
Packit 6c4009
	  mxcsr |= (FE_TONEAREST << 3);
Packit 6c4009
	  /* Clear the FZ and DAZ bits.  */
Packit 6c4009
	  mxcsr &= ~0x8040;
Packit 6c4009
	}
Packit 6c4009
      else if (envp == FE_NOMASK_ENV)
Packit 6c4009
	{
Packit 6c4009
	  /* Clear SSE exceptions.  */
Packit 6c4009
	  mxcsr &= ~FE_ALL_EXCEPT_X86;
Packit 6c4009
	  /* Do not mask exceptions.  */
Packit 6c4009
	  mxcsr &= ~(FE_ALL_EXCEPT << 7);
Packit 6c4009
	  /* Keep the "denormal operand" exception masked.  */
Packit 6c4009
	  mxcsr |= (__FE_DENORM << 7);
Packit 6c4009
	  /* Set rounding to FE_TONEAREST.  */
Packit 6c4009
	  mxcsr &= ~0x6000;
Packit 6c4009
	  mxcsr |= (FE_TONEAREST << 3);
Packit 6c4009
	  /* Clear the FZ and DAZ bits.  */
Packit 6c4009
	  mxcsr &= ~0x8040;
Packit 6c4009
	}
Packit 6c4009
      else
Packit 6c4009
	mxcsr = envp->__eip;
Packit 6c4009
Packit 6c4009
      __asm__ ("ldmxcsr %0" : : "m" (mxcsr));
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  /* Success.  */
Packit 6c4009
  return 0;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
#include <shlib-compat.h>
Packit 6c4009
#if SHLIB_COMPAT (libm, GLIBC_2_1, GLIBC_2_2)
Packit 6c4009
strong_alias (__fesetenv, __old_fesetenv)
Packit 6c4009
compat_symbol (libm, __old_fesetenv, fesetenv, GLIBC_2_1);
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
libm_hidden_def (__fesetenv)
Packit 6c4009
libm_hidden_ver (__fesetenv, fesetenv)
Packit 6c4009
versioned_symbol (libm, __fesetenv, fesetenv, GLIBC_2_2);