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

# · C · 188 lines · 159 code · 16 blank · 13 comment · 22 complexity · 7bf85d4b5c24c75088533798406ae410 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. /* log gamma(x+2), -.5 < x < .5 */
  7. static const float B[] = {
  8. 6.055172732649237E-004,
  9. -1.311620815545743E-003,
  10. 2.863437556468661E-003,
  11. -7.366775108654962E-003,
  12. 2.058355474821512E-002,
  13. -6.735323259371034E-002,
  14. 3.224669577325661E-001,
  15. 4.227843421859038E-001
  16. };
  17. /* log gamma(x+1), -.25 < x < .25 */
  18. static const float C[] = {
  19. 1.369488127325832E-001,
  20. -1.590086327657347E-001,
  21. 1.692415923504637E-001,
  22. -2.067882815621965E-001,
  23. 2.705806208275915E-001,
  24. -4.006931650563372E-001,
  25. 8.224670749082976E-001,
  26. -5.772156501719101E-001
  27. };
  28. /* log( sqrt( 2*pi ) ) */
  29. static const float LS2PI = 0.91893853320467274178;
  30. #define MAXLGM 2.035093e36
  31. static const float PIINV = 0.318309886183790671538;
  32. #include "cephes_mconf.h"
  33. /* Reentrant version */
  34. /* Logarithm of gamma function */
  35. float __lgammaf_r(float x, int* sgngamf);
  36. float __lgammaf_r(float x, int* sgngamf)
  37. {
  38. float p, q, w, z;
  39. float nx, tx;
  40. int i, direction;
  41. *sgngamf = 1;
  42. #ifdef NANS
  43. if (isnan(x))
  44. return (x);
  45. #endif
  46. #ifdef INFINITIES
  47. if (!isfinite(x))
  48. return (x);
  49. #endif
  50. if (x < 0.0)
  51. {
  52. q = -x;
  53. w = __lgammaf_r(q, sgngamf); /* note this modifies sgngam! */
  54. p = floorf(q);
  55. if (p == q)
  56. {
  57. lgsing:
  58. _SET_ERRNO(EDOM);
  59. mtherr("lgamf", SING);
  60. #ifdef INFINITIES
  61. return (INFINITYF);
  62. #else
  63. return( *sgngamf * MAXNUMF );
  64. #endif
  65. }
  66. i = p;
  67. if ((i & 1) == 0)
  68. *sgngamf = -1;
  69. else
  70. *sgngamf = 1;
  71. z = q - p;
  72. if (z > 0.5)
  73. {
  74. p += 1.0;
  75. z = p - q;
  76. }
  77. z = q * sinf(PIF * z);
  78. if (z == 0.0)
  79. goto lgsing;
  80. z = -logf(PIINV * z) - w;
  81. return (z);
  82. }
  83. if (x < 6.5)
  84. {
  85. direction = 0;
  86. z = 1.0;
  87. tx = x;
  88. nx = 0.0;
  89. if (x >= 1.5)
  90. {
  91. while (tx > 2.5)
  92. {
  93. nx -= 1.0;
  94. tx = x + nx;
  95. z *=tx;
  96. }
  97. x += nx - 2.0;
  98. iv1r5:
  99. p = x * polevlf( x, B, 7 );
  100. goto cont;
  101. }
  102. if (x >= 1.25)
  103. {
  104. z *= x;
  105. x -= 1.0; /* x + 1 - 2 */
  106. direction = 1;
  107. goto iv1r5;
  108. }
  109. if (x >= 0.75)
  110. {
  111. x -= 1.0;
  112. p = x * polevlf( x, C, 7 );
  113. q = 0.0;
  114. goto contz;
  115. }
  116. while (tx < 1.5)
  117. {
  118. if (tx == 0.0)
  119. goto lgsing;
  120. z *=tx;
  121. nx += 1.0;
  122. tx = x + nx;
  123. }
  124. direction = 1;
  125. x += nx - 2.0;
  126. p = x * polevlf(x, B, 7);
  127. cont:
  128. if (z < 0.0)
  129. {
  130. *sgngamf = -1;
  131. z = -z;
  132. }
  133. else
  134. {
  135. *sgngamf = 1;
  136. }
  137. q = logf(z);
  138. if (direction)
  139. q = -q;
  140. contz:
  141. return( p + q );
  142. }
  143. if (x > MAXLGM)
  144. {
  145. _SET_ERRNO(ERANGE);
  146. mtherr("lgamf", OVERFLOW);
  147. #ifdef INFINITIES
  148. return (*sgngamf * INFINITYF);
  149. #else
  150. return (*sgngamf * MAXNUMF);
  151. #endif
  152. }
  153. /* Note, though an asymptotic formula could be used for x >= 3,
  154. * there is cancellation error in the following if x < 6.5. */
  155. q = LS2PI - x;
  156. q += (x - 0.5) * logf(x);
  157. if (x <= 1.0e4)
  158. {
  159. z = 1.0/x;
  160. p = z * z;
  161. q += (( 6.789774945028216E-004 * p
  162. - 2.769887652139868E-003 ) * p
  163. + 8.333316229807355E-002 ) * z;
  164. }
  165. return (q);
  166. }
  167. /* This is the C99 version */
  168. float lgammaf(float x)
  169. {
  170. int local_sgngamf = 0;
  171. return (__lgammaf_r(x, &local_sgngamf));
  172. }