/* Functions for saving the floating-point exception status flags.
Copyright (C) 1997-2024 Free Software Foundation, Inc.
This file is free software: you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as
published by the Free Software Foundation; either version 2.1 of the
License, or (at your option) any later version.
This file is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with this program. If not, see . */
/* Based on glibc/sysdeps//{fgetexcptflg.c,fsetexcptflg.c}
together with glibc/sysdeps//{fpu_control.h,fenv_private.h,fenv_libc.h}. */
#include
/* Specification. */
#include
#include "fenv-private.h"
#if defined __GNUC__ || defined __clang__ || defined _MSC_VER
# if (defined __x86_64__ || defined _M_X64) || (defined __i386 || defined _M_IX86)
/* On most OSes, fexcept_t is binary-equivalent to an 'unsigned short'.
On NetBSD, OpenBSD, Solaris, Cygwin, MSVC, Android/x86_64, Minix, fexcept_t
is equivalent to an 'unsigned int'.
A simple C cast does the necessary conversion. */
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
# if defined _MSC_VER
unsigned int mxcsr;
_FPU_GETSSECW (mxcsr);
*saved_flags = x86hardware_to_exceptions (mxcsr) & FE_ALL_EXCEPT & exceptions;
# else
unsigned short fstat;
_FPU_GETSTAT (fstat);
unsigned int mxcsr = 0;
if (CPU_HAS_SSE ())
{
/* Look at the flags in the SSE unit as well. */
_FPU_GETSSECW (mxcsr);
}
*saved_flags = x86hardware_to_exceptions (fstat | mxcsr)
& FE_ALL_EXCEPT & exceptions;
# endif
return 0;
}
# elif defined __aarch64__ /* arm64 */
/* On Linux, NetBSD, and Android, fexcept_t is binary-equivalent to
an 'unsigned int'.
On macOS, fexcept_t is binary-equivalent to an 'unsigned short'.
On FreeBSD and OpenBSD, fexcept_t is binary-equivalent to an 'unsigned long'.
A simple C cast does the necessary conversion. */
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
unsigned long fpsr;
_FPU_GETFPSR (fpsr);
*saved_flags = fpsr & FE_ALL_EXCEPT & exceptions;
return 0;
}
# elif defined __arm__
/* On all OSes, fexcept_t is binary-equivalent to an 'unsigned int'. */
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
# ifdef __SOFTFP__
return -1;
# else
unsigned int fpscr;
_FPU_GETCW (fpscr);
*saved_flags = fpscr & FE_ALL_EXCEPT & exceptions;
return 0;
# endif
}
# elif defined __alpha
/* On all OSes except NetBSD and OpenBSD, fexcept_t is binary-equivalent to
an 'unsigned long'.
On NetBSD, it is equivalent to an 'unsigned short'.
On OpenBSD, it is equivalent to an 'unsigned int'.
A simple C cast does the necessary conversion. */
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
unsigned long swcr = __ieee_get_fp_control ();
*saved_flags = swcr & FE_ALL_EXCEPT & exceptions;
return 0;
}
# elif defined __hppa
/* On all OSes, fexcept_t is binary-equivalent to an 'unsigned int'. */
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
union { unsigned long long fpreg; unsigned int halfreg[2]; } s;
/* Get the current status word. */
__asm__ __volatile__ ("fstd %%fr0,0(%1)" : "=m" (s.fpreg) : "r" (&s.fpreg) : "%r0");
*saved_flags = (s.halfreg[0] >> 27) & FE_ALL_EXCEPT & exceptions;
return 0;
}
# elif defined __ia64__
/* On all OSes except NetBSD, fexcept_t is binary-equivalent to
an 'unsigned long'.
On NetBSD, it is equivalent to an 'unsigned short'.
A simple C cast does the necessary conversion. */
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
unsigned long fpsr;
_FPU_GETCW (fpsr);
*saved_flags = (fpsr >> 13) & FE_ALL_EXCEPT & exceptions;
return 0;
}
# elif defined __m68k__
/* On all OSes, fexcept_t is binary-equivalent to an 'unsigned int'. */
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
unsigned int fpsr;
_FPU_GETFPSR (fpsr);
*saved_flags = fpsr & FE_ALL_EXCEPT & exceptions;
return 0;
}
# elif defined __mips__
/* On all OSes except NetBSD and OpenBSD, fexcept_t is binary-equivalent to
an 'unsigned short'.
On NetBSD and OpenBSD, it is equivalent to an 'unsigned int'.
A simple C cast does the necessary conversion. */
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
unsigned int fcsr;
_FPU_GETCW (fcsr);
*saved_flags = fcsr & FE_ALL_EXCEPT & exceptions;
return 0;
}
# elif defined __loongarch__
/* On all OSes, fexcept_t is binary-equivalent to an 'unsigned int'. */
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
unsigned int fcsr;
_FPU_GETCW (fcsr);
*saved_flags = fcsr & FE_ALL_EXCEPT & exceptions;
return 0;
}
# elif defined __powerpc__
/* On all OSes, fexcept_t is binary-equivalent to an 'unsigned int'. */
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
union { unsigned long long u; double f; } memenv;
_FPU_GETCW_AS_DOUBLE (memenv.f);
*saved_flags = memenv.u & FE_ALL_EXCEPT & exceptions;
return 0;
}
# elif defined __riscv
/* On all OSes except FreeBSD, fexcept_t is binary-equivalent to
an 'unsigned int'.
On FreeBSD, it is equivalent to an 'unsigned long'.
A simple C cast does the necessary conversion. */
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
unsigned int flags;
__asm__ __volatile__ ("frflags %0" : "=r" (flags)); /* same as "csrr %0, fflags" */
*saved_flags = flags & FE_ALL_EXCEPT & exceptions;
return 0;
}
# elif defined __s390__ || defined __s390x__
/* On all OSes, fexcept_t is binary-equivalent to an 'unsigned int'. */
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
unsigned int fpc;
_FPU_GETCW (fpc);
# if FE_INEXACT == 8 /* glibc compatible FE_* values */
*saved_flags = ((fpc >> 16) | ((fpc & 0x00000300) == 0 ? fpc >> 8 : 0))
& FE_ALL_EXCEPT & exceptions;
# else /* musl libc compatible FE_* values */
*saved_flags = (fpc | ((fpc & 0x00000300) == 0 ? fpc << 8 : 0))
& FE_ALL_EXCEPT & exceptions;
# endif
return 0;
}
# elif defined __sh__
/* On glibc, fexcept_t is binary-equivalent to an 'unsigned short'.
On all other OSes, fexcept_t is binary-equivalent to an 'unsigned int'.
A simple C cast does the necessary conversion. */
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
unsigned int fpscr;
_FPU_GETCW (fpscr);
*saved_flags = fpscr & FE_ALL_EXCEPT & exceptions;
return 0;
}
# elif defined __sparc
/* On all OSes except Solaris, fexcept_t is binary-equivalent to an 'unsigned long'.
On Solaris, fexcept_t is an 'int'.
A simple C cast does the necessary conversion. */
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
unsigned long fsr;
_FPU_GETCW (fsr);
# if FE_INEXACT == 32 /* glibc compatible FE_* values */
*saved_flags = fsr & FE_ALL_EXCEPT & exceptions;
# else /* Solaris compatible FE_* values */
*saved_flags = (fsr >> 5) & FE_ALL_EXCEPT & exceptions;
# endif
return 0;
}
# else
# if defined __GNUC__ || defined __clang__
# warning "Unknown CPU / architecture. Please report your platform and compiler to ."
# endif
# define NEED_FALLBACK 1
# endif
#else
/* The compiler does not support __asm__ statements or equivalent
intrinsics. */
# if HAVE_FPSETSTICKY
/* FreeBSD ≥ 3.1, NetBSD ≥ 1.1, OpenBSD, IRIX, Solaris, Minix ≥ 3.2. */
/* Get fpgetsticky, fpsetsticky. */
# include
/* The type is called 'fp_except_t' on FreeBSD, but 'fp_except' on
all other systems. */
# if !defined __FreeBSD__
# define fp_except_t fp_except
# endif
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
fp_except_t flags = fpgetsticky ();
*saved_flags = flags & FE_ALL_EXCEPT & exceptions;
return 0;
}
# elif defined _AIX && defined __powerpc__ /* AIX */
# include
# include
/* Documentation:
*/
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
fpflag_t flags = fp_read_flag ();
*saved_flags = fpflag_to_exceptions (flags) & FE_ALL_EXCEPT & exceptions;
return 0;
}
# else
# define NEED_FALLBACK 1
# endif
#endif
#if NEED_FALLBACK
/* A dummy fallback. */
int
fegetexceptflag (fexcept_t *saved_flags, int exceptions)
{
/* Just like fetestexcept. */
*saved_flags = 0;
return 0;
}
#endif