mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-10-24 10:10:59 +00:00
parent
d53a344e18
commit
32e289b1d8
11 changed files with 1187 additions and 0 deletions
|
@ -147,6 +147,8 @@ double sqrt(double);
|
||||||
double tan(double);
|
double tan(double);
|
||||||
double tanh(double);
|
double tanh(double);
|
||||||
double trunc(double);
|
double trunc(double);
|
||||||
|
double lgamma(double);
|
||||||
|
double lgamma_r(double, int *);
|
||||||
|
|
||||||
float acosf(float);
|
float acosf(float);
|
||||||
float acoshf(float);
|
float acoshf(float);
|
||||||
|
|
336
libc/tinymath/erf.c
Normal file
336
libc/tinymath/erf.c
Normal file
|
@ -0,0 +1,336 @@
|
||||||
|
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
||||||
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
||||||
|
╚──────────────────────────────────────────────────────────────────────────────╝
|
||||||
|
│ │
|
||||||
|
│ Musl Libc │
|
||||||
|
│ Copyright © 2005-2014 Rich Felker, et al. │
|
||||||
|
│ │
|
||||||
|
│ Permission is hereby granted, free of charge, to any person obtaining │
|
||||||
|
│ a copy of this software and associated documentation files (the │
|
||||||
|
│ "Software"), to deal in the Software without restriction, including │
|
||||||
|
│ without limitation the rights to use, copy, modify, merge, publish, │
|
||||||
|
│ distribute, sublicense, and/or sell copies of the Software, and to │
|
||||||
|
│ permit persons to whom the Software is furnished to do so, subject to │
|
||||||
|
│ the following conditions: │
|
||||||
|
│ │
|
||||||
|
│ The above copyright notice and this permission notice shall be │
|
||||||
|
│ included in all copies or substantial portions of the Software. │
|
||||||
|
│ │
|
||||||
|
│ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, │
|
||||||
|
│ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF │
|
||||||
|
│ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. │
|
||||||
|
│ IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY │
|
||||||
|
│ CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, │
|
||||||
|
│ TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE │
|
||||||
|
│ SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. │
|
||||||
|
│ │
|
||||||
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
||||||
|
#include "libc/math.h"
|
||||||
|
|
||||||
|
asm(".ident\t\"\\n\\n\
|
||||||
|
fdlibm (fdlibm license)\\n\
|
||||||
|
Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.\"");
|
||||||
|
asm(".ident\t\"\\n\\n\
|
||||||
|
Musl libc (MIT License)\\n\
|
||||||
|
Copyright 2005-2014 Rich Felker, et. al.\"");
|
||||||
|
asm(".include \"libc/disclaimer.inc\"");
|
||||||
|
|
||||||
|
/* clang-format off */
|
||||||
|
/* origin: FreeBSD /usr/src/lib/msun/src/s_erf.c */
|
||||||
|
/*
|
||||||
|
* ====================================================
|
||||||
|
* Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.
|
||||||
|
*
|
||||||
|
* Developed at SunPro, a Sun Microsystems, Inc. business.
|
||||||
|
* Permission to use, copy, modify, and distribute this
|
||||||
|
* software is freely granted, provided that this notice
|
||||||
|
* is preserved.
|
||||||
|
* ====================================================
|
||||||
|
*/
|
||||||
|
/* double erf(double x)
|
||||||
|
* double erfc(double x)
|
||||||
|
* x
|
||||||
|
* 2 |\
|
||||||
|
* erf(x) = --------- | exp(-t*t)dt
|
||||||
|
* sqrt(pi) \|
|
||||||
|
* 0
|
||||||
|
*
|
||||||
|
* erfc(x) = 1-erf(x)
|
||||||
|
* Note that
|
||||||
|
* erf(-x) = -erf(x)
|
||||||
|
* erfc(-x) = 2 - erfc(x)
|
||||||
|
*
|
||||||
|
* Method:
|
||||||
|
* 1. For |x| in [0, 0.84375]
|
||||||
|
* erf(x) = x + x*R(x^2)
|
||||||
|
* erfc(x) = 1 - erf(x) if x in [-.84375,0.25]
|
||||||
|
* = 0.5 + ((0.5-x)-x*R) if x in [0.25,0.84375]
|
||||||
|
* where R = P/Q where P is an odd poly of degree 8 and
|
||||||
|
* Q is an odd poly of degree 10.
|
||||||
|
* -57.90
|
||||||
|
* | R - (erf(x)-x)/x | <= 2
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* Remark. The formula is derived by noting
|
||||||
|
* erf(x) = (2/sqrt(pi))*(x - x^3/3 + x^5/10 - x^7/42 + ....)
|
||||||
|
* and that
|
||||||
|
* 2/sqrt(pi) = 1.128379167095512573896158903121545171688
|
||||||
|
* is close to one. The interval is chosen because the fix
|
||||||
|
* point of erf(x) is near 0.6174 (i.e., erf(x)=x when x is
|
||||||
|
* near 0.6174), and by some experiment, 0.84375 is chosen to
|
||||||
|
* guarantee the error is less than one ulp for erf.
|
||||||
|
*
|
||||||
|
* 2. For |x| in [0.84375,1.25], let s = |x| - 1, and
|
||||||
|
* c = 0.84506291151 rounded to single (24 bits)
|
||||||
|
* erf(x) = sign(x) * (c + P1(s)/Q1(s))
|
||||||
|
* erfc(x) = (1-c) - P1(s)/Q1(s) if x > 0
|
||||||
|
* 1+(c+P1(s)/Q1(s)) if x < 0
|
||||||
|
* |P1/Q1 - (erf(|x|)-c)| <= 2**-59.06
|
||||||
|
* Remark: here we use the taylor series expansion at x=1.
|
||||||
|
* erf(1+s) = erf(1) + s*Poly(s)
|
||||||
|
* = 0.845.. + P1(s)/Q1(s)
|
||||||
|
* That is, we use rational approximation to approximate
|
||||||
|
* erf(1+s) - (c = (single)0.84506291151)
|
||||||
|
* Note that |P1/Q1|< 0.078 for x in [0.84375,1.25]
|
||||||
|
* where
|
||||||
|
* P1(s) = degree 6 poly in s
|
||||||
|
* Q1(s) = degree 6 poly in s
|
||||||
|
*
|
||||||
|
* 3. For x in [1.25,1/0.35(~2.857143)],
|
||||||
|
* erfc(x) = (1/x)*exp(-x*x-0.5625+R1/S1)
|
||||||
|
* erf(x) = 1 - erfc(x)
|
||||||
|
* where
|
||||||
|
* R1(z) = degree 7 poly in z, (z=1/x^2)
|
||||||
|
* S1(z) = degree 8 poly in z
|
||||||
|
*
|
||||||
|
* 4. For x in [1/0.35,28]
|
||||||
|
* erfc(x) = (1/x)*exp(-x*x-0.5625+R2/S2) if x > 0
|
||||||
|
* = 2.0 - (1/x)*exp(-x*x-0.5625+R2/S2) if -6<x<0
|
||||||
|
* = 2.0 - tiny (if x <= -6)
|
||||||
|
* erf(x) = sign(x)*(1.0 - erfc(x)) if x < 6, else
|
||||||
|
* erf(x) = sign(x)*(1.0 - tiny)
|
||||||
|
* where
|
||||||
|
* R2(z) = degree 6 poly in z, (z=1/x^2)
|
||||||
|
* S2(z) = degree 7 poly in z
|
||||||
|
*
|
||||||
|
* Note1:
|
||||||
|
* To compute exp(-x*x-0.5625+R/S), let s be a single
|
||||||
|
* precision number and s := x; then
|
||||||
|
* -x*x = -s*s + (s-x)*(s+x)
|
||||||
|
* exp(-x*x-0.5626+R/S) =
|
||||||
|
* exp(-s*s-0.5625)*exp((s-x)*(s+x)+R/S);
|
||||||
|
* Note2:
|
||||||
|
* Here 4 and 5 make use of the asymptotic series
|
||||||
|
* exp(-x*x)
|
||||||
|
* erfc(x) ~ ---------- * ( 1 + Poly(1/x^2) )
|
||||||
|
* x*sqrt(pi)
|
||||||
|
* We use rational approximation to approximate
|
||||||
|
* g(s)=f(1/x^2) = log(erfc(x)*x) - x*x + 0.5625
|
||||||
|
* Here is the error bound for R1/S1 and R2/S2
|
||||||
|
* |R1/S1 - f(x)| < 2**(-62.57)
|
||||||
|
* |R2/S2 - f(x)| < 2**(-61.52)
|
||||||
|
*
|
||||||
|
* 5. For inf > x >= 28
|
||||||
|
* erf(x) = sign(x) *(1 - tiny) (raise inexact)
|
||||||
|
* erfc(x) = tiny*tiny (raise underflow) if x > 0
|
||||||
|
* = 2 - tiny if x<0
|
||||||
|
*
|
||||||
|
* 7. Special case:
|
||||||
|
* erf(0) = 0, erf(inf) = 1, erf(-inf) = -1,
|
||||||
|
* erfc(0) = 1, erfc(inf) = 0, erfc(-inf) = 2,
|
||||||
|
* erfc/erf(NaN) is NaN
|
||||||
|
*/
|
||||||
|
|
||||||
|
static const double
|
||||||
|
erx = 8.45062911510467529297e-01, /* 0x3FEB0AC1, 0x60000000 */
|
||||||
|
/*
|
||||||
|
* Coefficients for approximation to erf on [0,0.84375]
|
||||||
|
*/
|
||||||
|
efx8 = 1.02703333676410069053e+00, /* 0x3FF06EBA, 0x8214DB69 */
|
||||||
|
pp0 = 1.28379167095512558561e-01, /* 0x3FC06EBA, 0x8214DB68 */
|
||||||
|
pp1 = -3.25042107247001499370e-01, /* 0xBFD4CD7D, 0x691CB913 */
|
||||||
|
pp2 = -2.84817495755985104766e-02, /* 0xBF9D2A51, 0xDBD7194F */
|
||||||
|
pp3 = -5.77027029648944159157e-03, /* 0xBF77A291, 0x236668E4 */
|
||||||
|
pp4 = -2.37630166566501626084e-05, /* 0xBEF8EAD6, 0x120016AC */
|
||||||
|
qq1 = 3.97917223959155352819e-01, /* 0x3FD97779, 0xCDDADC09 */
|
||||||
|
qq2 = 6.50222499887672944485e-02, /* 0x3FB0A54C, 0x5536CEBA */
|
||||||
|
qq3 = 5.08130628187576562776e-03, /* 0x3F74D022, 0xC4D36B0F */
|
||||||
|
qq4 = 1.32494738004321644526e-04, /* 0x3F215DC9, 0x221C1A10 */
|
||||||
|
qq5 = -3.96022827877536812320e-06, /* 0xBED09C43, 0x42A26120 */
|
||||||
|
/*
|
||||||
|
* Coefficients for approximation to erf in [0.84375,1.25]
|
||||||
|
*/
|
||||||
|
pa0 = -2.36211856075265944077e-03, /* 0xBF6359B8, 0xBEF77538 */
|
||||||
|
pa1 = 4.14856118683748331666e-01, /* 0x3FDA8D00, 0xAD92B34D */
|
||||||
|
pa2 = -3.72207876035701323847e-01, /* 0xBFD7D240, 0xFBB8C3F1 */
|
||||||
|
pa3 = 3.18346619901161753674e-01, /* 0x3FD45FCA, 0x805120E4 */
|
||||||
|
pa4 = -1.10894694282396677476e-01, /* 0xBFBC6398, 0x3D3E28EC */
|
||||||
|
pa5 = 3.54783043256182359371e-02, /* 0x3FA22A36, 0x599795EB */
|
||||||
|
pa6 = -2.16637559486879084300e-03, /* 0xBF61BF38, 0x0A96073F */
|
||||||
|
qa1 = 1.06420880400844228286e-01, /* 0x3FBB3E66, 0x18EEE323 */
|
||||||
|
qa2 = 5.40397917702171048937e-01, /* 0x3FE14AF0, 0x92EB6F33 */
|
||||||
|
qa3 = 7.18286544141962662868e-02, /* 0x3FB2635C, 0xD99FE9A7 */
|
||||||
|
qa4 = 1.26171219808761642112e-01, /* 0x3FC02660, 0xE763351F */
|
||||||
|
qa5 = 1.36370839120290507362e-02, /* 0x3F8BEDC2, 0x6B51DD1C */
|
||||||
|
qa6 = 1.19844998467991074170e-02, /* 0x3F888B54, 0x5735151D */
|
||||||
|
/*
|
||||||
|
* Coefficients for approximation to erfc in [1.25,1/0.35]
|
||||||
|
*/
|
||||||
|
ra0 = -9.86494403484714822705e-03, /* 0xBF843412, 0x600D6435 */
|
||||||
|
ra1 = -6.93858572707181764372e-01, /* 0xBFE63416, 0xE4BA7360 */
|
||||||
|
ra2 = -1.05586262253232909814e+01, /* 0xC0251E04, 0x41B0E726 */
|
||||||
|
ra3 = -6.23753324503260060396e+01, /* 0xC04F300A, 0xE4CBA38D */
|
||||||
|
ra4 = -1.62396669462573470355e+02, /* 0xC0644CB1, 0x84282266 */
|
||||||
|
ra5 = -1.84605092906711035994e+02, /* 0xC067135C, 0xEBCCABB2 */
|
||||||
|
ra6 = -8.12874355063065934246e+01, /* 0xC0545265, 0x57E4D2F2 */
|
||||||
|
ra7 = -9.81432934416914548592e+00, /* 0xC023A0EF, 0xC69AC25C */
|
||||||
|
sa1 = 1.96512716674392571292e+01, /* 0x4033A6B9, 0xBD707687 */
|
||||||
|
sa2 = 1.37657754143519042600e+02, /* 0x4061350C, 0x526AE721 */
|
||||||
|
sa3 = 4.34565877475229228821e+02, /* 0x407B290D, 0xD58A1A71 */
|
||||||
|
sa4 = 6.45387271733267880336e+02, /* 0x40842B19, 0x21EC2868 */
|
||||||
|
sa5 = 4.29008140027567833386e+02, /* 0x407AD021, 0x57700314 */
|
||||||
|
sa6 = 1.08635005541779435134e+02, /* 0x405B28A3, 0xEE48AE2C */
|
||||||
|
sa7 = 6.57024977031928170135e+00, /* 0x401A47EF, 0x8E484A93 */
|
||||||
|
sa8 = -6.04244152148580987438e-02, /* 0xBFAEEFF2, 0xEE749A62 */
|
||||||
|
/*
|
||||||
|
* Coefficients for approximation to erfc in [1/.35,28]
|
||||||
|
*/
|
||||||
|
rb0 = -9.86494292470009928597e-03, /* 0xBF843412, 0x39E86F4A */
|
||||||
|
rb1 = -7.99283237680523006574e-01, /* 0xBFE993BA, 0x70C285DE */
|
||||||
|
rb2 = -1.77579549177547519889e+01, /* 0xC031C209, 0x555F995A */
|
||||||
|
rb3 = -1.60636384855821916062e+02, /* 0xC064145D, 0x43C5ED98 */
|
||||||
|
rb4 = -6.37566443368389627722e+02, /* 0xC083EC88, 0x1375F228 */
|
||||||
|
rb5 = -1.02509513161107724954e+03, /* 0xC0900461, 0x6A2E5992 */
|
||||||
|
rb6 = -4.83519191608651397019e+02, /* 0xC07E384E, 0x9BDC383F */
|
||||||
|
sb1 = 3.03380607434824582924e+01, /* 0x403E568B, 0x261D5190 */
|
||||||
|
sb2 = 3.25792512996573918826e+02, /* 0x40745CAE, 0x221B9F0A */
|
||||||
|
sb3 = 1.53672958608443695994e+03, /* 0x409802EB, 0x189D5118 */
|
||||||
|
sb4 = 3.19985821950859553908e+03, /* 0x40A8FFB7, 0x688C246A */
|
||||||
|
sb5 = 2.55305040643316442583e+03, /* 0x40A3F219, 0xCEDF3BE6 */
|
||||||
|
sb6 = 4.74528541206955367215e+02, /* 0x407DA874, 0xE79FE763 */
|
||||||
|
sb7 = -2.24409524465858183362e+01; /* 0xC03670E2, 0x42712D62 */
|
||||||
|
|
||||||
|
#define asuint(f) ((union{float _f; uint32_t _i;}){f})._i
|
||||||
|
#define asfloat(i) ((union{uint32_t _i; float _f;}){i})._f
|
||||||
|
#define asuint64(f) ((union{double _f; uint64_t _i;}){f})._i
|
||||||
|
#define asdouble(i) ((union{uint64_t _i; double _f;}){i})._f
|
||||||
|
#define INSERT_WORDS(d,hi,lo) \
|
||||||
|
do { \
|
||||||
|
(d) = asdouble(((uint64_t)(hi)<<32) | (uint32_t)(lo)); \
|
||||||
|
} while (0)
|
||||||
|
#define GET_HIGH_WORD(hi,d) \
|
||||||
|
do { \
|
||||||
|
(hi) = asuint64(d) >> 32; \
|
||||||
|
} while (0)
|
||||||
|
#define GET_LOW_WORD(lo,d) \
|
||||||
|
do { \
|
||||||
|
(lo) = (uint32_t)asuint64(d); \
|
||||||
|
} while (0)
|
||||||
|
#define SET_HIGH_WORD(d,hi) \
|
||||||
|
INSERT_WORDS(d, hi, (uint32_t)asuint64(d))
|
||||||
|
#define SET_LOW_WORD(d,lo) \
|
||||||
|
INSERT_WORDS(d, asuint64(d)>>32, lo)
|
||||||
|
|
||||||
|
static double erfc1(double x)
|
||||||
|
{
|
||||||
|
double_t s,P,Q;
|
||||||
|
|
||||||
|
s = fabs(x) - 1;
|
||||||
|
P = pa0+s*(pa1+s*(pa2+s*(pa3+s*(pa4+s*(pa5+s*pa6)))));
|
||||||
|
Q = 1+s*(qa1+s*(qa2+s*(qa3+s*(qa4+s*(qa5+s*qa6)))));
|
||||||
|
return 1 - erx - P/Q;
|
||||||
|
}
|
||||||
|
|
||||||
|
static double erfc2(uint32_t ix, double x)
|
||||||
|
{
|
||||||
|
double_t s,R,S;
|
||||||
|
double z;
|
||||||
|
|
||||||
|
if (ix < 0x3ff40000) /* |x| < 1.25 */
|
||||||
|
return erfc1(x);
|
||||||
|
|
||||||
|
x = fabs(x);
|
||||||
|
s = 1/(x*x);
|
||||||
|
if (ix < 0x4006db6d) { /* |x| < 1/.35 ~ 2.85714 */
|
||||||
|
R = ra0+s*(ra1+s*(ra2+s*(ra3+s*(ra4+s*(
|
||||||
|
ra5+s*(ra6+s*ra7))))));
|
||||||
|
S = 1.0+s*(sa1+s*(sa2+s*(sa3+s*(sa4+s*(
|
||||||
|
sa5+s*(sa6+s*(sa7+s*sa8)))))));
|
||||||
|
} else { /* |x| > 1/.35 */
|
||||||
|
R = rb0+s*(rb1+s*(rb2+s*(rb3+s*(rb4+s*(
|
||||||
|
rb5+s*rb6)))));
|
||||||
|
S = 1.0+s*(sb1+s*(sb2+s*(sb3+s*(sb4+s*(
|
||||||
|
sb5+s*(sb6+s*sb7))))));
|
||||||
|
}
|
||||||
|
z = x;
|
||||||
|
SET_LOW_WORD(z,0);
|
||||||
|
return exp(-z*z-0.5625)*exp((z-x)*(z+x)+R/S)/x;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns error function of 𝑥.
|
||||||
|
*/
|
||||||
|
double erf(double x)
|
||||||
|
{
|
||||||
|
double r,s,z,y;
|
||||||
|
uint32_t ix;
|
||||||
|
int sign;
|
||||||
|
|
||||||
|
GET_HIGH_WORD(ix, x);
|
||||||
|
sign = ix>>31;
|
||||||
|
ix &= 0x7fffffff;
|
||||||
|
if (ix >= 0x7ff00000) {
|
||||||
|
/* erf(nan)=nan, erf(+-inf)=+-1 */
|
||||||
|
return 1-2*sign + 1/x;
|
||||||
|
}
|
||||||
|
if (ix < 0x3feb0000) { /* |x| < 0.84375 */
|
||||||
|
if (ix < 0x3e300000) { /* |x| < 2**-28 */
|
||||||
|
/* avoid underflow */
|
||||||
|
return 0.125*(8*x + efx8*x);
|
||||||
|
}
|
||||||
|
z = x*x;
|
||||||
|
r = pp0+z*(pp1+z*(pp2+z*(pp3+z*pp4)));
|
||||||
|
s = 1.0+z*(qq1+z*(qq2+z*(qq3+z*(qq4+z*qq5))));
|
||||||
|
y = r/s;
|
||||||
|
return x + x*y;
|
||||||
|
}
|
||||||
|
if (ix < 0x40180000) /* 0.84375 <= |x| < 6 */
|
||||||
|
y = 1 - erfc2(ix,x);
|
||||||
|
else
|
||||||
|
y = 1 - 0x1p-1022;
|
||||||
|
return sign ? -y : y;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns complementary error function of 𝑥.
|
||||||
|
*/
|
||||||
|
double erfc(double x)
|
||||||
|
{
|
||||||
|
double r,s,z,y;
|
||||||
|
uint32_t ix;
|
||||||
|
int sign;
|
||||||
|
|
||||||
|
GET_HIGH_WORD(ix, x);
|
||||||
|
sign = ix>>31;
|
||||||
|
ix &= 0x7fffffff;
|
||||||
|
if (ix >= 0x7ff00000) {
|
||||||
|
/* erfc(nan)=nan, erfc(+-inf)=0,2 */
|
||||||
|
return 2*sign + 1/x;
|
||||||
|
}
|
||||||
|
if (ix < 0x3feb0000) { /* |x| < 0.84375 */
|
||||||
|
if (ix < 0x3c700000) /* |x| < 2**-56 */
|
||||||
|
return 1.0 - x;
|
||||||
|
z = x*x;
|
||||||
|
r = pp0+z*(pp1+z*(pp2+z*(pp3+z*pp4)));
|
||||||
|
s = 1.0+z*(qq1+z*(qq2+z*(qq3+z*(qq4+z*qq5))));
|
||||||
|
y = r/s;
|
||||||
|
if (sign || ix < 0x3fd00000) { /* x < 1/4 */
|
||||||
|
return 1.0 - (x+x*y);
|
||||||
|
}
|
||||||
|
return 0.5 - (x - 0.5 + x*y);
|
||||||
|
}
|
||||||
|
if (ix < 0x403c0000) { /* 0.84375 <= |x| < 28 */
|
||||||
|
return sign ? 2 - erfc2(ix,x) : erfc2(ix,x);
|
||||||
|
}
|
||||||
|
return sign ? 2 - 0x1p-1022 : 0x1p-1022*0x1p-1022;
|
||||||
|
}
|
329
libc/tinymath/gamma.c
Normal file
329
libc/tinymath/gamma.c
Normal file
|
@ -0,0 +1,329 @@
|
||||||
|
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
||||||
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
||||||
|
╚──────────────────────────────────────────────────────────────────────────────╝
|
||||||
|
│ │
|
||||||
|
│ Musl Libc │
|
||||||
|
│ Copyright © 2005-2014 Rich Felker, et al. │
|
||||||
|
│ │
|
||||||
|
│ Permission is hereby granted, free of charge, to any person obtaining │
|
||||||
|
│ a copy of this software and associated documentation files (the │
|
||||||
|
│ "Software"), to deal in the Software without restriction, including │
|
||||||
|
│ without limitation the rights to use, copy, modify, merge, publish, │
|
||||||
|
│ distribute, sublicense, and/or sell copies of the Software, and to │
|
||||||
|
│ permit persons to whom the Software is furnished to do so, subject to │
|
||||||
|
│ the following conditions: │
|
||||||
|
│ │
|
||||||
|
│ The above copyright notice and this permission notice shall be │
|
||||||
|
│ included in all copies or substantial portions of the Software. │
|
||||||
|
│ │
|
||||||
|
│ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, │
|
||||||
|
│ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF │
|
||||||
|
│ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. │
|
||||||
|
│ IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY │
|
||||||
|
│ CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, │
|
||||||
|
│ TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE │
|
||||||
|
│ SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. │
|
||||||
|
│ │
|
||||||
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
||||||
|
#include "libc/math.h"
|
||||||
|
|
||||||
|
asm(".ident\t\"\\n\\n\
|
||||||
|
fdlibm (fdlibm license)\\n\
|
||||||
|
Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.\"");
|
||||||
|
asm(".ident\t\"\\n\\n\
|
||||||
|
Musl libc (MIT License)\\n\
|
||||||
|
Copyright 2005-2014 Rich Felker, et. al.\"");
|
||||||
|
asm(".include \"libc/disclaimer.inc\"");
|
||||||
|
|
||||||
|
double __sin(double, double, int);
|
||||||
|
double __cos(double, double);
|
||||||
|
|
||||||
|
/* clang-format off */
|
||||||
|
/* origin: FreeBSD /usr/src/lib/msun/src/e_lgamma_r.c */
|
||||||
|
/*
|
||||||
|
* ====================================================
|
||||||
|
* Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.
|
||||||
|
*
|
||||||
|
* Developed at SunSoft, a Sun Microsystems, Inc. business.
|
||||||
|
* Permission to use, copy, modify, and distribute this
|
||||||
|
* software is freely granted, provided that this notice
|
||||||
|
* is preserved.
|
||||||
|
* ====================================================
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
/* lgamma_r(x, signgamp)
|
||||||
|
* Reentrant version of the logarithm of the Gamma function
|
||||||
|
* with user provide pointer for the sign of Gamma(x).
|
||||||
|
*
|
||||||
|
* Method:
|
||||||
|
* 1. Argument Reduction for 0 < x <= 8
|
||||||
|
* Since gamma(1+s)=s*gamma(s), for x in [0,8], we may
|
||||||
|
* reduce x to a number in [1.5,2.5] by
|
||||||
|
* lgamma(1+s) = log(s) + lgamma(s)
|
||||||
|
* for example,
|
||||||
|
* lgamma(7.3) = log(6.3) + lgamma(6.3)
|
||||||
|
* = log(6.3*5.3) + lgamma(5.3)
|
||||||
|
* = log(6.3*5.3*4.3*3.3*2.3) + lgamma(2.3)
|
||||||
|
* 2. Polynomial approximation of lgamma around its
|
||||||
|
* minimun ymin=1.461632144968362245 to maintain monotonicity.
|
||||||
|
* On [ymin-0.23, ymin+0.27] (i.e., [1.23164,1.73163]), use
|
||||||
|
* Let z = x-ymin;
|
||||||
|
* lgamma(x) = -1.214862905358496078218 + z^2*poly(z)
|
||||||
|
* where
|
||||||
|
* poly(z) is a 14 degree polynomial.
|
||||||
|
* 2. Rational approximation in the primary interval [2,3]
|
||||||
|
* We use the following approximation:
|
||||||
|
* s = x-2.0;
|
||||||
|
* lgamma(x) = 0.5*s + s*P(s)/Q(s)
|
||||||
|
* with accuracy
|
||||||
|
* |P/Q - (lgamma(x)-0.5s)| < 2**-61.71
|
||||||
|
* Our algorithms are based on the following observation
|
||||||
|
*
|
||||||
|
* zeta(2)-1 2 zeta(3)-1 3
|
||||||
|
* lgamma(2+s) = s*(1-Euler) + --------- * s - --------- * s + ...
|
||||||
|
* 2 3
|
||||||
|
*
|
||||||
|
* where Euler = 0.5771... is the Euler constant, which is very
|
||||||
|
* close to 0.5.
|
||||||
|
*
|
||||||
|
* 3. For x>=8, we have
|
||||||
|
* lgamma(x)~(x-0.5)log(x)-x+0.5*log(2pi)+1/(12x)-1/(360x**3)+....
|
||||||
|
* (better formula:
|
||||||
|
* lgamma(x)~(x-0.5)*(log(x)-1)-.5*(log(2pi)-1) + ...)
|
||||||
|
* Let z = 1/x, then we approximation
|
||||||
|
* f(z) = lgamma(x) - (x-0.5)(log(x)-1)
|
||||||
|
* by
|
||||||
|
* 3 5 11
|
||||||
|
* w = w0 + w1*z + w2*z + w3*z + ... + w6*z
|
||||||
|
* where
|
||||||
|
* |w - f(z)| < 2**-58.74
|
||||||
|
*
|
||||||
|
* 4. For negative x, since (G is gamma function)
|
||||||
|
* -x*G(-x)*G(x) = pi/sin(pi*x),
|
||||||
|
* we have
|
||||||
|
* G(x) = pi/(sin(pi*x)*(-x)*G(-x))
|
||||||
|
* since G(-x) is positive, sign(G(x)) = sign(sin(pi*x)) for x<0
|
||||||
|
* Hence, for x<0, signgam = sign(sin(pi*x)) and
|
||||||
|
* lgamma(x) = log(|Gamma(x)|)
|
||||||
|
* = log(pi/(|x*sin(pi*x)|)) - lgamma(-x);
|
||||||
|
* Note: one should avoid compute pi*(-x) directly in the
|
||||||
|
* computation of sin(pi*(-x)).
|
||||||
|
*
|
||||||
|
* 5. Special Cases
|
||||||
|
* lgamma(2+s) ~ s*(1-Euler) for tiny s
|
||||||
|
* lgamma(1) = lgamma(2) = 0
|
||||||
|
* lgamma(x) ~ -log(|x|) for tiny x
|
||||||
|
* lgamma(0) = lgamma(neg.integer) = inf and raise divide-by-zero
|
||||||
|
* lgamma(inf) = inf
|
||||||
|
* lgamma(-inf) = inf (bug for bug compatible with C99!?)
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
static const double
|
||||||
|
pi = 3.14159265358979311600e+00, /* 0x400921FB, 0x54442D18 */
|
||||||
|
a0 = 7.72156649015328655494e-02, /* 0x3FB3C467, 0xE37DB0C8 */
|
||||||
|
a1 = 3.22467033424113591611e-01, /* 0x3FD4A34C, 0xC4A60FAD */
|
||||||
|
a2 = 6.73523010531292681824e-02, /* 0x3FB13E00, 0x1A5562A7 */
|
||||||
|
a3 = 2.05808084325167332806e-02, /* 0x3F951322, 0xAC92547B */
|
||||||
|
a4 = 7.38555086081402883957e-03, /* 0x3F7E404F, 0xB68FEFE8 */
|
||||||
|
a5 = 2.89051383673415629091e-03, /* 0x3F67ADD8, 0xCCB7926B */
|
||||||
|
a6 = 1.19270763183362067845e-03, /* 0x3F538A94, 0x116F3F5D */
|
||||||
|
a7 = 5.10069792153511336608e-04, /* 0x3F40B6C6, 0x89B99C00 */
|
||||||
|
a8 = 2.20862790713908385557e-04, /* 0x3F2CF2EC, 0xED10E54D */
|
||||||
|
a9 = 1.08011567247583939954e-04, /* 0x3F1C5088, 0x987DFB07 */
|
||||||
|
a10 = 2.52144565451257326939e-05, /* 0x3EFA7074, 0x428CFA52 */
|
||||||
|
a11 = 4.48640949618915160150e-05, /* 0x3F07858E, 0x90A45837 */
|
||||||
|
tc = 1.46163214496836224576e+00, /* 0x3FF762D8, 0x6356BE3F */
|
||||||
|
tf = -1.21486290535849611461e-01, /* 0xBFBF19B9, 0xBCC38A42 */
|
||||||
|
/* tt = -(tail of tf) */
|
||||||
|
tt = -3.63867699703950536541e-18, /* 0xBC50C7CA, 0xA48A971F */
|
||||||
|
t0 = 4.83836122723810047042e-01, /* 0x3FDEF72B, 0xC8EE38A2 */
|
||||||
|
t1 = -1.47587722994593911752e-01, /* 0xBFC2E427, 0x8DC6C509 */
|
||||||
|
t2 = 6.46249402391333854778e-02, /* 0x3FB08B42, 0x94D5419B */
|
||||||
|
t3 = -3.27885410759859649565e-02, /* 0xBFA0C9A8, 0xDF35B713 */
|
||||||
|
t4 = 1.79706750811820387126e-02, /* 0x3F9266E7, 0x970AF9EC */
|
||||||
|
t5 = -1.03142241298341437450e-02, /* 0xBF851F9F, 0xBA91EC6A */
|
||||||
|
t6 = 6.10053870246291332635e-03, /* 0x3F78FCE0, 0xE370E344 */
|
||||||
|
t7 = -3.68452016781138256760e-03, /* 0xBF6E2EFF, 0xB3E914D7 */
|
||||||
|
t8 = 2.25964780900612472250e-03, /* 0x3F6282D3, 0x2E15C915 */
|
||||||
|
t9 = -1.40346469989232843813e-03, /* 0xBF56FE8E, 0xBF2D1AF1 */
|
||||||
|
t10 = 8.81081882437654011382e-04, /* 0x3F4CDF0C, 0xEF61A8E9 */
|
||||||
|
t11 = -5.38595305356740546715e-04, /* 0xBF41A610, 0x9C73E0EC */
|
||||||
|
t12 = 3.15632070903625950361e-04, /* 0x3F34AF6D, 0x6C0EBBF7 */
|
||||||
|
t13 = -3.12754168375120860518e-04, /* 0xBF347F24, 0xECC38C38 */
|
||||||
|
t14 = 3.35529192635519073543e-04, /* 0x3F35FD3E, 0xE8C2D3F4 */
|
||||||
|
u0 = -7.72156649015328655494e-02, /* 0xBFB3C467, 0xE37DB0C8 */
|
||||||
|
u1 = 6.32827064025093366517e-01, /* 0x3FE4401E, 0x8B005DFF */
|
||||||
|
u2 = 1.45492250137234768737e+00, /* 0x3FF7475C, 0xD119BD6F */
|
||||||
|
u3 = 9.77717527963372745603e-01, /* 0x3FEF4976, 0x44EA8450 */
|
||||||
|
u4 = 2.28963728064692451092e-01, /* 0x3FCD4EAE, 0xF6010924 */
|
||||||
|
u5 = 1.33810918536787660377e-02, /* 0x3F8B678B, 0xBF2BAB09 */
|
||||||
|
v1 = 2.45597793713041134822e+00, /* 0x4003A5D7, 0xC2BD619C */
|
||||||
|
v2 = 2.12848976379893395361e+00, /* 0x40010725, 0xA42B18F5 */
|
||||||
|
v3 = 7.69285150456672783825e-01, /* 0x3FE89DFB, 0xE45050AF */
|
||||||
|
v4 = 1.04222645593369134254e-01, /* 0x3FBAAE55, 0xD6537C88 */
|
||||||
|
v5 = 3.21709242282423911810e-03, /* 0x3F6A5ABB, 0x57D0CF61 */
|
||||||
|
s0 = -7.72156649015328655494e-02, /* 0xBFB3C467, 0xE37DB0C8 */
|
||||||
|
s1 = 2.14982415960608852501e-01, /* 0x3FCB848B, 0x36E20878 */
|
||||||
|
s2 = 3.25778796408930981787e-01, /* 0x3FD4D98F, 0x4F139F59 */
|
||||||
|
s3 = 1.46350472652464452805e-01, /* 0x3FC2BB9C, 0xBEE5F2F7 */
|
||||||
|
s4 = 2.66422703033638609560e-02, /* 0x3F9B481C, 0x7E939961 */
|
||||||
|
s5 = 1.84028451407337715652e-03, /* 0x3F5E26B6, 0x7368F239 */
|
||||||
|
s6 = 3.19475326584100867617e-05, /* 0x3F00BFEC, 0xDD17E945 */
|
||||||
|
r1 = 1.39200533467621045958e+00, /* 0x3FF645A7, 0x62C4AB74 */
|
||||||
|
r2 = 7.21935547567138069525e-01, /* 0x3FE71A18, 0x93D3DCDC */
|
||||||
|
r3 = 1.71933865632803078993e-01, /* 0x3FC601ED, 0xCCFBDF27 */
|
||||||
|
r4 = 1.86459191715652901344e-02, /* 0x3F9317EA, 0x742ED475 */
|
||||||
|
r5 = 7.77942496381893596434e-04, /* 0x3F497DDA, 0xCA41A95B */
|
||||||
|
r6 = 7.32668430744625636189e-06, /* 0x3EDEBAF7, 0xA5B38140 */
|
||||||
|
w0 = 4.18938533204672725052e-01, /* 0x3FDACFE3, 0x90C97D69 */
|
||||||
|
w1 = 8.33333333333329678849e-02, /* 0x3FB55555, 0x5555553B */
|
||||||
|
w2 = -2.77777777728775536470e-03, /* 0xBF66C16C, 0x16B02E5C */
|
||||||
|
w3 = 7.93650558643019558500e-04, /* 0x3F4A019F, 0x98CF38B6 */
|
||||||
|
w4 = -5.95187557450339963135e-04, /* 0xBF4380CB, 0x8C0FE741 */
|
||||||
|
w5 = 8.36339918996282139126e-04, /* 0x3F4B67BA, 0x4CDAD5D1 */
|
||||||
|
w6 = -1.63092934096575273989e-03; /* 0xBF5AB89D, 0x0B9E43E4 */
|
||||||
|
|
||||||
|
/* sin(pi*x) assuming x > 2^-100, if sin(pi*x)==0 the sign is arbitrary */
|
||||||
|
static double sin_pi(double x)
|
||||||
|
{
|
||||||
|
int n;
|
||||||
|
|
||||||
|
/* spurious inexact if odd int */
|
||||||
|
x = 2.0*(x*0.5 - floor(x*0.5)); /* x mod 2.0 */
|
||||||
|
|
||||||
|
n = (int)(x*4.0);
|
||||||
|
n = (n+1)/2;
|
||||||
|
x -= n*0.5f;
|
||||||
|
x *= pi;
|
||||||
|
|
||||||
|
switch (n) {
|
||||||
|
default: /* case 4: */
|
||||||
|
case 0: return __sin(x, 0.0, 0);
|
||||||
|
case 1: return __cos(x, 0.0);
|
||||||
|
case 2: return __sin(-x, 0.0, 0);
|
||||||
|
case 3: return -__cos(x, 0.0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
double lgamma_r(double x, int *signgamp)
|
||||||
|
{
|
||||||
|
union {double f; uint64_t i;} u = {x};
|
||||||
|
double_t t,y,z,nadj,p,p1,p2,p3,q,r,w;
|
||||||
|
uint32_t ix;
|
||||||
|
int sign,i;
|
||||||
|
|
||||||
|
/* purge off +-inf, NaN, +-0, tiny and negative arguments */
|
||||||
|
*signgamp = 1;
|
||||||
|
sign = u.i>>63;
|
||||||
|
ix = u.i>>32 & 0x7fffffff;
|
||||||
|
if (ix >= 0x7ff00000)
|
||||||
|
return x*x;
|
||||||
|
if (ix < (0x3ff-70)<<20) { /* |x|<2**-70, return -log(|x|) */
|
||||||
|
if(sign) {
|
||||||
|
x = -x;
|
||||||
|
*signgamp = -1;
|
||||||
|
}
|
||||||
|
return -log(x);
|
||||||
|
}
|
||||||
|
if (sign) {
|
||||||
|
x = -x;
|
||||||
|
t = sin_pi(x);
|
||||||
|
if (t == 0.0) /* -integer */
|
||||||
|
return 1.0/(x-x);
|
||||||
|
if (t > 0.0)
|
||||||
|
*signgamp = -1;
|
||||||
|
else
|
||||||
|
t = -t;
|
||||||
|
nadj = log(pi/(t*x));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* purge off 1 and 2 */
|
||||||
|
if ((ix == 0x3ff00000 || ix == 0x40000000) && (uint32_t)u.i == 0)
|
||||||
|
r = 0;
|
||||||
|
/* for x < 2.0 */
|
||||||
|
else if (ix < 0x40000000) {
|
||||||
|
if (ix <= 0x3feccccc) { /* lgamma(x) = lgamma(x+1)-log(x) */
|
||||||
|
r = -log(x);
|
||||||
|
if (ix >= 0x3FE76944) {
|
||||||
|
y = 1.0 - x;
|
||||||
|
i = 0;
|
||||||
|
} else if (ix >= 0x3FCDA661) {
|
||||||
|
y = x - (tc-1.0);
|
||||||
|
i = 1;
|
||||||
|
} else {
|
||||||
|
y = x;
|
||||||
|
i = 2;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
r = 0.0;
|
||||||
|
if (ix >= 0x3FFBB4C3) { /* [1.7316,2] */
|
||||||
|
y = 2.0 - x;
|
||||||
|
i = 0;
|
||||||
|
} else if(ix >= 0x3FF3B4C4) { /* [1.23,1.73] */
|
||||||
|
y = x - tc;
|
||||||
|
i = 1;
|
||||||
|
} else {
|
||||||
|
y = x - 1.0;
|
||||||
|
i = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
switch (i) {
|
||||||
|
case 0:
|
||||||
|
z = y*y;
|
||||||
|
p1 = a0+z*(a2+z*(a4+z*(a6+z*(a8+z*a10))));
|
||||||
|
p2 = z*(a1+z*(a3+z*(a5+z*(a7+z*(a9+z*a11)))));
|
||||||
|
p = y*p1+p2;
|
||||||
|
r += (p-0.5*y);
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
z = y*y;
|
||||||
|
w = z*y;
|
||||||
|
p1 = t0+w*(t3+w*(t6+w*(t9 +w*t12))); /* parallel comp */
|
||||||
|
p2 = t1+w*(t4+w*(t7+w*(t10+w*t13)));
|
||||||
|
p3 = t2+w*(t5+w*(t8+w*(t11+w*t14)));
|
||||||
|
p = z*p1-(tt-w*(p2+y*p3));
|
||||||
|
r += tf + p;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
p1 = y*(u0+y*(u1+y*(u2+y*(u3+y*(u4+y*u5)))));
|
||||||
|
p2 = 1.0+y*(v1+y*(v2+y*(v3+y*(v4+y*v5))));
|
||||||
|
r += -0.5*y + p1/p2;
|
||||||
|
}
|
||||||
|
} else if (ix < 0x40200000) { /* x < 8.0 */
|
||||||
|
i = (int)x;
|
||||||
|
y = x - (double)i;
|
||||||
|
p = y*(s0+y*(s1+y*(s2+y*(s3+y*(s4+y*(s5+y*s6))))));
|
||||||
|
q = 1.0+y*(r1+y*(r2+y*(r3+y*(r4+y*(r5+y*r6)))));
|
||||||
|
r = 0.5*y+p/q;
|
||||||
|
z = 1.0; /* lgamma(1+s) = log(s) + lgamma(s) */
|
||||||
|
switch (i) {
|
||||||
|
case 7: z *= y + 6.0; /* FALLTHRU */
|
||||||
|
case 6: z *= y + 5.0; /* FALLTHRU */
|
||||||
|
case 5: z *= y + 4.0; /* FALLTHRU */
|
||||||
|
case 4: z *= y + 3.0; /* FALLTHRU */
|
||||||
|
case 3: z *= y + 2.0; /* FALLTHRU */
|
||||||
|
r += log(z);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else if (ix < 0x43900000) { /* 8.0 <= x < 2**58 */
|
||||||
|
t = log(x);
|
||||||
|
z = 1.0/x;
|
||||||
|
y = z*z;
|
||||||
|
w = w0+z*(w1+y*(w2+y*(w3+y*(w4+y*(w5+y*w6)))));
|
||||||
|
r = (x-0.5)*(t-1.0)+w;
|
||||||
|
} else /* 2**58 <= x <= inf */
|
||||||
|
r = x*(log(x)-1.0);
|
||||||
|
if (sign)
|
||||||
|
r = nadj - r;
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns natural logarithm of absolute value of gamma function.
|
||||||
|
*/
|
||||||
|
double lgamma(double x)
|
||||||
|
{
|
||||||
|
extern int __signgam;
|
||||||
|
return lgamma_r(x, &__signgam);
|
||||||
|
}
|
105
libc/tinymath/kcos.c
Normal file
105
libc/tinymath/kcos.c
Normal file
|
@ -0,0 +1,105 @@
|
||||||
|
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
||||||
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
||||||
|
╚──────────────────────────────────────────────────────────────────────────────╝
|
||||||
|
│ │
|
||||||
|
│ Musl Libc │
|
||||||
|
│ Copyright © 2005-2014 Rich Felker, et al. │
|
||||||
|
│ │
|
||||||
|
│ Permission is hereby granted, free of charge, to any person obtaining │
|
||||||
|
│ a copy of this software and associated documentation files (the │
|
||||||
|
│ "Software"), to deal in the Software without restriction, including │
|
||||||
|
│ without limitation the rights to use, copy, modify, merge, publish, │
|
||||||
|
│ distribute, sublicense, and/or sell copies of the Software, and to │
|
||||||
|
│ permit persons to whom the Software is furnished to do so, subject to │
|
||||||
|
│ the following conditions: │
|
||||||
|
│ │
|
||||||
|
│ The above copyright notice and this permission notice shall be │
|
||||||
|
│ included in all copies or substantial portions of the Software. │
|
||||||
|
│ │
|
||||||
|
│ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, │
|
||||||
|
│ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF │
|
||||||
|
│ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. │
|
||||||
|
│ IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY │
|
||||||
|
│ CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, │
|
||||||
|
│ TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE │
|
||||||
|
│ SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. │
|
||||||
|
│ │
|
||||||
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
||||||
|
#include "libc/math.h"
|
||||||
|
|
||||||
|
asm(".ident\t\"\\n\\n\
|
||||||
|
fdlibm (fdlibm license)\\n\
|
||||||
|
Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.\"");
|
||||||
|
asm(".ident\t\"\\n\\n\
|
||||||
|
Musl libc (MIT License)\\n\
|
||||||
|
Copyright 2005-2014 Rich Felker, et. al.\"");
|
||||||
|
asm(".include \"libc/disclaimer.inc\"");
|
||||||
|
|
||||||
|
/* clang-format off */
|
||||||
|
/* origin: FreeBSD /usr/src/lib/msun/src/k_cos.c */
|
||||||
|
/*
|
||||||
|
* ====================================================
|
||||||
|
* Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.
|
||||||
|
*
|
||||||
|
* Developed at SunSoft, a Sun Microsystems, Inc. business.
|
||||||
|
* Permission to use, copy, modify, and distribute this
|
||||||
|
* software is freely granted, provided that this notice
|
||||||
|
* is preserved.
|
||||||
|
* ====================================================
|
||||||
|
*/
|
||||||
|
/*
|
||||||
|
* __cos( x, y )
|
||||||
|
* kernel cos function on [-pi/4, pi/4], pi/4 ~ 0.785398164
|
||||||
|
* Input x is assumed to be bounded by ~pi/4 in magnitude.
|
||||||
|
* Input y is the tail of x.
|
||||||
|
*
|
||||||
|
* Algorithm
|
||||||
|
* 1. Since cos(-x) = cos(x), we need only to consider positive x.
|
||||||
|
* 2. if x < 2^-27 (hx<0x3e400000 0), return 1 with inexact if x!=0.
|
||||||
|
* 3. cos(x) is approximated by a polynomial of degree 14 on
|
||||||
|
* [0,pi/4]
|
||||||
|
* 4 14
|
||||||
|
* cos(x) ~ 1 - x*x/2 + C1*x + ... + C6*x
|
||||||
|
* where the remez error is
|
||||||
|
*
|
||||||
|
* | 2 4 6 8 10 12 14 | -58
|
||||||
|
* |cos(x)-(1-.5*x +C1*x +C2*x +C3*x +C4*x +C5*x +C6*x )| <= 2
|
||||||
|
* | |
|
||||||
|
*
|
||||||
|
* 4 6 8 10 12 14
|
||||||
|
* 4. let r = C1*x +C2*x +C3*x +C4*x +C5*x +C6*x , then
|
||||||
|
* cos(x) ~ 1 - x*x/2 + r
|
||||||
|
* since cos(x+y) ~ cos(x) - sin(x)*y
|
||||||
|
* ~ cos(x) - x*y,
|
||||||
|
* a correction term is necessary in cos(x) and hence
|
||||||
|
* cos(x+y) = 1 - (x*x/2 - (r - x*y))
|
||||||
|
* For better accuracy, rearrange to
|
||||||
|
* cos(x+y) ~ w + (tmp + (r-x*y))
|
||||||
|
* where w = 1 - x*x/2 and tmp is a tiny correction term
|
||||||
|
* (1 - x*x/2 == w + tmp exactly in infinite precision).
|
||||||
|
* The exactness of w + tmp in infinite precision depends on w
|
||||||
|
* and tmp having the same precision as x. If they have extra
|
||||||
|
* precision due to compiler bugs, then the extra precision is
|
||||||
|
* only good provided it is retained in all terms of the final
|
||||||
|
* expression for cos(). Retention happens in all cases tested
|
||||||
|
* under FreeBSD, so don't pessimize things by forcibly clipping
|
||||||
|
* any extra precision in w.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define C1 4.16666666666666019037e-02
|
||||||
|
#define C2 -1.38888888888741095749e-03
|
||||||
|
#define C3 2.48015872894767294178e-05
|
||||||
|
#define C4 -2.75573143513906633035e-07
|
||||||
|
#define C5 2.08757232129817482790e-09
|
||||||
|
#define C6 -1.13596475577881948265e-11
|
||||||
|
|
||||||
|
double __cos(double x, double y)
|
||||||
|
{
|
||||||
|
double_t hz,z,r,w;
|
||||||
|
z = x*x;
|
||||||
|
w = z*z;
|
||||||
|
r = z*(C1+z*(C2+z*C3)) + w*w*(C4+z*(C5+z*C6));
|
||||||
|
hz = 0.5*z;
|
||||||
|
w = 1.0-hz;
|
||||||
|
return w + (((1.0-w)-hz) + (z*r-x*y));
|
||||||
|
}
|
98
libc/tinymath/ksin.c
Normal file
98
libc/tinymath/ksin.c
Normal file
|
@ -0,0 +1,98 @@
|
||||||
|
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
||||||
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
||||||
|
╚──────────────────────────────────────────────────────────────────────────────╝
|
||||||
|
│ │
|
||||||
|
│ Musl Libc │
|
||||||
|
│ Copyright © 2005-2014 Rich Felker, et al. │
|
||||||
|
│ │
|
||||||
|
│ Permission is hereby granted, free of charge, to any person obtaining │
|
||||||
|
│ a copy of this software and associated documentation files (the │
|
||||||
|
│ "Software"), to deal in the Software without restriction, including │
|
||||||
|
│ without limitation the rights to use, copy, modify, merge, publish, │
|
||||||
|
│ distribute, sublicense, and/or sell copies of the Software, and to │
|
||||||
|
│ permit persons to whom the Software is furnished to do so, subject to │
|
||||||
|
│ the following conditions: │
|
||||||
|
│ │
|
||||||
|
│ The above copyright notice and this permission notice shall be │
|
||||||
|
│ included in all copies or substantial portions of the Software. │
|
||||||
|
│ │
|
||||||
|
│ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, │
|
||||||
|
│ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF │
|
||||||
|
│ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. │
|
||||||
|
│ IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY │
|
||||||
|
│ CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, │
|
||||||
|
│ TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE │
|
||||||
|
│ SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. │
|
||||||
|
│ │
|
||||||
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
||||||
|
#include "libc/math.h"
|
||||||
|
|
||||||
|
asm(".ident\t\"\\n\\n\
|
||||||
|
fdlibm (fdlibm license)\\n\
|
||||||
|
Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.\"");
|
||||||
|
asm(".ident\t\"\\n\\n\
|
||||||
|
Musl libc (MIT License)\\n\
|
||||||
|
Copyright 2005-2014 Rich Felker, et. al.\"");
|
||||||
|
asm(".include \"libc/disclaimer.inc\"");
|
||||||
|
|
||||||
|
/* clang-format off */
|
||||||
|
/* origin: FreeBSD /usr/src/lib/msun/src/k_sin.c */
|
||||||
|
/*
|
||||||
|
* ====================================================
|
||||||
|
* Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.
|
||||||
|
*
|
||||||
|
* Developed at SunSoft, a Sun Microsystems, Inc. business.
|
||||||
|
* Permission to use, copy, modify, and distribute this
|
||||||
|
* software is freely granted, provided that this notice
|
||||||
|
* is preserved.
|
||||||
|
* ====================================================
|
||||||
|
*/
|
||||||
|
/* __sin( x, y, iy)
|
||||||
|
* kernel sin function on ~[-pi/4, pi/4] (except on -0), pi/4 ~ 0.7854
|
||||||
|
* Input x is assumed to be bounded by ~pi/4 in magnitude.
|
||||||
|
* Input y is the tail of x.
|
||||||
|
* Input iy indicates whether y is 0. (if iy=0, y assume to be 0).
|
||||||
|
*
|
||||||
|
* Algorithm
|
||||||
|
* 1. Since sin(-x) = -sin(x), we need only to consider positive x.
|
||||||
|
* 2. Callers must return sin(-0) = -0 without calling here since our
|
||||||
|
* odd polynomial is not evaluated in a way that preserves -0.
|
||||||
|
* Callers may do the optimization sin(x) ~ x for tiny x.
|
||||||
|
* 3. sin(x) is approximated by a polynomial of degree 13 on
|
||||||
|
* [0,pi/4]
|
||||||
|
* 3 13
|
||||||
|
* sin(x) ~ x + S1*x + ... + S6*x
|
||||||
|
* where
|
||||||
|
*
|
||||||
|
* |sin(x) 2 4 6 8 10 12 | -58
|
||||||
|
* |----- - (1+S1*x +S2*x +S3*x +S4*x +S5*x +S6*x )| <= 2
|
||||||
|
* | x |
|
||||||
|
*
|
||||||
|
* 4. sin(x+y) = sin(x) + sin'(x')*y
|
||||||
|
* ~ sin(x) + (1-x*x/2)*y
|
||||||
|
* For better accuracy, let
|
||||||
|
* 3 2 2 2 2
|
||||||
|
* r = x *(S2+x *(S3+x *(S4+x *(S5+x *S6))))
|
||||||
|
* then 3 2
|
||||||
|
* sin(x) = x + (S1*x + (x *(r-y/2)+y))
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define S1 -1.66666666666666324348e-01
|
||||||
|
#define S2 8.33333333332248946124e-03
|
||||||
|
#define S3 -1.98412698298579493134e-04
|
||||||
|
#define S4 2.75573137070700676789e-06
|
||||||
|
#define S5 -2.50507602534068634195e-08
|
||||||
|
#define S6 1.58969099521155010221e-10
|
||||||
|
|
||||||
|
double __sin(double x, double y, int iy)
|
||||||
|
{
|
||||||
|
double_t z,r,v,w;
|
||||||
|
z = x*x;
|
||||||
|
w = z*z;
|
||||||
|
r = S2 + z*(S3 + z*S4) + z*w*(S5 + z*S6);
|
||||||
|
v = z*x;
|
||||||
|
if (iy == 0)
|
||||||
|
return x + v*(S1 + z*r);
|
||||||
|
else
|
||||||
|
return x - ((z*(0.5*y - v*r) - y) - v*S1);
|
||||||
|
}
|
71
libc/tinymath/nextafter.c
Normal file
71
libc/tinymath/nextafter.c
Normal file
|
@ -0,0 +1,71 @@
|
||||||
|
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
||||||
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
||||||
|
╚──────────────────────────────────────────────────────────────────────────────╝
|
||||||
|
│ │
|
||||||
|
│ Musl Libc │
|
||||||
|
│ Copyright © 2005-2014 Rich Felker, et al. │
|
||||||
|
│ │
|
||||||
|
│ Permission is hereby granted, free of charge, to any person obtaining │
|
||||||
|
│ a copy of this software and associated documentation files (the │
|
||||||
|
│ "Software"), to deal in the Software without restriction, including │
|
||||||
|
│ without limitation the rights to use, copy, modify, merge, publish, │
|
||||||
|
│ distribute, sublicense, and/or sell copies of the Software, and to │
|
||||||
|
│ permit persons to whom the Software is furnished to do so, subject to │
|
||||||
|
│ the following conditions: │
|
||||||
|
│ │
|
||||||
|
│ The above copyright notice and this permission notice shall be │
|
||||||
|
│ included in all copies or substantial portions of the Software. │
|
||||||
|
│ │
|
||||||
|
│ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, │
|
||||||
|
│ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF │
|
||||||
|
│ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. │
|
||||||
|
│ IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY │
|
||||||
|
│ CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, │
|
||||||
|
│ TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE │
|
||||||
|
│ SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. │
|
||||||
|
│ │
|
||||||
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
||||||
|
#include "libc/math.h"
|
||||||
|
|
||||||
|
asm(".ident\t\"\\n\\n\
|
||||||
|
Musl libc (MIT License)\\n\
|
||||||
|
Copyright 2005-2014 Rich Felker, et. al.\"");
|
||||||
|
asm(".include \"libc/disclaimer.inc\"");
|
||||||
|
|
||||||
|
/* clang-format off */
|
||||||
|
|
||||||
|
static inline void force_eval(double x)
|
||||||
|
{
|
||||||
|
volatile double y;
|
||||||
|
y = x;
|
||||||
|
}
|
||||||
|
|
||||||
|
double nextafter(double x, double y)
|
||||||
|
{
|
||||||
|
union {double f; uint64_t i;} ux={x}, uy={y};
|
||||||
|
uint64_t ax, ay;
|
||||||
|
int e;
|
||||||
|
|
||||||
|
if (isnan(x) || isnan(y))
|
||||||
|
return x + y;
|
||||||
|
if (ux.i == uy.i)
|
||||||
|
return y;
|
||||||
|
ax = ux.i & -1ULL/2;
|
||||||
|
ay = uy.i & -1ULL/2;
|
||||||
|
if (ax == 0) {
|
||||||
|
if (ay == 0)
|
||||||
|
return y;
|
||||||
|
ux.i = (uy.i & 1ULL<<63) | 1;
|
||||||
|
} else if (ax > ay || ((ux.i ^ uy.i) & 1ULL<<63))
|
||||||
|
ux.i--;
|
||||||
|
else
|
||||||
|
ux.i++;
|
||||||
|
e = ux.i >> 52 & 0x7ff;
|
||||||
|
/* raise overflow if ux.f is infinite and x is finite */
|
||||||
|
if (e == 0x7ff)
|
||||||
|
force_eval(x+x);
|
||||||
|
/* raise underflow if ux.f is subnormal or zero */
|
||||||
|
if (e == 0)
|
||||||
|
force_eval(x*x + ux.f*ux.f);
|
||||||
|
return ux.f;
|
||||||
|
}
|
70
libc/tinymath/nextafterf.c
Normal file
70
libc/tinymath/nextafterf.c
Normal file
|
@ -0,0 +1,70 @@
|
||||||
|
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
||||||
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
||||||
|
╚──────────────────────────────────────────────────────────────────────────────╝
|
||||||
|
│ │
|
||||||
|
│ Musl Libc │
|
||||||
|
│ Copyright © 2005-2014 Rich Felker, et al. │
|
||||||
|
│ │
|
||||||
|
│ Permission is hereby granted, free of charge, to any person obtaining │
|
||||||
|
│ a copy of this software and associated documentation files (the │
|
||||||
|
│ "Software"), to deal in the Software without restriction, including │
|
||||||
|
│ without limitation the rights to use, copy, modify, merge, publish, │
|
||||||
|
│ distribute, sublicense, and/or sell copies of the Software, and to │
|
||||||
|
│ permit persons to whom the Software is furnished to do so, subject to │
|
||||||
|
│ the following conditions: │
|
||||||
|
│ │
|
||||||
|
│ The above copyright notice and this permission notice shall be │
|
||||||
|
│ included in all copies or substantial portions of the Software. │
|
||||||
|
│ │
|
||||||
|
│ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, │
|
||||||
|
│ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF │
|
||||||
|
│ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. │
|
||||||
|
│ IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY │
|
||||||
|
│ CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, │
|
||||||
|
│ TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE │
|
||||||
|
│ SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. │
|
||||||
|
│ │
|
||||||
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
||||||
|
#include "libc/math.h"
|
||||||
|
|
||||||
|
asm(".ident\t\"\\n\\n\
|
||||||
|
Musl libc (MIT License)\\n\
|
||||||
|
Copyright 2005-2014 Rich Felker, et. al.\"");
|
||||||
|
asm(".include \"libc/disclaimer.inc\"");
|
||||||
|
|
||||||
|
/* clang-format off */
|
||||||
|
|
||||||
|
static inline void force_eval(float x)
|
||||||
|
{
|
||||||
|
volatile float y;
|
||||||
|
y = x;
|
||||||
|
}
|
||||||
|
|
||||||
|
float nextafterf(float x, float y)
|
||||||
|
{
|
||||||
|
union {float f; uint32_t i;} ux={x}, uy={y};
|
||||||
|
uint32_t ax, ay, e;
|
||||||
|
|
||||||
|
if (isnan(x) || isnan(y))
|
||||||
|
return x + y;
|
||||||
|
if (ux.i == uy.i)
|
||||||
|
return y;
|
||||||
|
ax = ux.i & 0x7fffffff;
|
||||||
|
ay = uy.i & 0x7fffffff;
|
||||||
|
if (ax == 0) {
|
||||||
|
if (ay == 0)
|
||||||
|
return y;
|
||||||
|
ux.i = (uy.i & 0x80000000) | 1;
|
||||||
|
} else if (ax > ay || ((ux.i ^ uy.i) & 0x80000000))
|
||||||
|
ux.i--;
|
||||||
|
else
|
||||||
|
ux.i++;
|
||||||
|
e = ux.i & 0x7f800000;
|
||||||
|
/* raise overflow if ux.f is infinite and x is finite */
|
||||||
|
if (e == 0x7f800000)
|
||||||
|
force_eval(x+x);
|
||||||
|
/* raise underflow if ux.f is subnormal or zero */
|
||||||
|
if (e == 0)
|
||||||
|
force_eval(x*x + ux.f*ux.f);
|
||||||
|
return ux.f;
|
||||||
|
}
|
85
libc/tinymath/nextafterl.c
Normal file
85
libc/tinymath/nextafterl.c
Normal file
|
@ -0,0 +1,85 @@
|
||||||
|
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
||||||
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
||||||
|
╚──────────────────────────────────────────────────────────────────────────────╝
|
||||||
|
│ │
|
||||||
|
│ Musl Libc │
|
||||||
|
│ Copyright © 2005-2014 Rich Felker, et al. │
|
||||||
|
│ │
|
||||||
|
│ Permission is hereby granted, free of charge, to any person obtaining │
|
||||||
|
│ a copy of this software and associated documentation files (the │
|
||||||
|
│ "Software"), to deal in the Software without restriction, including │
|
||||||
|
│ without limitation the rights to use, copy, modify, merge, publish, │
|
||||||
|
│ distribute, sublicense, and/or sell copies of the Software, and to │
|
||||||
|
│ permit persons to whom the Software is furnished to do so, subject to │
|
||||||
|
│ the following conditions: │
|
||||||
|
│ │
|
||||||
|
│ The above copyright notice and this permission notice shall be │
|
||||||
|
│ included in all copies or substantial portions of the Software. │
|
||||||
|
│ │
|
||||||
|
│ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, │
|
||||||
|
│ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF │
|
||||||
|
│ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. │
|
||||||
|
│ IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY │
|
||||||
|
│ CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, │
|
||||||
|
│ TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE │
|
||||||
|
│ SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. │
|
||||||
|
│ │
|
||||||
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
||||||
|
#include "libc/math.h"
|
||||||
|
|
||||||
|
asm(".ident\t\"\\n\\n\
|
||||||
|
Musl libc (MIT License)\\n\
|
||||||
|
Copyright 2005-2014 Rich Felker, et. al.\"");
|
||||||
|
asm(".include \"libc/disclaimer.inc\"");
|
||||||
|
|
||||||
|
/* clang-format off */
|
||||||
|
|
||||||
|
union ldshape {
|
||||||
|
long double f;
|
||||||
|
struct {
|
||||||
|
uint64_t m;
|
||||||
|
uint16_t se;
|
||||||
|
} i;
|
||||||
|
};
|
||||||
|
|
||||||
|
static inline void force_eval(long double x)
|
||||||
|
{
|
||||||
|
volatile long double y;
|
||||||
|
y = x;
|
||||||
|
}
|
||||||
|
|
||||||
|
long double nextafterl(long double x, long double y)
|
||||||
|
{
|
||||||
|
union ldshape ux, uy;
|
||||||
|
|
||||||
|
if (isnan(x) || isnan(y))
|
||||||
|
return x + y;
|
||||||
|
if (x == y)
|
||||||
|
return y;
|
||||||
|
ux.f = x;
|
||||||
|
if (x == 0) {
|
||||||
|
uy.f = y;
|
||||||
|
ux.i.m = 1;
|
||||||
|
ux.i.se = uy.i.se & 0x8000;
|
||||||
|
} else if ((x < y) == !(ux.i.se & 0x8000)) {
|
||||||
|
ux.i.m++;
|
||||||
|
if (ux.i.m << 1 == 0) {
|
||||||
|
ux.i.m = 1ULL << 63;
|
||||||
|
ux.i.se++;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (ux.i.m << 1 == 0) {
|
||||||
|
ux.i.se--;
|
||||||
|
if (ux.i.se)
|
||||||
|
ux.i.m = 0;
|
||||||
|
}
|
||||||
|
ux.i.m--;
|
||||||
|
}
|
||||||
|
/* raise overflow if ux is infinite and x is finite */
|
||||||
|
if ((ux.i.se & 0x7fff) == 0x7fff)
|
||||||
|
return x + x;
|
||||||
|
/* raise underflow if ux is subnormal or zero */
|
||||||
|
if ((ux.i.se & 0x7fff) == 0)
|
||||||
|
force_eval(x*x + ux.f*ux.f);
|
||||||
|
return ux.f;
|
||||||
|
}
|
20
libc/tinymath/signgam.c
Normal file
20
libc/tinymath/signgam.c
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
||||||
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
||||||
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
||||||
|
│ Copyright 2021 Justine Alexandra Roberts Tunney │
|
||||||
|
│ │
|
||||||
|
│ Permission to use, copy, modify, and/or distribute this software for │
|
||||||
|
│ any purpose with or without fee is hereby granted, provided that the │
|
||||||
|
│ above copyright notice and this permission notice appear in all copies. │
|
||||||
|
│ │
|
||||||
|
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
|
||||||
|
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
|
||||||
|
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
|
||||||
|
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
|
||||||
|
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
|
||||||
|
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
|
||||||
|
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
|
||||||
|
│ PERFORMANCE OF THIS SOFTWARE. │
|
||||||
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
||||||
|
|
||||||
|
int __signgam;
|
38
test/libc/tinymath/erf_test.c
Normal file
38
test/libc/tinymath/erf_test.c
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
||||||
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
||||||
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
||||||
|
│ Copyright 2021 Justine Alexandra Roberts Tunney │
|
||||||
|
│ │
|
||||||
|
│ Permission to use, copy, modify, and/or distribute this software for │
|
||||||
|
│ any purpose with or without fee is hereby granted, provided that the │
|
||||||
|
│ above copyright notice and this permission notice appear in all copies. │
|
||||||
|
│ │
|
||||||
|
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
|
||||||
|
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
|
||||||
|
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
|
||||||
|
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
|
||||||
|
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
|
||||||
|
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
|
||||||
|
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
|
||||||
|
│ PERFORMANCE OF THIS SOFTWARE. │
|
||||||
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
||||||
|
#include "libc/math.h"
|
||||||
|
#include "libc/runtime/gc.h"
|
||||||
|
#include "libc/testlib/testlib.h"
|
||||||
|
#include "libc/x/x.h"
|
||||||
|
|
||||||
|
TEST(erf, test) {
|
||||||
|
EXPECT_STREQ("NAN", gc(xdtoa(erf(NAN))));
|
||||||
|
EXPECT_STREQ("0", gc(xdtoa(erf(0))));
|
||||||
|
EXPECT_STREQ("1", gc(xdtoa(erf(INFINITY))));
|
||||||
|
EXPECT_STREQ(".999977909503001", gc(xdtoa(erf(3))));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(erfc, test) {
|
||||||
|
EXPECT_STREQ("NAN", gc(xdtoa(erfc(NAN))));
|
||||||
|
EXPECT_STREQ("1", gc(xdtoa(erfc(0))));
|
||||||
|
EXPECT_STREQ("1", gc(xdtoa(erfc(-0.))));
|
||||||
|
EXPECT_STREQ("0", gc(xdtoa(erfc(INFINITY))));
|
||||||
|
EXPECT_STREQ("2", gc(xdtoa(erfc(-INFINITY))));
|
||||||
|
EXPECT_STREQ("2.20904969985854e-05", gc(xdtoa(erfc(3))));
|
||||||
|
}
|
33
test/libc/tinymath/gamma_test.c
Normal file
33
test/libc/tinymath/gamma_test.c
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
||||||
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
||||||
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
||||||
|
│ Copyright 2021 Justine Alexandra Roberts Tunney │
|
||||||
|
│ │
|
||||||
|
│ Permission to use, copy, modify, and/or distribute this software for │
|
||||||
|
│ any purpose with or without fee is hereby granted, provided that the │
|
||||||
|
│ above copyright notice and this permission notice appear in all copies. │
|
||||||
|
│ │
|
||||||
|
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
|
||||||
|
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
|
||||||
|
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
|
||||||
|
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
|
||||||
|
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
|
||||||
|
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
|
||||||
|
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
|
||||||
|
│ PERFORMANCE OF THIS SOFTWARE. │
|
||||||
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
||||||
|
#include "libc/math.h"
|
||||||
|
#include "libc/runtime/gc.h"
|
||||||
|
#include "libc/testlib/testlib.h"
|
||||||
|
#include "libc/x/x.h"
|
||||||
|
|
||||||
|
TEST(lgamma, test) {
|
||||||
|
EXPECT_STREQ("NAN", gc(xdtoa(lgamma(NAN))));
|
||||||
|
EXPECT_STREQ("0", gc(xdtoa(lgamma(1))));
|
||||||
|
EXPECT_STREQ("0", gc(xdtoa(lgamma(2))));
|
||||||
|
EXPECT_STREQ("INFINITY", gc(xdtoa(lgamma(INFINITY))));
|
||||||
|
EXPECT_STREQ("INFINITY", gc(xdtoa(lgamma(-INFINITY))));
|
||||||
|
EXPECT_STREQ("INFINITY", gc(xdtoa(lgamma(-1))));
|
||||||
|
EXPECT_STREQ("INFINITY", gc(xdtoa(lgamma(-2))));
|
||||||
|
EXPECT_STREQ("1.26551212348465", gc(xdtoa(lgamma(-.5))));
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue