123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176 |
- #if defined(LIBM_SCCS) && !defined(lint)
- static char rcsid[] = "$NetBSD: s_log1p.c,v 1.8 1995/05/10 20:47:46 jtc Exp $";
- #endif
- #include "math.h"
- #include "math_private.h"
- #ifdef __STDC__
- static const double
- #else
- static double
- #endif
- ln2_hi = 6.93147180369123816490e-01,
- ln2_lo = 1.90821492927058770002e-10,
- two54 = 1.80143985094819840000e+16,
- Lp1 = 6.666666666666735130e-01,
- Lp2 = 3.999999999940941908e-01,
- Lp3 = 2.857142874366239149e-01,
- Lp4 = 2.222219843214978396e-01,
- Lp5 = 1.818357216161805012e-01,
- Lp6 = 1.531383769920937332e-01,
- Lp7 = 1.479819860511658591e-01;
- #ifdef __STDC__
- static const double zero = 0.0;
- #else
- static double zero = 0.0;
- #endif
- libm_hidden_proto(log1p)
- #ifdef __STDC__
- double log1p(double x)
- #else
- double log1p(x)
- double x;
- #endif
- {
- double hfsq,f=0,c=0,s,z,R,u;
- int32_t k,hx,hu=0,ax;
- GET_HIGH_WORD(hx,x);
- ax = hx&0x7fffffff;
- k = 1;
- if (hx < 0x3FDA827A) {
- if(ax>=0x3ff00000) {
- if(x==-1.0) return -two54/zero;
- else return (x-x)/(x-x);
- }
- if(ax<0x3e200000) {
- if(two54+x>zero
- &&ax<0x3c900000)
- return x;
- else
- return x - x*x*0.5;
- }
- if(hx>0||hx<=((int32_t)0xbfd2bec3)) {
- k=0;f=x;hu=1;}
- }
- if (hx >= 0x7ff00000) return x+x;
- if(k!=0) {
- if(hx<0x43400000) {
- u = 1.0+x;
- GET_HIGH_WORD(hu,u);
- k = (hu>>20)-1023;
- c = (k>0)? 1.0-(u-x):x-(u-1.0);
- c /= u;
- } else {
- u = x;
- GET_HIGH_WORD(hu,u);
- k = (hu>>20)-1023;
- c = 0;
- }
- hu &= 0x000fffff;
- if(hu<0x6a09e) {
- SET_HIGH_WORD(u,hu|0x3ff00000);
- } else {
- k += 1;
- SET_HIGH_WORD(u,hu|0x3fe00000);
- hu = (0x00100000-hu)>>2;
- }
- f = u-1.0;
- }
- hfsq=0.5*f*f;
- if(hu==0) {
- if(f==zero) {if(k==0) return zero;
- else {c += k*ln2_lo; return k*ln2_hi+c;}
- }
- R = hfsq*(1.0-0.66666666666666666*f);
- if(k==0) return f-R; else
- return k*ln2_hi-((R-(k*ln2_lo+c))-f);
- }
- s = f/(2.0+f);
- z = s*s;
- R = z*(Lp1+z*(Lp2+z*(Lp3+z*(Lp4+z*(Lp5+z*(Lp6+z*Lp7))))));
- if(k==0) return f-(hfsq-s*(hfsq+R)); else
- return k*ln2_hi-((hfsq-(s*(hfsq+R)+(k*ln2_lo+c)))-f);
- }
- libm_hidden_def(log1p)
|