fraiseexcpt.c 1.8 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970
  1. /* Raise given exceptions.
  2. Copyright (C) 1997-2025 Free Software Foundation, Inc.
  3. The GNU C Library is free software; you can redistribute it and/or
  4. modify it under the terms of the GNU Lesser General Public
  5. License as published by the Free Software Foundation; either
  6. version 2.1 of the License, or (at your option) any later version.
  7. The GNU C Library is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  10. Lesser General Public License for more details.
  11. You should have received a copy of the GNU Lesser General Public
  12. License along with the GNU C Library; if not, see
  13. <https://www.gnu.org/licenses/>. */
  14. #include <fenv.h>
  15. #include <float.h>
  16. #include <fpu_control.h>
  17. #include <math.h>
  18. int
  19. feraiseexcept (int excepts)
  20. {
  21. if (excepts == 0)
  22. return 0;
  23. /* Raise exceptions represented by EXPECTS. */
  24. if (excepts & FE_INEXACT)
  25. {
  26. double d = 1.0, x = 3.0;
  27. __asm__ __volatile__ ("fdiv %1, %0" : "+d" (d) : "d" (x));
  28. }
  29. if (excepts & FE_UNDERFLOW)
  30. {
  31. long double d = LDBL_MIN, x = 10;
  32. __asm__ __volatile__ ("fdiv %1, %0" : "+d" (d) : "d" (x));
  33. }
  34. if (excepts & FE_OVERFLOW)
  35. {
  36. long double d = LDBL_MAX;
  37. __asm__ __volatile__ ("fmul %0, %0" : "+d" (d) : "d" (d));
  38. }
  39. if (excepts & FE_DIVBYZERO)
  40. {
  41. double d = 1.0, x = 0.0;
  42. __asm__ __volatile__ ("fdiv %1, %0" : "+d" (d) : "d" (x));
  43. }
  44. if (excepts & FE_INVALID)
  45. {
  46. double d = HUGE_VAL, x = 0.0;
  47. __asm__ __volatile__ ("fmul %1, %0" : "+d" (d) : "d" (x));
  48. }
  49. {
  50. /* Restore flag fields. */
  51. fpu_control_t cw;
  52. _FPU_GETCW (cw);
  53. cw |= (excepts & FE_ALL_EXCEPT);
  54. _FPU_SETCW (cw);
  55. }
  56. return 0;
  57. }