1 /* Raise given exceptions.
2 Copyright (C) 2018-2022 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
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.
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.
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/>. */
20 #include <fenv_libc.h>
21 #include <fpu_control.h>
26 __feraiseexcept (int excepts
)
28 /* Raise exceptions represented by EXCEPTS. But we must raise only one
29 signal at a time. It is important that if the overflow/underflow
30 exception and the divide by zero exception are given at the same
31 time, the overflow/underflow exception follows the divide by zero
34 # ifndef __csky_fpuv1__
35 /* First: invalid exception. */
36 if (FE_INVALID
& excepts
)
38 /* One example of a invalid operation is 0 * Infinity. */
39 float x
= HUGE_VALF
, y
= 0.0f
;
40 __asm__
__volatile__ ("fmuls %0, %0, %1" : "+v" (x
) : "v" (y
));
43 /* Next: division by zero. */
44 if (FE_DIVBYZERO
& excepts
)
46 float x
= 1.0f
, y
= 0.0f
;
47 __asm__
__volatile__ ("fdivs %0, %0, %1" : "+v" (x
) : "v" (y
));
51 if (FE_OVERFLOW
& excepts
)
54 __asm__
__volatile__ ("fmuls %0, %0, %0" : "+v" (x
));
56 /* Next: underflow. */
57 if (FE_UNDERFLOW
& excepts
)
61 __asm__
__volatile__ ("fmuls %0, %0, %0" : "+v" (x
));
65 if (FE_INEXACT
& excepts
)
67 float x
= 1.0f
, y
= 3.0f
;
68 __asm__
__volatile__ ("fdivs %0, %0, %1" : "+v" (x
) : "v" (y
));
71 if (__FE_DENORMAL
& excepts
)
73 double x
= 4.9406564584124654e-324;
74 __asm__
__volatile__ ("fstod %0, %0" : "+v" (x
));
78 /* First: invalid exception. */
79 if (FE_INVALID
& excepts
)
81 /* One example of a invalid operation is 0 * Infinity. */
82 float x
= HUGE_VALF
, y
= 0.0f
;
83 __asm__
__volatile__ ("fmuls %0, %0, %2, %1"
84 : "+f" (x
), "+r"(tmp
) : "f" (y
));
87 /* Next: division by zero. */
88 if (FE_DIVBYZERO
& excepts
)
90 float x
= 1.0f
, y
= 0.0f
;
91 __asm__
__volatile__ ("fdivs %0, %0, %2, %1"
92 : "+f" (x
), "+r"(tmp
) : "f" (y
));
96 if (FE_OVERFLOW
& excepts
)
98 float x
= FLT_MAX
, y
= FLT_MAX
;
99 __asm__
__volatile__ ("fmuls %0, %0, %2, %1"
100 : "+f" (x
), "+r"(tmp
) : "f" (y
));
103 /* Next: underflow. */
104 if (FE_UNDERFLOW
& excepts
)
106 float x
= -FLT_MIN
, y
= -FLT_MIN
;
108 __asm__
__volatile__ ("fmuls %0, %0, %2, %1"
109 : "+f" (x
), "+r"(tmp
) : "f" (y
));
113 if (FE_INEXACT
& excepts
)
115 float x
= 1.0f
, y
= 3.0f
;
116 __asm__
__volatile__ ("fdivs %0, %0, %2, %1"
117 : "+f" (x
), "+r"(tmp
) : "f" (y
));
119 # endif /* __csky_fpuv2__ */
124 libm_hidden_def (__feraiseexcept
)
125 weak_alias (__feraiseexcept
, feraiseexcept
)
126 libm_hidden_weak (feraiseexcept
)