/* Functions for tracking which floating-point exceptions have occurred. 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//{fclrexcpt.c,fraiseexcpt.c,ftestexcept.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) int feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; # if defined _MSC_VER exceptions = exceptions_to_x86hardware (exceptions); /* Clear the bits only in the SSE unit. */ unsigned int mxcsr, orig_mxcsr; _FPU_GETSSECW (orig_mxcsr); mxcsr = orig_mxcsr & ~exceptions; if (mxcsr != orig_mxcsr) _FPU_SETSSECW (mxcsr); # else /* Clear the bits in the 387 unit. */ x86_387_fenv_t env; __asm__ __volatile__ ("fnstenv %0" : "=m" (*&env)); /* Note: fnstenv masks all floating-point exceptions until the fldenv below. */ env.__status_word &= ~exceptions; __asm__ __volatile__ ("fldenv %0" : : "m" (*&env)); if (CPU_HAS_SSE ()) { /* Clear the bits in the SSE unit as well. */ unsigned int mxcsr, orig_mxcsr; _FPU_GETSSECW (orig_mxcsr); mxcsr = orig_mxcsr & ~exceptions; if (mxcsr != orig_mxcsr) _FPU_SETSSECW (mxcsr); } # endif return 0; } # elif defined __aarch64__ /* arm64 */ int feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned long fpsr, orig_fpsr; _FPU_GETFPSR (orig_fpsr); fpsr = orig_fpsr & ~exceptions; if (fpsr != orig_fpsr) _FPU_SETFPSR (fpsr); return 0; } # elif defined __arm__ int feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; # ifdef __SOFTFP__ if (exceptions != 0) return -1; # else unsigned int fpscr, orig_fpscr; _FPU_GETCW (orig_fpscr); fpscr = orig_fpscr & ~exceptions; if (fpscr != orig_fpscr) _FPU_SETCW (fpscr); # endif return 0; } # elif defined __alpha int feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned long swcr, orig_swcr; orig_swcr = __ieee_get_fp_control (); swcr = orig_swcr & ~exceptions; if (swcr != orig_swcr) __ieee_set_fp_control (swcr); return 0; } # elif defined __hppa int feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; 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"); unsigned int old_halfreg0 = s.halfreg[0]; /* Clear all the relevant bits. */ s.halfreg[0] &= ~ ((unsigned int) exceptions << 27); if (s.halfreg[0] != old_halfreg0) { /* Store the new status word. */ __asm__ __volatile__ ("fldd 0(%0),%%fr0" : : "r" (&s.fpreg), "m" (s.fpreg) : "%r0"); } return 0; } # elif defined __ia64__ int feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned long fpsr, orig_fpsr; _FPU_GETCW (orig_fpsr); fpsr = orig_fpsr & ~ (unsigned long) (exceptions << 13); if (fpsr != orig_fpsr) _FPU_SETCW (fpsr); return 0; } # elif defined __m68k__ int feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned int fpsr, orig_fpsr; _FPU_GETFPSR (orig_fpsr); fpsr = orig_fpsr & ~ exceptions; if (fpsr != orig_fpsr) _FPU_SETFPSR (fpsr); return 0; } # elif defined __mips__ int feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; /* Clear also the cause bits. If the cause bit is not cleared, the next CTC instruction (just below) will re-generate the exception. */ unsigned int fcsr, orig_fcsr; _FPU_GETCW (orig_fcsr); fcsr = orig_fcsr & ~ ((exceptions << 10) | exceptions); if (fcsr != orig_fcsr) _FPU_SETCW (fcsr); return 0; } # elif defined __loongarch__ int feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; /* Clear also the cause bits. If the cause bit is not cleared, the next CTC instruction (just below) will re-generate the exception. */ unsigned int fcsr, orig_fcsr; _FPU_GETCW (orig_fcsr); fcsr = orig_fcsr & ~ ((exceptions << 8) | exceptions); if (fcsr != orig_fcsr) _FPU_SETCW (fcsr); return 0; } # elif defined __powerpc__ int feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; union { unsigned long long u; double f; } memenv, orig_memenv; _FPU_GETCW_AS_DOUBLE (memenv.f); orig_memenv = memenv; /* Instead of clearing FE_INVALID (= bit 29), we need to clear the individual bits. */ memenv.u &= ~ (exceptions & FE_INVALID ? (exceptions & ~FE_INVALID) | 0x01F80700U : exceptions); if (!(memenv.u == orig_memenv.u)) _FPU_SETCW_AS_DOUBLE (memenv.f); return 0; } # elif defined __riscv int feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; __asm__ __volatile__ ("csrc fflags, %0" : : "r" (exceptions)); return 0; } # elif defined __s390__ || defined __s390x__ int feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned int fpc, orig_fpc; _FPU_GETCW (orig_fpc); # if FE_INEXACT == 8 /* glibc compatible FE_* values */ fpc = orig_fpc & ~(exceptions << 16); if ((fpc & 0x00000300) == 0) fpc &= ~(exceptions << 8); # else /* musl libc compatible FE_* values */ fpc = orig_fpc & ~exceptions; if ((fpc & 0x00000300) == 0) fpc &= ~(exceptions >> 8); # endif if (fpc != orig_fpc) _FPU_SETCW (fpc); return 0; } # elif defined __sh__ int feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned int fpscr, orig_fpscr; _FPU_GETCW (orig_fpscr); fpscr = orig_fpscr & ~exceptions; if (fpscr != orig_fpscr) _FPU_SETCW (fpscr); return 0; } # elif defined __sparc int feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned long fsr, orig_fsr; _FPU_GETCW (orig_fsr); # if FE_INEXACT == 32 /* glibc compatible FE_* values */ fsr = orig_fsr & ~exceptions; # else /* Solaris compatible FE_* values */ fsr = orig_fsr & ~(exceptions << 5); # endif if (fsr != orig_fsr) _FPU_SETCW (fsr); 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 feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; fp_except_t flags, orig_flags; orig_flags = fpgetsticky (); flags = orig_flags & ~exceptions; if (flags != orig_flags) fpsetsticky (flags); return 0; } # elif defined _AIX && defined __powerpc__ /* AIX */ # include # include /* Documentation: */ int feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; /* In addition to clearing FE_INVALID (= bit 29), we also need to clear the individual bits. */ fpflag_t f_to_clear = exceptions_to_fpflag (exceptions) | (exceptions & FE_INVALID ? 0x01F80700U : 0); if (f_to_clear != 0) fp_clr_flag (f_to_clear); return 0; } # else # define NEED_FALLBACK 1 # endif #endif #if NEED_FALLBACK /* A dummy fallback. */ int feclearexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; if (exceptions != 0) return -1; return 0; } #endif