#if defined(_MSC_VER)
#define NOMINMAX
#pragma float_control(except, off)
#pragma float_control(precise, off)
#pragma fp_contract(on)
#pragma fenv_access(off)
#endif
#include "normaldistribution.h"
#include <float.h>
namespace {
const double norm_cdf_asymptotic_expansion_first_threshold = -10.0;
const double norm_cdf_asymptotic_expansion_second_threshold = -1 / sqrt(DBL_EPSILON);
}
double norm_cdf(double z) {
if (z <= norm_cdf_asymptotic_expansion_first_threshold) {
double sum = 1;
if (z >= norm_cdf_asymptotic_expansion_second_threshold) {
double zsqr = z * z, i = 1, g = 1, x, y, a = DBL_MAX, lasta;
do {
lasta = a;
x = (4 * i - 3) / zsqr;
y = x * ((4 * i - 1) / zsqr);
a = g * (x - y);
sum -= a;
g *= y;
++i;
a = fabs(a);
} while (lasta > a && a >= fabs(sum * DBL_EPSILON));
}
return -norm_pdf(z) * sum / z;
}
return 0.5 * erfc_cody(-z * ONE_OVER_SQRT_TWO);
}
double inverse_norm_cdf(double u) {
const double split1 = 0.425;
const double split2 = 5.0;
const double const1 = 0.180625;
const double const2 = 1.6;
const double A0 = 3.3871328727963666080E0;
const double A1 = 1.3314166789178437745E+2;
const double A2 = 1.9715909503065514427E+3;
const double A3 = 1.3731693765509461125E+4;
const double A4 = 4.5921953931549871457E+4;
const double A5 = 6.7265770927008700853E+4;
const double A6 = 3.3430575583588128105E+4;
const double A7 = 2.5090809287301226727E+3;
const double B1 = 4.2313330701600911252E+1;
const double B2 = 6.8718700749205790830E+2;
const double B3 = 5.3941960214247511077E+3;
const double B4 = 2.1213794301586595867E+4;
const double B5 = 3.9307895800092710610E+4;
const double B6 = 2.8729085735721942674E+4;
const double B7 = 5.2264952788528545610E+3;
const double C0 = 1.42343711074968357734E0;
const double C1 = 4.63033784615654529590E0;
const double C2 = 5.76949722146069140550E0;
const double C3 = 3.64784832476320460504E0;
const double C4 = 1.27045825245236838258E0;
const double C5 = 2.41780725177450611770E-1;
const double C6 = 2.27238449892691845833E-2;
const double C7 = 7.74545014278341407640E-4;
const double D1 = 2.05319162663775882187E0;
const double D2 = 1.67638483018380384940E0;
const double D3 = 6.89767334985100004550E-1;
const double D4 = 1.48103976427480074590E-1;
const double D5 = 1.51986665636164571966E-2;
const double D6 = 5.47593808499534494600E-4;
const double D7 = 1.05075007164441684324E-9;
const double E0 = 6.65790464350110377720E0;
const double E1 = 5.46378491116411436990E0;
const double E2 = 1.78482653991729133580E0;
const double E3 = 2.96560571828504891230E-1;
const double E4 = 2.65321895265761230930E-2;
const double E5 = 1.24266094738807843860E-3;
const double E6 = 2.71155556874348757815E-5;
const double E7 = 2.01033439929228813265E-7;
const double F1 = 5.99832206555887937690E-1;
const double F2 = 1.36929880922735805310E-1;
const double F3 = 1.48753612908506148525E-2;
const double F4 = 7.86869131145613259100E-4;
const double F5 = 1.84631831751005468180E-5;
const double F6 = 1.42151175831644588870E-7;
const double F7 = 2.04426310338993978564E-15;
if (u <= 0)
return log(u);
if (u >= 1)
return log(1 - u);
const double q = u - 0.5;
if (fabs(q) <= split1) {
const double r = const1 - q * q;
return q * (((((((A7 * r + A6) * r + A5) * r + A4) * r + A3) * r + A2) * r + A1) * r + A0) /
(((((((B7 * r + B6) * r + B5) * r + B4) * r + B3) * r + B2) * r + B1) * r + 1.0);
} else {
double r = q < 0.0 ? u : 1.0 - u;
r = sqrt(-log(r));
double ret;
if (r < split2) {
r = r - const2;
ret = (((((((C7 * r + C6) * r + C5) * r + C4) * r + C3) * r + C2) * r + C1) * r + C0) /
(((((((D7 * r + D6) * r + D5) * r + D4) * r + D3) * r + D2) * r + D1) * r + 1.0);
} else {
r = r - split2;
ret = (((((((E7 * r + E6) * r + E5) * r + E4) * r + E3) * r + E2) * r + E1) * r + E0) /
(((((((F7 * r + F6) * r + F5) * r + F4) * r + F3) * r + F2) * r + F1) * r + 1.0);
}
return q < 0.0 ? -ret : ret;
}
}