/mingw-w64-v2.0.999/mingw/mingw-w64-crt/math/tgammaf.c

# · C · 194 lines · 150 code · 18 blank · 26 comment · 27 complexity · bea1eee88da425f3523f641327fd9867 MD5 · raw file

  1. /**
  2. * This file has no copyright assigned and is placed in the Public Domain.
  3. * This file is part of the w64 mingw-runtime package.
  4. * No warranty is given; refer to the file DISCLAIMER.PD within this package.
  5. */
  6. #include "cephes_mconf.h"
  7. /* define MAXGAM 34.84425627277176174 */
  8. /* Stirling's formula for the gamma function
  9. * gamma(x) = sqrt(2 pi) x^(x-.5) exp(-x) ( 1 + 1/x P(1/x) )
  10. * .028 < 1/x < .1
  11. * relative error < 1.9e-11
  12. */
  13. static const float STIR[] = {
  14. -2.705194986674176E-003,
  15. 3.473255786154910E-003,
  16. 8.333331788340907E-002,
  17. };
  18. static const float MAXSTIR = 26.77;
  19. static const float SQTPIF = 2.50662827463100050242; /* sqrt( 2 pi ) */
  20. static float stirf(float);
  21. /* Gamma function computed by Stirling's formula,
  22. * sqrt(2 pi) x^(x-.5) exp(-x) (1 + 1/x P(1/x))
  23. * The polynomial STIR is valid for 33 <= x <= 172.
  24. */
  25. static float stirf( float x )
  26. {
  27. float y, w, v;
  28. w = 1.0/x;
  29. w = 1.0 + w * polevlf(w, STIR, 2);
  30. y = expf(-x);
  31. if (x > MAXSTIR)
  32. { /* Avoid overflow in pow() */
  33. v = powf(x, 0.5 * x - 0.25);
  34. y *= v;
  35. y *= v;
  36. }
  37. else
  38. {
  39. y = powf(x, x - 0.5) * y;
  40. }
  41. y = SQTPIF * y * w;
  42. return (y);
  43. }
  44. /* gamma(x+2), 0 < x < 1 */
  45. static const float P[] = {
  46. 1.536830450601906E-003,
  47. 5.397581592950993E-003,
  48. 4.130370201859976E-003,
  49. 7.232307985516519E-002,
  50. 8.203960091619193E-002,
  51. 4.117857447645796E-001,
  52. 4.227867745131584E-001,
  53. 9.999999822945073E-001,
  54. };
  55. float __tgammaf_r( float x, int* sgngamf);
  56. float __tgammaf_r( float x, int* sgngamf)
  57. {
  58. float p, q, z, nz;
  59. int i, direction, negative;
  60. #ifdef NANS
  61. if (isnan(x))
  62. return (x);
  63. #endif
  64. #ifdef INFINITIES
  65. #ifdef NANS
  66. if (x == INFINITYF)
  67. return (x);
  68. if (x == -INFINITYF)
  69. return (NANF);
  70. #else
  71. if (!isfinite(x))
  72. return (x);
  73. #endif
  74. #endif
  75. *sgngamf = 1;
  76. negative = 0;
  77. nz = 0.0;
  78. if (x < 0.0)
  79. {
  80. negative = 1;
  81. q = -x;
  82. p = floorf(q);
  83. if (p == q)
  84. {
  85. gsing:
  86. _SET_ERRNO(EDOM);
  87. mtherr("tgammaf", SING);
  88. #ifdef INFINITIES
  89. return (INFINITYF);
  90. #else
  91. return (MAXNUMF);
  92. #endif
  93. }
  94. i = p;
  95. if ((i & 1) == 0)
  96. *sgngamf = -1;
  97. nz = q - p;
  98. if (nz > 0.5)
  99. {
  100. p += 1.0;
  101. nz = q - p;
  102. }
  103. nz = q * sinf(PIF * nz);
  104. if (nz == 0.0)
  105. {
  106. _SET_ERRNO(ERANGE);
  107. mtherr("tgamma", OVERFLOW);
  108. #ifdef INFINITIES
  109. return(*sgngamf * INFINITYF);
  110. #else
  111. return(*sgngamf * MAXNUMF);
  112. #endif
  113. }
  114. if (nz < 0)
  115. nz = -nz;
  116. x = q;
  117. }
  118. if (x >= 10.0)
  119. {
  120. z = stirf(x);
  121. }
  122. if (x < 2.0)
  123. direction = 1;
  124. else
  125. direction = 0;
  126. z = 1.0;
  127. while (x >= 3.0)
  128. {
  129. x -= 1.0;
  130. z *= x;
  131. }
  132. /*
  133. while (x < 0.0)
  134. {
  135. if (x > -1.E-4)
  136. goto Small;
  137. z *=x;
  138. x += 1.0;
  139. }
  140. */
  141. while (x < 2.0)
  142. {
  143. if (x < 1.e-4)
  144. goto Small;
  145. z *=x;
  146. x += 1.0;
  147. }
  148. if (direction)
  149. z = 1.0/z;
  150. if (x == 2.0)
  151. return (z);
  152. x -= 2.0;
  153. p = z * polevlf(x, P, 7);
  154. gdone:
  155. if (negative)
  156. {
  157. p = *sgngamf * PIF/(nz * p );
  158. }
  159. return (p);
  160. Small:
  161. if (x == 0.0)
  162. {
  163. goto gsing;
  164. }
  165. else
  166. {
  167. p = z / ((1.0 + 0.5772156649015329 * x) * x);
  168. goto gdone;
  169. }
  170. }
  171. /* This is the C99 version */
  172. float tgammaf(float x)
  173. {
  174. int local_sgngamf = 0;
  175. return (__tgammaf_r(x, &local_sgngamf));
  176. }