feupdateenv.c 1.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748
  1. /* Install given floating-point environment and raise exceptions.
  2. Copyright (C) 2004 Free Software Foundation, Inc.
  3. This file is part of the GNU C Library.
  4. Contributed by Aldy Hernandez <aldyh@redhat.com>, 2004.
  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. The GNU C Library is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  12. Lesser General Public License for more details.
  13. You should have received a copy of the GNU Lesser General Public
  14. License along with the GNU C Library; if not, see
  15. <http://www.gnu.org/licenses/>. */
  16. #include "fenv_libc.h"
  17. #include <syscall.h>
  18. #include <sys/prctl.h>
  19. int
  20. __feupdateenv (const fenv_t *envp)
  21. {
  22. unsigned long fpescr, old, new, pflags;
  23. fenv_union_t u;
  24. INTERNAL_SYSCALL_DECL (err);
  25. /* Save the currently set exceptions. */
  26. u.fenv = *envp;
  27. new = u.l[1];
  28. old = fegetenv_register ();
  29. new |= (old & FE_ALL_EXCEPT);
  30. INTERNAL_SYSCALL (prctl, err, 2, PR_GET_FPEXC, &pflags);
  31. pflags |= u.l[0];
  32. INTERNAL_SYSCALL (prctl, err, 2, PR_SET_FPEXC, pflags);
  33. /* Enable and raise (if appropriate) exceptions set in `new'. */
  34. fesetenv_register (new);
  35. feraiseexcept (new & FE_ALL_EXCEPT);
  36. /* Success. */
  37. return 0;
  38. }