2 #define FENV_PRIVATE_H 1
4 #include <bits/floatn.h>
6 #include <fpu_control.h>
9 # define math_opt_barrier(x) \
11 if (sizeof (x) <= sizeof (double) \
12 || __builtin_types_compatible_p (__typeof (x), _Float128)) \
13 __asm ("" : "=x" (__x) : "0" (x)); \
15 __asm ("" : "=t" (__x) : "0" (x)); \
17 # define math_force_eval(x) \
19 if (sizeof (x) <= sizeof (double) \
20 || __builtin_types_compatible_p (__typeof (x), _Float128)) \
21 __asm __volatile ("" : : "x" (x)); \
23 __asm __volatile ("" : : "f" (x)); \
26 # define math_opt_barrier(x) \
27 ({ __typeof (x) __x; \
28 if (__builtin_types_compatible_p (__typeof (x), _Float128)) \
31 __asm ("" : "+m" (__x)); \
34 __asm ("" : "=t" (__x) : "0" (x)); \
36 # define math_force_eval(x) \
38 __typeof (x) __x = (x); \
39 if (sizeof (x) <= sizeof (double) \
40 || __builtin_types_compatible_p (__typeof (x), _Float128)) \
41 __asm __volatile ("" : : "m" (__x)); \
43 __asm __volatile ("" : : "f" (__x)); \
47 /* This file is used by both the 32- and 64-bit ports. The 64-bit port
48 has a field in the fenv_t for the mxcsr; the 32-bit port does not.
49 Instead, we (ab)use the only 32-bit field extant in the struct. */
51 # define __mxcsr __eip
55 /* All of these functions are private to libm, and are all used in pairs
56 to save+change the fp state and restore the original state. Thus we
57 need not care for both the 387 and the sse unit, only the one we're
60 #if defined __AVX__ || defined SSE2AVX
61 # define STMXCSR "vstmxcsr"
62 # define LDMXCSR "vldmxcsr"
64 # define STMXCSR "stmxcsr"
65 # define LDMXCSR "ldmxcsr"
68 static __always_inline
void
69 libc_feholdexcept_sse (fenv_t
*e
)
72 asm (STMXCSR
" %0" : "=m" (*&mxcsr
));
74 mxcsr
= (mxcsr
| 0x1f80) & ~0x3f;
75 asm volatile (LDMXCSR
" %0" : : "m" (*&mxcsr
));
78 static __always_inline
void
79 libc_feholdexcept_387 (fenv_t
*e
)
81 /* Recall that fnstenv has a side-effect of masking exceptions.
82 Clobber all of the fp registers so that the TOS field is 0. */
83 asm volatile ("fnstenv %0; fnclex"
85 : : "st", "st(1)", "st(2)", "st(3)",
86 "st(4)", "st(5)", "st(6)", "st(7)");
89 static __always_inline
void
90 libc_fesetround_sse (int r
)
93 asm (STMXCSR
" %0" : "=m" (*&mxcsr
));
94 mxcsr
= (mxcsr
& ~0x6000) | (r
<< 3);
95 asm volatile (LDMXCSR
" %0" : : "m" (*&mxcsr
));
98 static __always_inline
void
99 libc_fesetround_387 (int r
)
103 cw
= (cw
& ~0xc00) | r
;
107 static __always_inline
void
108 libc_feholdexcept_setround_sse (fenv_t
*e
, int r
)
111 asm (STMXCSR
" %0" : "=m" (*&mxcsr
));
113 mxcsr
= ((mxcsr
| 0x1f80) & ~0x603f) | (r
<< 3);
114 asm volatile (LDMXCSR
" %0" : : "m" (*&mxcsr
));
117 /* Set both rounding mode and precision. A convenience function for use
118 by libc_feholdexcept_setround and libc_feholdexcept_setround_53bit. */
119 static __always_inline
void
120 libc_feholdexcept_setround_387_prec (fenv_t
*e
, int r
)
122 libc_feholdexcept_387 (e
);
124 fpu_control_t cw
= e
->__control_word
;
125 cw
&= ~(_FPU_RC_ZERO
| _FPU_EXTENDED
);
130 static __always_inline
void
131 libc_feholdexcept_setround_387 (fenv_t
*e
, int r
)
133 libc_feholdexcept_setround_387_prec (e
, r
| _FPU_EXTENDED
);
136 static __always_inline
void
137 libc_feholdexcept_setround_387_53bit (fenv_t
*e
, int r
)
139 libc_feholdexcept_setround_387_prec (e
, r
| _FPU_DOUBLE
);
142 static __always_inline
int
143 libc_fetestexcept_sse (int e
)
146 asm volatile (STMXCSR
" %0" : "=m" (*&mxcsr
));
147 return mxcsr
& e
& FE_ALL_EXCEPT
;
150 static __always_inline
int
151 libc_fetestexcept_387 (int ex
)
154 asm volatile ("fnstsw %0" : "=a" (temp
));
155 return temp
& ex
& FE_ALL_EXCEPT
;
158 static __always_inline
void
159 libc_fesetenv_sse (fenv_t
*e
)
161 asm volatile (LDMXCSR
" %0" : : "m" (e
->__mxcsr
));
164 static __always_inline
void
165 libc_fesetenv_387 (fenv_t
*e
)
167 /* Clobber all fp registers so that the TOS value we saved earlier is
168 compatible with the current state of the compiler. */
169 asm volatile ("fldenv %0"
171 : "st", "st(1)", "st(2)", "st(3)",
172 "st(4)", "st(5)", "st(6)", "st(7)");
175 static __always_inline
int
176 libc_feupdateenv_test_sse (fenv_t
*e
, int ex
)
178 unsigned int mxcsr
, old_mxcsr
, cur_ex
;
179 asm volatile (STMXCSR
" %0" : "=m" (*&mxcsr
));
180 cur_ex
= mxcsr
& FE_ALL_EXCEPT
;
182 /* Merge current exceptions with the old environment. */
183 old_mxcsr
= e
->__mxcsr
;
184 mxcsr
= old_mxcsr
| cur_ex
;
185 asm volatile (LDMXCSR
" %0" : : "m" (*&mxcsr
));
187 /* Raise SIGFPE for any new exceptions since the hold. Expect that
188 the normal environment has all exceptions masked. */
189 if (__glibc_unlikely (~(old_mxcsr
>> 7) & cur_ex
))
190 __feraiseexcept (cur_ex
);
192 /* Test for exceptions raised since the hold. */
196 static __always_inline
int
197 libc_feupdateenv_test_387 (fenv_t
*e
, int ex
)
201 /* Save current exceptions. */
202 asm volatile ("fnstsw %0" : "=a" (cur_ex
));
203 cur_ex
&= FE_ALL_EXCEPT
;
205 /* Reload original environment. */
206 libc_fesetenv_387 (e
);
208 /* Merge current exceptions. */
209 __feraiseexcept (cur_ex
);
211 /* Test for exceptions raised since the hold. */
215 static __always_inline
void
216 libc_feupdateenv_sse (fenv_t
*e
)
218 libc_feupdateenv_test_sse (e
, 0);
221 static __always_inline
void
222 libc_feupdateenv_387 (fenv_t
*e
)
224 libc_feupdateenv_test_387 (e
, 0);
227 static __always_inline
void
228 libc_feholdsetround_sse (fenv_t
*e
, int r
)
231 asm (STMXCSR
" %0" : "=m" (*&mxcsr
));
233 mxcsr
= (mxcsr
& ~0x6000) | (r
<< 3);
234 asm volatile (LDMXCSR
" %0" : : "m" (*&mxcsr
));
237 static __always_inline
void
238 libc_feholdsetround_387_prec (fenv_t
*e
, int r
)
243 e
->__control_word
= cw
;
244 cw
&= ~(_FPU_RC_ZERO
| _FPU_EXTENDED
);
249 static __always_inline
void
250 libc_feholdsetround_387 (fenv_t
*e
, int r
)
252 libc_feholdsetround_387_prec (e
, r
| _FPU_EXTENDED
);
255 static __always_inline
void
256 libc_feholdsetround_387_53bit (fenv_t
*e
, int r
)
258 libc_feholdsetround_387_prec (e
, r
| _FPU_DOUBLE
);
261 static __always_inline
void
262 libc_feresetround_sse (fenv_t
*e
)
265 asm (STMXCSR
" %0" : "=m" (*&mxcsr
));
266 mxcsr
= (mxcsr
& ~0x6000) | (e
->__mxcsr
& 0x6000);
267 asm volatile (LDMXCSR
" %0" : : "m" (*&mxcsr
));
270 static __always_inline
void
271 libc_feresetround_387 (fenv_t
*e
)
273 _FPU_SETCW (e
->__control_word
);
277 # define libc_feholdexceptf libc_feholdexcept_sse
278 # define libc_fesetroundf libc_fesetround_sse
279 # define libc_feholdexcept_setroundf libc_feholdexcept_setround_sse
280 # define libc_fetestexceptf libc_fetestexcept_sse
281 # define libc_fesetenvf libc_fesetenv_sse
282 # define libc_feupdateenv_testf libc_feupdateenv_test_sse
283 # define libc_feupdateenvf libc_feupdateenv_sse
284 # define libc_feholdsetroundf libc_feholdsetround_sse
285 # define libc_feresetroundf libc_feresetround_sse
287 # define libc_feholdexceptf libc_feholdexcept_387
288 # define libc_fesetroundf libc_fesetround_387
289 # define libc_feholdexcept_setroundf libc_feholdexcept_setround_387
290 # define libc_fetestexceptf libc_fetestexcept_387
291 # define libc_fesetenvf libc_fesetenv_387
292 # define libc_feupdateenv_testf libc_feupdateenv_test_387
293 # define libc_feupdateenvf libc_feupdateenv_387
294 # define libc_feholdsetroundf libc_feholdsetround_387
295 # define libc_feresetroundf libc_feresetround_387
296 #endif /* __SSE_MATH__ */
299 # define libc_feholdexcept libc_feholdexcept_sse
300 # define libc_fesetround libc_fesetround_sse
301 # define libc_feholdexcept_setround libc_feholdexcept_setround_sse
302 # define libc_fetestexcept libc_fetestexcept_sse
303 # define libc_fesetenv libc_fesetenv_sse
304 # define libc_feupdateenv_test libc_feupdateenv_test_sse
305 # define libc_feupdateenv libc_feupdateenv_sse
306 # define libc_feholdsetround libc_feholdsetround_sse
307 # define libc_feresetround libc_feresetround_sse
309 # define libc_feholdexcept libc_feholdexcept_387
310 # define libc_fesetround libc_fesetround_387
311 # define libc_feholdexcept_setround libc_feholdexcept_setround_387
312 # define libc_fetestexcept libc_fetestexcept_387
313 # define libc_fesetenv libc_fesetenv_387
314 # define libc_feupdateenv_test libc_feupdateenv_test_387
315 # define libc_feupdateenv libc_feupdateenv_387
316 # define libc_feholdsetround libc_feholdsetround_387
317 # define libc_feresetround libc_feresetround_387
318 #endif /* __SSE2_MATH__ */
320 #define libc_feholdexceptl libc_feholdexcept_387
321 #define libc_fesetroundl libc_fesetround_387
322 #define libc_feholdexcept_setroundl libc_feholdexcept_setround_387
323 #define libc_fetestexceptl libc_fetestexcept_387
324 #define libc_fesetenvl libc_fesetenv_387
325 #define libc_feupdateenv_testl libc_feupdateenv_test_387
326 #define libc_feupdateenvl libc_feupdateenv_387
327 #define libc_feholdsetroundl libc_feholdsetround_387
328 #define libc_feresetroundl libc_feresetround_387
330 #ifndef __SSE2_MATH__
331 # define libc_feholdexcept_setround_53bit libc_feholdexcept_setround_387_53bit
332 # define libc_feholdsetround_53bit libc_feholdsetround_387_53bit
336 /* The SSE rounding mode is used by soft-fp (libgcc and glibc) on
337 x86_64, so that must be set for float128 computations. */
338 # define SET_RESTORE_ROUNDF128(RM) \
339 SET_RESTORE_ROUND_GENERIC (RM, libc_feholdsetround_sse, libc_feresetround_sse)
340 # define libc_feholdexcept_setroundf128 libc_feholdexcept_setround_sse
341 # define libc_feupdateenv_testf128 libc_feupdateenv_test_sse
344 /* We have support for rounding mode context. */
345 #define HAVE_RM_CTX 1
347 static __always_inline
void
348 libc_feholdexcept_setround_sse_ctx (struct rm_ctx
*ctx
, int r
)
350 unsigned int mxcsr
, new_mxcsr
;
351 asm (STMXCSR
" %0" : "=m" (*&mxcsr
));
352 new_mxcsr
= ((mxcsr
| 0x1f80) & ~0x603f) | (r
<< 3);
354 ctx
->env
.__mxcsr
= mxcsr
;
355 if (__glibc_unlikely (mxcsr
!= new_mxcsr
))
357 asm volatile (LDMXCSR
" %0" : : "m" (*&new_mxcsr
));
358 ctx
->updated_status
= true;
361 ctx
->updated_status
= false;
364 /* Unconditional since we want to overwrite any exceptions that occurred in the
365 context. This is also why all fehold* functions unconditionally write into
367 static __always_inline
void
368 libc_fesetenv_sse_ctx (struct rm_ctx
*ctx
)
370 libc_fesetenv_sse (&ctx
->env
);
373 static __always_inline
void
374 libc_feupdateenv_sse_ctx (struct rm_ctx
*ctx
)
376 if (__glibc_unlikely (ctx
->updated_status
))
377 libc_feupdateenv_test_sse (&ctx
->env
, 0);
380 static __always_inline
void
381 libc_feholdexcept_setround_387_prec_ctx (struct rm_ctx
*ctx
, int r
)
383 libc_feholdexcept_387 (&ctx
->env
);
385 fpu_control_t cw
= ctx
->env
.__control_word
;
386 fpu_control_t old_cw
= cw
;
387 cw
&= ~(_FPU_RC_ZERO
| _FPU_EXTENDED
);
390 if (__glibc_unlikely (old_cw
!= cw
))
393 ctx
->updated_status
= true;
396 ctx
->updated_status
= false;
399 static __always_inline
void
400 libc_feholdexcept_setround_387_ctx (struct rm_ctx
*ctx
, int r
)
402 libc_feholdexcept_setround_387_prec_ctx (ctx
, r
| _FPU_EXTENDED
);
405 static __always_inline
void
406 libc_feholdexcept_setround_387_53bit_ctx (struct rm_ctx
*ctx
, int r
)
408 libc_feholdexcept_setround_387_prec_ctx (ctx
, r
| _FPU_DOUBLE
);
411 static __always_inline
void
412 libc_feholdsetround_387_prec_ctx (struct rm_ctx
*ctx
, int r
)
414 fpu_control_t cw
, new_cw
;
418 new_cw
&= ~(_FPU_RC_ZERO
| _FPU_EXTENDED
);
421 ctx
->env
.__control_word
= cw
;
422 if (__glibc_unlikely (new_cw
!= cw
))
425 ctx
->updated_status
= true;
428 ctx
->updated_status
= false;
431 static __always_inline
void
432 libc_feholdsetround_387_ctx (struct rm_ctx
*ctx
, int r
)
434 libc_feholdsetround_387_prec_ctx (ctx
, r
| _FPU_EXTENDED
);
437 static __always_inline
void
438 libc_feholdsetround_387_53bit_ctx (struct rm_ctx
*ctx
, int r
)
440 libc_feholdsetround_387_prec_ctx (ctx
, r
| _FPU_DOUBLE
);
443 static __always_inline
void
444 libc_feholdsetround_sse_ctx (struct rm_ctx
*ctx
, int r
)
446 unsigned int mxcsr
, new_mxcsr
;
448 asm (STMXCSR
" %0" : "=m" (*&mxcsr
));
449 new_mxcsr
= (mxcsr
& ~0x6000) | (r
<< 3);
451 ctx
->env
.__mxcsr
= mxcsr
;
452 if (__glibc_unlikely (new_mxcsr
!= mxcsr
))
454 asm volatile (LDMXCSR
" %0" : : "m" (*&new_mxcsr
));
455 ctx
->updated_status
= true;
458 ctx
->updated_status
= false;
461 static __always_inline
void
462 libc_feresetround_sse_ctx (struct rm_ctx
*ctx
)
464 if (__glibc_unlikely (ctx
->updated_status
))
465 libc_feresetround_sse (&ctx
->env
);
468 static __always_inline
void
469 libc_feresetround_387_ctx (struct rm_ctx
*ctx
)
471 if (__glibc_unlikely (ctx
->updated_status
))
472 _FPU_SETCW (ctx
->env
.__control_word
);
475 static __always_inline
void
476 libc_feupdateenv_387_ctx (struct rm_ctx
*ctx
)
478 if (__glibc_unlikely (ctx
->updated_status
))
479 libc_feupdateenv_test_387 (&ctx
->env
, 0);
483 # define libc_feholdexcept_setroundf_ctx libc_feholdexcept_setround_sse_ctx
484 # define libc_fesetenvf_ctx libc_fesetenv_sse_ctx
485 # define libc_feupdateenvf_ctx libc_feupdateenv_sse_ctx
486 # define libc_feholdsetroundf_ctx libc_feholdsetround_sse_ctx
487 # define libc_feresetroundf_ctx libc_feresetround_sse_ctx
489 # define libc_feholdexcept_setroundf_ctx libc_feholdexcept_setround_387_ctx
490 # define libc_feupdateenvf_ctx libc_feupdateenv_387_ctx
491 # define libc_feholdsetroundf_ctx libc_feholdsetround_387_ctx
492 # define libc_feresetroundf_ctx libc_feresetround_387_ctx
493 #endif /* __SSE_MATH__ */
496 # define libc_feholdexcept_setround_ctx libc_feholdexcept_setround_sse_ctx
497 # define libc_fesetenv_ctx libc_fesetenv_sse_ctx
498 # define libc_feupdateenv_ctx libc_feupdateenv_sse_ctx
499 # define libc_feholdsetround_ctx libc_feholdsetround_sse_ctx
500 # define libc_feresetround_ctx libc_feresetround_sse_ctx
502 # define libc_feholdexcept_setround_ctx libc_feholdexcept_setround_387_ctx
503 # define libc_feupdateenv_ctx libc_feupdateenv_387_ctx
504 # define libc_feholdsetround_ctx libc_feholdsetround_387_ctx
505 # define libc_feresetround_ctx libc_feresetround_387_ctx
506 #endif /* __SSE2_MATH__ */
508 #define libc_feholdexcept_setroundl_ctx libc_feholdexcept_setround_387_ctx
509 #define libc_feupdateenvl_ctx libc_feupdateenv_387_ctx
510 #define libc_feholdsetroundl_ctx libc_feholdsetround_387_ctx
511 #define libc_feresetroundl_ctx libc_feresetround_387_ctx
513 #ifndef __SSE2_MATH__
514 # define libc_feholdsetround_53bit_ctx libc_feholdsetround_387_53bit_ctx
515 # define libc_feresetround_53bit_ctx libc_feresetround_387_ctx
520 #endif /* FENV_PRIVATE_H */