fmod.c 851 B

12345678910111213141516171819202122232425262728293031323334353637383940
  1. /*
  2. * This file is part of the UCB release of Plan 9. It is subject to the license
  3. * terms in the LICENSE file found in the top-level directory of this
  4. * distribution and at http://akaros.cs.berkeley.edu/files/Plan9License. No
  5. * part of the UCB release of Plan 9, including this file, may be copied,
  6. * modified, propagated, or distributed except according to the terms contained
  7. * in the LICENSE file.
  8. */
  9. #include <u.h>
  10. #include <libc.h>
  11. /*
  12. * floating-point mod function without infinity or NaN checking
  13. */
  14. double
  15. fmod (double x, double y)
  16. {
  17. int sign, yexp, rexp;
  18. double r, yfr, rfr;
  19. if (y == 0)
  20. return x;
  21. if (y < 0)
  22. y = -y;
  23. yfr = frexp(y, &yexp);
  24. sign = 0;
  25. if(x < 0) {
  26. r = -x;
  27. sign++;
  28. } else
  29. r = x;
  30. while(r >= y) {
  31. rfr = frexp(r, &rexp);
  32. r -= ldexp(y, rexp - yexp - (rfr < yfr));
  33. }
  34. if(sign)
  35. r = -r;
  36. return r;
  37. }