1 1.7 riastrad /* $NetBSD: fenv.h,v 1.7 2024/10/30 15:56:11 riastradh Exp $ */ 2 1.1 christos 3 1.1 christos /*- 4 1.1 christos * Copyright (c) 2004-2005 David Schultz <das (at) FreeBSD.ORG> 5 1.1 christos * All rights reserved. 6 1.1 christos * 7 1.1 christos * Redistribution and use in source and binary forms, with or without 8 1.1 christos * modification, are permitted provided that the following conditions 9 1.1 christos * are met: 10 1.1 christos * 1. Redistributions of source code must retain the above copyright 11 1.1 christos * notice, this list of conditions and the following disclaimer. 12 1.1 christos * 2. Redistributions in binary form must reproduce the above copyright 13 1.1 christos * notice, this list of conditions and the following disclaimer in the 14 1.1 christos * documentation and/or other materials provided with the distribution. 15 1.1 christos * 16 1.1 christos * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 17 1.1 christos * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 18 1.1 christos * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 19 1.1 christos * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 20 1.1 christos * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 21 1.1 christos * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 22 1.1 christos * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 23 1.1 christos * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 24 1.1 christos * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 25 1.1 christos * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 26 1.1 christos * SUCH DAMAGE. 27 1.1 christos * 28 1.1 christos * $FreeBSD: head/lib/msun/mips/fenv.h 226218 2011-10-10 15:43:09Z das $ 29 1.1 christos */ 30 1.1 christos 31 1.1 christos #ifndef _MIPS_FENV_H_ 32 1.1 christos #define _MIPS_FENV_H_ 33 1.1 christos 34 1.7 riastrad #include <sys/featuretest.h> 35 1.1 christos #include <sys/stdint.h> 36 1.1 christos 37 1.1 christos /* Exception flags */ 38 1.2 christos #define FE_INEXACT 0x0004 39 1.1 christos #define FE_UNDERFLOW 0x0008 40 1.2 christos #define FE_OVERFLOW 0x0010 41 1.2 christos #define FE_DIVBYZERO 0x0020 42 1.2 christos #define FE_INVALID 0x0040 43 1.1 christos #define FE_ALL_EXCEPT (FE_DIVBYZERO | FE_INEXACT | \ 44 1.1 christos FE_INVALID | FE_OVERFLOW | FE_UNDERFLOW) 45 1.1 christos 46 1.1 christos /* Rounding modes */ 47 1.1 christos #define FE_TONEAREST 0x0000 48 1.1 christos #define FE_TOWARDZERO 0x0001 49 1.1 christos #define FE_UPWARD 0x0002 50 1.1 christos #define FE_DOWNWARD 0x0003 51 1.1 christos #define _ROUND_MASK (FE_TONEAREST | FE_DOWNWARD | \ 52 1.1 christos FE_UPWARD | FE_TOWARDZERO) 53 1.4 chs 54 1.4 chs #ifndef __mips_soft_float 55 1.4 chs 56 1.4 chs #ifndef __fenv_static 57 1.4 chs #define __fenv_static static 58 1.4 chs #endif 59 1.4 chs 60 1.4 chs typedef uint32_t fpu_control_t __attribute__((__mode__(__SI__))); 61 1.4 chs typedef fpu_control_t fenv_t; 62 1.4 chs typedef fpu_control_t fexcept_t; 63 1.4 chs 64 1.1 christos __BEGIN_DECLS 65 1.1 christos 66 1.1 christos /* Default floating-point environment */ 67 1.1 christos extern const fenv_t __fe_dfl_env; 68 1.1 christos #define FE_DFL_ENV (&__fe_dfl_env) 69 1.1 christos 70 1.1 christos /* We need to be able to map status flag positions to mask flag positions */ 71 1.2 christos #define _ENABLE_MASK (FE_ALL_EXCEPT << _ENABLE_SHIFT) 72 1.6 simonb #define _ENABLE_SHIFT 5 73 1.1 christos 74 1.3 chs static inline fpu_control_t 75 1.3 chs __rfs(void) 76 1.3 chs { 77 1.3 chs fpu_control_t __fpsr; 78 1.3 chs 79 1.3 chs __asm __volatile("cfc1 %0,$31" : "=r" (__fpsr)); 80 1.3 chs return __fpsr; 81 1.3 chs } 82 1.2 christos 83 1.3 chs static inline void 84 1.3 chs __wfs(fpu_control_t __fpsr) 85 1.3 chs { 86 1.3 chs 87 1.3 chs __asm __volatile("ctc1 %0,$31" : : "r" (__fpsr)); 88 1.3 chs } 89 1.1 christos 90 1.5 christos #if __GNUC_PREREQ__(8, 0) 91 1.5 christos #pragma GCC diagnostic push 92 1.5 christos #pragma GCC diagnostic ignored "-Wshadow" 93 1.5 christos #endif 94 1.5 christos 95 1.1 christos __fenv_static inline int 96 1.1 christos feclearexcept(int __excepts) 97 1.1 christos { 98 1.1 christos fexcept_t __fpsr; 99 1.1 christos 100 1.2 christos __excepts &= FE_ALL_EXCEPT; 101 1.3 chs __fpsr = __rfs(); 102 1.2 christos __fpsr &= ~(__excepts | (__excepts << _ENABLE_SHIFT)); 103 1.1 christos __wfs(__fpsr); 104 1.2 christos return 0; 105 1.1 christos } 106 1.1 christos 107 1.1 christos __fenv_static inline int 108 1.1 christos fegetexceptflag(fexcept_t *__flagp, int __excepts) 109 1.1 christos { 110 1.1 christos fexcept_t __fpsr; 111 1.1 christos 112 1.3 chs __fpsr = __rfs(); 113 1.1 christos *__flagp = __fpsr & __excepts; 114 1.1 christos return (0); 115 1.1 christos } 116 1.1 christos 117 1.1 christos __fenv_static inline int 118 1.1 christos fesetexceptflag(const fexcept_t *__flagp, int __excepts) 119 1.1 christos { 120 1.1 christos fexcept_t __fpsr; 121 1.1 christos 122 1.3 chs __fpsr = __rfs(); 123 1.1 christos __fpsr &= ~__excepts; 124 1.1 christos __fpsr |= *__flagp & __excepts; 125 1.1 christos __wfs(__fpsr); 126 1.1 christos return (0); 127 1.1 christos } 128 1.1 christos 129 1.1 christos __fenv_static inline int 130 1.1 christos feraiseexcept(int __excepts) 131 1.1 christos { 132 1.1 christos fexcept_t __ex = __excepts; 133 1.1 christos 134 1.1 christos fesetexceptflag(&__ex, __excepts); /* XXX */ 135 1.1 christos return (0); 136 1.1 christos } 137 1.1 christos 138 1.1 christos __fenv_static inline int 139 1.1 christos fetestexcept(int __excepts) 140 1.1 christos { 141 1.1 christos fexcept_t __fpsr; 142 1.1 christos 143 1.3 chs __fpsr = __rfs(); 144 1.1 christos return (__fpsr & __excepts); 145 1.1 christos } 146 1.1 christos 147 1.1 christos __fenv_static inline int 148 1.1 christos fegetround(void) 149 1.1 christos { 150 1.2 christos fexcept_t __fpsr; 151 1.1 christos 152 1.3 chs __fpsr = __rfs(); 153 1.2 christos return __fpsr & _ROUND_MASK; 154 1.1 christos } 155 1.1 christos 156 1.1 christos __fenv_static inline int 157 1.1 christos fesetround(int __round) 158 1.1 christos { 159 1.2 christos fexcept_t __fpsr; 160 1.2 christos 161 1.2 christos if (__round & ~_ROUND_MASK) 162 1.2 christos return 1; 163 1.3 chs __fpsr = __rfs(); 164 1.2 christos __fpsr &= ~_ROUND_MASK; 165 1.2 christos __fpsr |= __round; 166 1.3 chs __wfs(__fpsr); 167 1.1 christos 168 1.2 christos return 0; 169 1.1 christos } 170 1.1 christos 171 1.1 christos __fenv_static inline int 172 1.1 christos fegetenv(fenv_t *__envp) 173 1.1 christos { 174 1.1 christos 175 1.3 chs *__envp = __rfs(); 176 1.1 christos return (0); 177 1.1 christos } 178 1.1 christos 179 1.1 christos __fenv_static inline int 180 1.1 christos feholdexcept(fenv_t *__envp) 181 1.1 christos { 182 1.1 christos fenv_t __env; 183 1.1 christos 184 1.3 chs __env = __rfs(); 185 1.1 christos *__envp = __env; 186 1.1 christos __env &= ~(FE_ALL_EXCEPT | _ENABLE_MASK); 187 1.1 christos __wfs(__env); 188 1.1 christos return (0); 189 1.1 christos } 190 1.1 christos 191 1.1 christos __fenv_static inline int 192 1.1 christos fesetenv(const fenv_t *__envp) 193 1.1 christos { 194 1.1 christos 195 1.1 christos __wfs(*__envp); 196 1.1 christos return (0); 197 1.1 christos } 198 1.1 christos 199 1.1 christos __fenv_static inline int 200 1.1 christos feupdateenv(const fenv_t *__envp) 201 1.1 christos { 202 1.1 christos fexcept_t __fpsr; 203 1.1 christos 204 1.3 chs __fpsr = __rfs(); 205 1.1 christos __wfs(*__envp); 206 1.1 christos feraiseexcept(__fpsr & FE_ALL_EXCEPT); 207 1.1 christos return (0); 208 1.1 christos } 209 1.1 christos 210 1.5 christos #if __GNUC_PREREQ__(8, 0) 211 1.5 christos #pragma GCC diagnostic pop 212 1.5 christos #endif 213 1.5 christos 214 1.1 christos #if defined(_NETBSD_SOURCE) || defined(_GNU_SOURCE) 215 1.1 christos 216 1.4 chs __fenv_static inline int 217 1.2 christos feenableexcept(int __excepts) 218 1.1 christos { 219 1.1 christos fenv_t __old_fpsr, __new_fpsr; 220 1.1 christos 221 1.3 chs __new_fpsr = __rfs(); 222 1.2 christos __old_fpsr = (__new_fpsr & _ENABLE_MASK) >> _ENABLE_SHIFT; 223 1.2 christos __excepts &= FE_ALL_EXCEPT; 224 1.2 christos __new_fpsr |= __excepts << _ENABLE_SHIFT; 225 1.1 christos __wfs(__new_fpsr); 226 1.2 christos return __old_fpsr; 227 1.1 christos } 228 1.1 christos 229 1.4 chs __fenv_static inline int 230 1.2 christos fedisableexcept(int __excepts) 231 1.1 christos { 232 1.1 christos fenv_t __old_fpsr, __new_fpsr; 233 1.1 christos 234 1.3 chs __new_fpsr = __rfs(); 235 1.2 christos __old_fpsr = (__new_fpsr & _ENABLE_MASK) >> _ENABLE_SHIFT; 236 1.2 christos __excepts &= FE_ALL_EXCEPT; 237 1.2 christos __new_fpsr &= ~(__excepts << _ENABLE_SHIFT); 238 1.1 christos __wfs(__new_fpsr); 239 1.2 christos return __old_fpsr; 240 1.1 christos } 241 1.1 christos 242 1.4 chs __fenv_static inline int 243 1.1 christos fegetexcept(void) 244 1.1 christos { 245 1.1 christos fenv_t __fpsr; 246 1.1 christos 247 1.3 chs __fpsr = __rfs(); 248 1.2 christos return ((__fpsr & _ENABLE_MASK) >> _ENABLE_SHIFT); 249 1.1 christos } 250 1.1 christos 251 1.1 christos #endif /* _NETBSD_SOURCE || _GNU_SOURCE */ 252 1.1 christos 253 1.1 christos __END_DECLS 254 1.1 christos 255 1.4 chs #endif /* __mips_soft_float */ 256 1.4 chs 257 1.1 christos #endif /* !_FENV_H_ */ 258