s_log1pf.c 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. /* s_log1pf.c -- float version of s_log1p.c.
  2. * Conversion to float by Ian Lance Taylor, Cygnus Support, ian@cygnus.com.
  3. */
  4. /*
  5. * ====================================================
  6. * Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.
  7. *
  8. * Developed at SunPro, a Sun Microsystems, Inc. business.
  9. * Permission to use, copy, modify, and distribute this
  10. * software is freely granted, provided that this notice
  11. * is preserved.
  12. * ====================================================
  13. */
  14. #include "cdefs-compat.h"
  15. //__FBSDID("$FreeBSD: src/lib/msun/src/s_log1pf.c,v 1.12 2008/03/29 16:37:59 das Exp $");
  16. #include <float.h>
  17. #include <openlibm_math.h>
  18. #include "math_private.h"
  19. static const float
  20. ln2_hi = 6.9313812256e-01, /* 0x3f317180 */
  21. ln2_lo = 9.0580006145e-06, /* 0x3717f7d1 */
  22. two25 = 3.355443200e+07, /* 0x4c000000 */
  23. Lp1 = 6.6666668653e-01, /* 3F2AAAAB */
  24. Lp2 = 4.0000000596e-01, /* 3ECCCCCD */
  25. Lp3 = 2.8571429849e-01, /* 3E924925 */
  26. Lp4 = 2.2222198546e-01, /* 3E638E29 */
  27. Lp5 = 1.8183572590e-01, /* 3E3A3325 */
  28. Lp6 = 1.5313838422e-01, /* 3E1CD04F */
  29. Lp7 = 1.4798198640e-01; /* 3E178897 */
  30. static const float zero = 0.0;
  31. OLM_DLLEXPORT float
  32. log1pf(float x)
  33. {
  34. float hfsq,f,c,s,z,R,u;
  35. int32_t k,hx,hu,ax;
  36. GET_FLOAT_WORD(hx,x);
  37. ax = hx&0x7fffffff;
  38. k = 1;
  39. if (hx < 0x3ed413d0) { /* 1+x < sqrt(2)+ */
  40. if(ax>=0x3f800000) { /* x <= -1.0 */
  41. if(x==(float)-1.0) return -two25/zero; /* log1p(-1)=+inf */
  42. else return (x-x)/(x-x); /* log1p(x<-1)=NaN */
  43. }
  44. if(ax<0x38000000) { /* |x| < 2**-15 */
  45. if(two25+x>zero /* raise inexact */
  46. &&ax<0x33800000) /* |x| < 2**-24 */
  47. return x;
  48. else
  49. return x - x*x*(float)0.5;
  50. }
  51. if(hx>0||hx<=((int32_t)0xbe95f619)) {
  52. k=0;f=x;hu=1;} /* sqrt(2)/2- <= 1+x < sqrt(2)+ */
  53. }
  54. if (hx >= 0x7f800000) return x+x;
  55. if(k!=0) {
  56. if(hx<0x5a000000) {
  57. STRICT_ASSIGN(float,u,(float)1.0+x);
  58. GET_FLOAT_WORD(hu,u);
  59. k = (hu>>23)-127;
  60. /* correction term */
  61. c = (k>0)? (float)1.0-(u-x):x-(u-(float)1.0);
  62. c /= u;
  63. } else {
  64. u = x;
  65. GET_FLOAT_WORD(hu,u);
  66. k = (hu>>23)-127;
  67. c = 0;
  68. }
  69. hu &= 0x007fffff;
  70. /*
  71. * The approximation to sqrt(2) used in thresholds is not
  72. * critical. However, the ones used above must give less
  73. * strict bounds than the one here so that the k==0 case is
  74. * never reached from here, since here we have committed to
  75. * using the correction term but don't use it if k==0.
  76. */
  77. if(hu<0x3504f4) { /* u < sqrt(2) */
  78. SET_FLOAT_WORD(u,hu|0x3f800000);/* normalize u */
  79. } else {
  80. k += 1;
  81. SET_FLOAT_WORD(u,hu|0x3f000000); /* normalize u/2 */
  82. hu = (0x00800000-hu)>>2;
  83. }
  84. f = u-(float)1.0;
  85. }
  86. hfsq=(float)0.5*f*f;
  87. if(hu==0) { /* |f| < 2**-20 */
  88. if(f==zero) {
  89. if(k==0) {
  90. return zero;
  91. } else {
  92. c += k*ln2_lo;
  93. return k*ln2_hi+c;
  94. }
  95. }
  96. R = hfsq*((float)1.0-(float)0.66666666666666666*f);
  97. if(k==0) return f-R; else
  98. return k*ln2_hi-((R-(k*ln2_lo+c))-f);
  99. }
  100. s = f/((float)2.0+f);
  101. z = s*s;
  102. R = z*(Lp1+z*(Lp2+z*(Lp3+z*(Lp4+z*(Lp5+z*(Lp6+z*Lp7))))));
  103. if(k==0) return f-(hfsq-s*(hfsq+R)); else
  104. return k*ln2_hi-((hfsq-(s*(hfsq+R)+(k*ln2_lo+c)))-f);
  105. }