/* Functions for turning floating-point exceptions into traps (signals). 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//{feenablxcpt.c,fedisblxcpt.c,fegetexcept.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 feenableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; # if defined _MSC_VER exceptions = exceptions_to_x86hardware (exceptions); /* Enable the traps only in the SSE unit. */ unsigned int mxcsr, orig_mxcsr; _FPU_GETSSECW (orig_mxcsr); mxcsr = orig_mxcsr & ~(exceptions << 7); if (mxcsr != orig_mxcsr) _FPU_SETSSECW (mxcsr); unsigned int trapbits = 0x3f & ~(orig_mxcsr >> 7); return x86hardware_to_exceptions (trapbits); # else /* Enable the traps in the 387 unit. */ unsigned short fctrl, orig_fctrl; _FPU_GETCW (orig_fctrl); fctrl = orig_fctrl & ~exceptions; if (fctrl != orig_fctrl) _FPU_SETCW (fctrl); if (CPU_HAS_SSE ()) { /* Enable the traps in the SSE unit as well. */ unsigned int mxcsr, orig_mxcsr; _FPU_GETSSECW (orig_mxcsr); mxcsr = orig_mxcsr & ~(exceptions << 7); if (mxcsr != orig_mxcsr) _FPU_SETSSECW (mxcsr); } return FE_ALL_EXCEPT & ~orig_fctrl; # endif } int fedisableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; # if defined _MSC_VER exceptions = exceptions_to_x86hardware (exceptions); /* Disable the traps only in the SSE unit. */ unsigned int mxcsr, orig_mxcsr; _FPU_GETSSECW (orig_mxcsr); mxcsr = orig_mxcsr | (exceptions << 7); if (mxcsr != orig_mxcsr) _FPU_SETSSECW (mxcsr); unsigned int trapbits = 0x3f & ~(orig_mxcsr >> 7); return x86hardware_to_exceptions (trapbits); # else /* Disable the traps in the 387 unit. */ unsigned short fctrl, orig_fctrl; _FPU_GETCW (orig_fctrl); fctrl = orig_fctrl | exceptions; if (fctrl != orig_fctrl) _FPU_SETCW (fctrl); if (CPU_HAS_SSE ()) { /* Disable the traps in the SSE unit as well. */ unsigned int mxcsr, orig_mxcsr; _FPU_GETSSECW (orig_mxcsr); mxcsr = orig_mxcsr | (exceptions << 7); if (mxcsr != orig_mxcsr) _FPU_SETSSECW (mxcsr); } return FE_ALL_EXCEPT & ~orig_fctrl; # endif } int fegetexcept (void) { # if defined _MSC_VER /* Look at the trap bits in the SSE unit. */ unsigned int mxcsr; _FPU_GETSSECW (mxcsr); unsigned int trapbits = 0x3f & ~(mxcsr >> 7); return x86hardware_to_exceptions (trapbits); # else /* Look at the trap bits in the 387 unit. */ unsigned short fctrl; _FPU_GETCW (fctrl); return FE_ALL_EXCEPT & ~fctrl; # endif } # elif defined __aarch64__ /* arm64 */ int feenableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned long fpcr, orig_fpcr; _FPU_GETCW (orig_fpcr); fpcr = orig_fpcr | (exceptions << 8); if (fpcr != orig_fpcr) { _FPU_SETCW (fpcr); /* Test whether fpcr was actually changed as desired. */ unsigned long new_fpcr; _FPU_GETCW (new_fpcr); if (new_fpcr != fpcr) return -1; } return FE_ALL_EXCEPT & (orig_fpcr >> 8); } int fedisableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned long fpcr, orig_fpcr; _FPU_GETCW (orig_fpcr); fpcr = orig_fpcr & ~(exceptions << 8); if (fpcr != orig_fpcr) { _FPU_SETCW (fpcr); /* Testing whether fpcr was actually changed as desired is not needed here, since we only cleared some bits. */ } return FE_ALL_EXCEPT & (orig_fpcr >> 8); } int fegetexcept (void) { unsigned long fpcr; _FPU_GETCW (fpcr); return FE_ALL_EXCEPT & (fpcr >> 8); } # elif defined __arm__ int feenableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; # ifdef __SOFTFP__ if (exceptions != 0) return -1; return 0; # else unsigned int fpscr, orig_fpscr; _FPU_GETCW (orig_fpscr); fpscr = orig_fpscr | (exceptions << 8); if (fpscr != orig_fpscr) { _FPU_SETCW (fpscr); /* Test whether fpscr was actually changed as desired. */ unsigned int new_fpscr; _FPU_GETCW (new_fpscr); if (new_fpscr != fpscr) return -1; } return FE_ALL_EXCEPT & (orig_fpscr >> 8); # endif } int fedisableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; # ifdef __SOFTFP__ return 0; # else unsigned int fpscr, orig_fpscr; _FPU_GETCW (orig_fpscr); fpscr = orig_fpscr & ~(exceptions << 8); if (fpscr != orig_fpscr) { _FPU_SETCW (fpscr); /* Testing whether fpscr was actually changed as desired is not needed here, since we only cleared some bits. */ } return FE_ALL_EXCEPT & (orig_fpscr >> 8); # endif } int fegetexcept (void) { # ifdef __SOFTFP__ return 0; # else unsigned int fpscr; _FPU_GETCW (fpscr); return FE_ALL_EXCEPT & (fpscr >> 8); # endif } # elif defined __alpha int feenableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned long swcr, orig_swcr; orig_swcr = __ieee_get_fp_control (); swcr = orig_swcr | ((unsigned long) exceptions >> 16); if (swcr != orig_swcr) __ieee_set_fp_control (swcr); return FE_ALL_EXCEPT & (orig_swcr << 16); } int fedisableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned long swcr, orig_swcr; orig_swcr = __ieee_get_fp_control (); swcr = orig_swcr & ~((unsigned long) exceptions >> 16); if (swcr != orig_swcr) __ieee_set_fp_control (swcr); return FE_ALL_EXCEPT & (orig_swcr << 16); } int fegetexcept (void) { unsigned long swcr = __ieee_get_fp_control (); return FE_ALL_EXCEPT & (swcr << 16); } # elif defined __hppa int feenableexcept (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]; /* Set all the relevant bits. */ s.halfreg[0] |= (unsigned int) exceptions; 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 FE_ALL_EXCEPT & old_halfreg0; } int fedisableexcept (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; 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 FE_ALL_EXCEPT & old_halfreg0; } int fegetexcept (void) { 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"); return FE_ALL_EXCEPT & s.halfreg[0]; } # elif defined __ia64__ int feenableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned long fpsr, orig_fpsr; _FPU_GETCW (orig_fpsr); fpsr = orig_fpsr & ~ (unsigned long) exceptions; if (fpsr != orig_fpsr) _FPU_SETCW (fpsr); return FE_ALL_EXCEPT & ~orig_fpsr; } int fedisableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned long fpsr, orig_fpsr; _FPU_GETCW (orig_fpsr); fpsr = orig_fpsr | (unsigned long) exceptions; if (fpsr != orig_fpsr) _FPU_SETCW (fpsr); return FE_ALL_EXCEPT & ~orig_fpsr; } int fegetexcept (void) { unsigned long fpsr; _FPU_GETCW (fpsr); return FE_ALL_EXCEPT & ~fpsr; } # elif defined __m68k__ int feenableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned int fpcr, orig_fpcr; _FPU_GETCW (orig_fpcr); fpcr = orig_fpcr | ((exceptions << 6) | (exceptions & FE_INVALID ? 0xc000 : 0)); if (fpcr != orig_fpcr) _FPU_SETCW (fpcr); return FE_ALL_EXCEPT & (orig_fpcr >> 6); } int fedisableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned int fpcr, orig_fpcr; _FPU_GETCW (orig_fpcr); fpcr = orig_fpcr & ~ ((exceptions << 6) | (exceptions & FE_INVALID ? 0xc000 : 0)); if (fpcr != orig_fpcr) _FPU_SETCW (fpcr); return FE_ALL_EXCEPT & (orig_fpcr >> 6); } int fegetexcept (void) { unsigned int fpcr; _FPU_GETCW (fpcr); return FE_ALL_EXCEPT & (fpcr >> 6); } # elif defined __mips__ int feenableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned int fcsr, orig_fcsr; _FPU_GETCW (orig_fcsr); fcsr = orig_fcsr | (exceptions << 5); if (fcsr != orig_fcsr) _FPU_SETCW (fcsr); return FE_ALL_EXCEPT & (orig_fcsr >> 5); } int fedisableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned int fcsr, orig_fcsr; _FPU_GETCW (orig_fcsr); fcsr = orig_fcsr & ~ (exceptions << 5); if (fcsr != orig_fcsr) _FPU_SETCW (fcsr); return FE_ALL_EXCEPT & (orig_fcsr >> 5); } int fegetexcept (void) { unsigned int fcsr; _FPU_GETCW (fcsr); return FE_ALL_EXCEPT & (fcsr >> 5); } # elif defined __loongarch__ int feenableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned int fcsr, orig_fcsr; _FPU_GETCW (orig_fcsr); fcsr = orig_fcsr | (exceptions >> 16); if (fcsr != orig_fcsr) _FPU_SETCW (fcsr); return FE_ALL_EXCEPT & (orig_fcsr << 16); } int fedisableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned int fcsr, orig_fcsr; _FPU_GETCW (orig_fcsr); fcsr = orig_fcsr & ~ (exceptions >> 16); if (fcsr != orig_fcsr) _FPU_SETCW (fcsr); return FE_ALL_EXCEPT & (orig_fcsr << 16); } int fegetexcept (void) { unsigned int fcsr; _FPU_GETCW (fcsr); return FE_ALL_EXCEPT & (fcsr << 16); } # elif defined __powerpc__ # if defined __linux__ # include # elif defined _AIX # include # endif int feenableexcept (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; memenv.u |= ((unsigned int) exceptions >> 22); if (!(memenv.u == orig_memenv.u)) { if ((orig_memenv.u & 0x000000f8) == 0 && (memenv.u & 0x000000f8) != 0) { /* Put the thread into precise trapping mode. */ # if defined __linux__ || defined __NetBSD__ prctl (PR_SET_FPEXC, PR_FP_EXC_PRECISE); # elif defined _AIX /* Documentation: */ fp_trap (FP_TRAP_SYNC); # endif } _FPU_SETCW_AS_DOUBLE (memenv.f); } return FE_ALL_EXCEPT & ((unsigned int) orig_memenv.u << 22); } int fedisableexcept (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; memenv.u &= ~ ((unsigned int) exceptions >> 22); if (!(memenv.u == orig_memenv.u)) { _FPU_SETCW_AS_DOUBLE (memenv.f); if ((orig_memenv.u & 0x000000f8) != 0 && (memenv.u & 0x000000f8) == 0) { /* Put the thread into no-trapping mode. */ # if defined __linux__ || defined __NetBSD__ prctl (PR_SET_FPEXC, PR_FP_EXC_DISABLED); # elif defined _AIX /* Documentation: */ fp_trap (FP_TRAP_OFF); # endif } } return FE_ALL_EXCEPT & ((unsigned int) orig_memenv.u << 22); } int fegetexcept (void) { union { unsigned long long u; double f; } memenv; _FPU_GETCW_AS_DOUBLE (memenv.f); return FE_ALL_EXCEPT & ((unsigned int) memenv.u << 22); } # elif defined __riscv int feenableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; if (exceptions != 0) return -1; return 0; } int fedisableexcept (int exceptions) { return 0; } int fegetexcept (void) { return 0; } # elif defined __s390__ || defined __s390x__ int feenableexcept (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 | ((unsigned int) exceptions << 24); # else /* musl libc compatible FE_* values */ fpc = orig_fpc | ((unsigned int) exceptions << 8); # endif if (fpc != orig_fpc) _FPU_SETCW (fpc); # if FE_INEXACT == 8 /* glibc compatible FE_* values */ return FE_ALL_EXCEPT & (orig_fpc >> 24); # else /* musl libc compatible FE_* values */ return FE_ALL_EXCEPT & (orig_fpc >> 8); # endif } int fedisableexcept (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 & ~((unsigned int) exceptions << 24); # else /* musl libc compatible FE_* values */ fpc = orig_fpc & ~((unsigned int) exceptions << 8); # endif if (fpc != orig_fpc) _FPU_SETCW (fpc); # if FE_INEXACT == 8 /* glibc compatible FE_* values */ return FE_ALL_EXCEPT & (orig_fpc >> 24); # else /* musl libc compatible FE_* values */ return FE_ALL_EXCEPT & (orig_fpc >> 8); # endif } int fegetexcept (void) { unsigned int fpc; _FPU_GETCW (fpc); # if FE_INEXACT == 8 /* glibc compatible FE_* values */ return FE_ALL_EXCEPT & (fpc >> 24); # else /* musl libc compatible FE_* values */ return FE_ALL_EXCEPT & (fpc >> 8); # endif } # elif defined __sh__ int feenableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned int fpscr, orig_fpscr; _FPU_GETCW (orig_fpscr); fpscr = orig_fpscr | ((unsigned int) exceptions << 5); if (fpscr != orig_fpscr) _FPU_SETCW (fpscr); return FE_ALL_EXCEPT & (orig_fpscr >> 5); } int fedisableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; unsigned int fpscr, orig_fpscr; _FPU_GETCW (orig_fpscr); fpscr = orig_fpscr & ~ ((unsigned int) exceptions << 5); if (fpscr != orig_fpscr) _FPU_SETCW (fpscr); return FE_ALL_EXCEPT & (orig_fpscr >> 5); } int fegetexcept (void) { unsigned int fpscr; _FPU_GETCW (fpscr); return FE_ALL_EXCEPT & (fpscr >> 5); } # elif defined __sparc int feenableexcept (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 << 18); # else /* Solaris compatible FE_* values */ fsr = orig_fsr | (exceptions << 23); # endif if (fsr != orig_fsr) _FPU_SETCW (fsr); # if FE_INEXACT == 32 /* glibc compatible FE_* values */ return FE_ALL_EXCEPT & (orig_fsr >> 18); # else /* Solaris compatible FE_* values */ return FE_ALL_EXCEPT & (orig_fsr >> 23); # endif } int fedisableexcept (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 << 18); # else /* Solaris compatible FE_* values */ fsr = orig_fsr & ~(exceptions << 23); # endif if (fsr != orig_fsr) _FPU_SETCW (fsr); # if FE_INEXACT == 32 /* glibc compatible FE_* values */ return FE_ALL_EXCEPT & (orig_fsr >> 18); # else /* Solaris compatible FE_* values */ return FE_ALL_EXCEPT & (orig_fsr >> 23); # endif } int fegetexcept (void) { unsigned long fsr; _FPU_GETCW (fsr); # if FE_INEXACT == 32 /* glibc compatible FE_* values */ return FE_ALL_EXCEPT & (fsr >> 18); # else /* Solaris compatible FE_* values */ return FE_ALL_EXCEPT & (fsr >> 23); # endif } # 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_FPSETMASK /* FreeBSD ≥ 3.1, NetBSD ≥ 1.1, OpenBSD, IRIX, Solaris, Minix ≥ 3.2. */ /* Get fpgetmask, fpsetmask. */ # 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 static fp_except_t exceptions_to_mask (int exceptions) { fp_except_t m = 0; if (exceptions & FE_INVALID) m |= FP_X_INV; if (exceptions & FE_DIVBYZERO) m |= FP_X_DZ; if (exceptions & FE_OVERFLOW) m |= FP_X_OFL; if (exceptions & FE_UNDERFLOW) m |= FP_X_UFL; if (exceptions & FE_INEXACT) m |= FP_X_IMP; return m; } static int mask_to_exceptions (fp_except_t m) { int exceptions = 0; if (m & FP_X_INV) exceptions |= FE_INVALID; if (m & FP_X_DZ) exceptions |= FE_DIVBYZERO; if (m & FP_X_OFL) exceptions |= FE_OVERFLOW; if (m & FP_X_UFL) exceptions |= FE_UNDERFLOW; if (m & FP_X_IMP) exceptions |= FE_INEXACT; return exceptions; } int feenableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; fp_except_t trapbits = fpgetmask (); fpsetmask (trapbits | exceptions_to_mask (exceptions)); return mask_to_exceptions (trapbits); } int fedisableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; fp_except_t trapbits = fpgetmask (); fpsetmask (trapbits & ~ exceptions_to_mask (exceptions)); return mask_to_exceptions (trapbits); } int fegetexcept (void) { fp_except_t trapbits = fpgetmask (); return mask_to_exceptions (trapbits); } # elif defined _AIX && defined __powerpc__ /* AIX */ # include /* Documentation: */ /* Convert from an 'int exceptions' to an fptrap_t. */ # if 0 /* Unoptimized */ # define exceptions_to_fptrap(exceptions) \ ( ((exceptions) & FE_INVALID ? TRP_INVALID : 0) \ | ((exceptions) & FE_DIVBYZERO ? TRP_DIV_BY_ZERO : 0) \ | ((exceptions) & FE_OVERFLOW ? TRP_OVERFLOW : 0) \ | ((exceptions) & FE_UNDERFLOW ? TRP_UNDERFLOW : 0) \ | ((exceptions) & FE_INEXACT ? TRP_INEXACT : 0)) # else /* Optimized */ # define exceptions_to_fptrap(exceptions) \ (((unsigned int) (exceptions) & FE_ALL_EXCEPT) >> 22) # endif int feenableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; int prev_enabled = fegetexcept (); if (exceptions != 0) { fp_enable (exceptions_to_fptrap (exceptions)); if (prev_enabled == 0 /* && fegetexcept () != 0 */) /* Enable precise trapping mode. */ fp_trap (FP_TRAP_SYNC); } return prev_enabled; } int fedisableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; int prev_enabled = fegetexcept (); if (exceptions != 0) { fp_disable (exceptions_to_fptrap (exceptions)); if (prev_enabled != 0 && fegetexcept () == 0) /* Disable precise trapping mode. */ fp_trap (FP_TRAP_OFF); } return prev_enabled; } int fegetexcept (void) { return fegetexcept_impl (); } # elif HAVE_FESETTRAPENABLE /* HP-UX, QNX */ int feenableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; int prev_enabled = fegettrapenable (); if (exceptions != 0) fesettrapenable (prev_enabled | exceptions); return prev_enabled; } int fedisableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; int prev_enabled = fegettrapenable (); if (exceptions != 0) fesettrapenable (prev_enabled & ~exceptions); return prev_enabled; } int fegetexcept (void) { return fegettrapenable (); } # else # define NEED_FALLBACK 1 # endif #endif #if NEED_FALLBACK /* A dummy fallback. */ int feenableexcept (int exceptions) { exceptions &= FE_ALL_EXCEPT; if (exceptions != 0) return -1; return 0; } int fedisableexcept (int exceptions) { return 0; } int fegetexcept (void) { return 0; } #endif