x86: Add SSE2 optimized chacha20
[glibc.git] / sysdeps / csky / fpu / fraiseexcpt.c
blobef22217bc218688b7cd59a2e01fd51813c3f9526
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/>. */
19 #include <fenv.h>
20 #include <fenv_libc.h>
21 #include <fpu_control.h>
22 #include <float.h>
23 #include <math.h>
25 int
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
32 exception. */
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));
50 /* Next: overflow. */
51 if (FE_OVERFLOW & excepts)
53 float x = FLT_MAX;
54 __asm__ __volatile__ ("fmuls %0, %0, %0" : "+v" (x));
56 /* Next: underflow. */
57 if (FE_UNDERFLOW & excepts)
59 float x = -FLT_MIN;
61 __asm__ __volatile__ ("fmuls %0, %0, %0" : "+v" (x));
64 /* Last: inexact. */
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));
76 # else
77 int tmp = 0;
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));
95 /* Next: overflow. */
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));
112 /* Last: inexact. */
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__ */
121 /* Success. */
122 return 0;
124 libm_hidden_def (__feraiseexcept)
125 weak_alias (__feraiseexcept, feraiseexcept)
126 libm_hidden_weak (feraiseexcept)