1/* Set floating-point environment exception handling.
2 Copyright (C) 1997-2024 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
4
5 The GNU C Library is free software; you can redistribute it and/or
6 modify it under the terms of the GNU Lesser General Public
7 License as published by the Free Software Foundation; either
8 version 2.1 of the License, or (at your option) any later version.
9
10 The GNU C Library is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Lesser General Public License for more details.
14
15 You should have received a copy of the GNU Lesser General Public
16 License along with the GNU C Library; if not, see
17 <https://www.gnu.org/licenses/>. */
18
19#include <fenv.h>
20#include <ldsodefs.h>
21
22int
23__fesetexceptflag (const fexcept_t *flagp, int excepts)
24{
25 /* The flags can be set in the 387 unit or in the SSE unit. When we need to
26 clear a flag, we need to do so in both units, due to the way fetestexcept
27 is implemented.
28 When we need to set a flag, it is sufficient to do it in the SSE unit,
29 because that is guaranteed to not trap. However, on i386 CPUs that have
30 only a 387 unit, set the flags in the 387, as long as this cannot trap. */
31
32 fenv_t temp;
33
34 excepts &= FE_ALL_EXCEPT;
35
36 /* Get the current x87 FPU environment. We have to do this since we
37 cannot separately set the status word.
38 Note: fnstenv masks all floating-point exceptions until the fldenv
39 or fldcw below. */
40 __asm__ ("fnstenv %0" : "=m" (*&temp));
41
42 if (CPU_FEATURE_USABLE (SSE))
43 {
44 unsigned int mxcsr;
45
46 /* Clear relevant flags. */
47 temp.__status_word &= ~(excepts & ~ *flagp);
48
49 /* Store the new status word (along with the rest of the environment). */
50 __asm__ ("fldenv %0" : : "m" (*&temp));
51
52 /* And now similarly for SSE. */
53 __asm__ ("stmxcsr %0" : "=m" (*&mxcsr));
54
55 /* Clear or set relevant flags. */
56 mxcsr ^= (mxcsr ^ *flagp) & excepts;
57
58 /* Put the new data in effect. */
59 __asm__ ("ldmxcsr %0" : : "m" (*&mxcsr));
60 }
61 else
62 {
63 /* Clear or set relevant flags. */
64 temp.__status_word ^= (temp.__status_word ^ *flagp) & excepts;
65
66 if ((~temp.__control_word) & temp.__status_word & excepts)
67 {
68 /* Setting the exception flags may trigger a trap (at the next
69 floating-point instruction, but that does not matter).
70 ISO C 23 ยง 7.6.4.5 does not allow it. */
71 __asm__ volatile ("fldcw %0" : : "m" (*&temp.__control_word));
72 return -1;
73 }
74
75 /* Store the new status word (along with the rest of the environment). */
76 __asm__ ("fldenv %0" : : "m" (*&temp));
77 }
78
79 /* Success. */
80 return 0;
81}
82
83#include <shlib-compat.h>
84#if SHLIB_COMPAT (libm, GLIBC_2_1, GLIBC_2_2)
85strong_alias (__fesetexceptflag, __old_fesetexceptflag)
86compat_symbol (libm, __old_fesetexceptflag, fesetexceptflag, GLIBC_2_1);
87#endif
88
89versioned_symbol (libm, __fesetexceptflag, fesetexceptflag, GLIBC_2_2);
90

source code of glibc/sysdeps/i386/fpu/fsetexcptflg.c