rintl.c 2.7 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485
  1. /* origin: FreeBSD /usr/src/lib/msun/src/s_rintl.c */
  2. /*-
  3. * Copyright (c) 2008 David Schultz <[email protected]>
  4. * All rights reserved.
  5. *
  6. * Redistribution and use in source and binary forms, with or without
  7. * modification, are permitted provided that the following conditions
  8. * are met:
  9. * 1. Redistributions of source code must retain the above copyright
  10. * notice, this list of conditions and the following disclaimer.
  11. * 2. Redistributions in binary form must reproduce the above copyright
  12. * notice, this list of conditions and the following disclaimer in the
  13. * documentation and/or other materials provided with the distribution.
  14. *
  15. * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
  16. * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  17. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  18. * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
  19. * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  20. * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  21. * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  22. * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  23. * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  24. * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  25. * SUCH DAMAGE.
  26. */
  27. #include "libm.h"
  28. #if LDBL_MANT_DIG == 53 && LDBL_MAX_EXP == 1024
  29. long double rintl(long double x)
  30. {
  31. return rint(x);
  32. }
  33. #elif (LDBL_MANT_DIG == 64 || LDBL_MANT_DIG == 113) && LDBL_MAX_EXP == 16384
  34. #define BIAS (LDBL_MAX_EXP - 1)
  35. static const float
  36. shift[2] = {
  37. #if LDBL_MANT_DIG == 64
  38. 0x1.0p63, -0x1.0p63
  39. #elif LDBL_MANT_DIG == 113
  40. 0x1.0p112, -0x1.0p112
  41. #endif
  42. };
  43. static const float zero[2] = { 0.0, -0.0 };
  44. long double rintl(long double x)
  45. {
  46. union IEEEl2bits u;
  47. uint32_t expsign;
  48. int ex, sign;
  49. u.e = x;
  50. expsign = u.xbits.expsign;
  51. ex = expsign & 0x7fff;
  52. if (ex >= BIAS + LDBL_MANT_DIG - 1) {
  53. if (ex == BIAS + LDBL_MAX_EXP)
  54. return x + x; /* Inf, NaN, or unsupported format */
  55. return x; /* finite and already an integer */
  56. }
  57. sign = expsign >> 15;
  58. /*
  59. * The following code assumes that intermediate results are
  60. * evaluated in long double precision. If they are evaluated in
  61. * greater precision, double rounding may occur, and if they are
  62. * evaluated in less precision (as on i386), results will be
  63. * wildly incorrect.
  64. */
  65. x += shift[sign];
  66. x -= shift[sign];
  67. /*
  68. * If the result is +-0, then it must have the same sign as x, but
  69. * the above calculation doesn't always give this. Fix up the sign.
  70. */
  71. if (ex < BIAS && x == 0.0)
  72. return zero[sign];
  73. return x;
  74. }
  75. #endif