BlosSOM
Interactive dimensionality reduction on large datasets (EmbedSOM and FLOWER combined)
pnorm.cpp
Go to the documentation of this file.
1
2#include "pnorm.h"
3
4#include <cfloat>
5#include <cmath>
6
7/** The actual implementation of pnormf().
8 *
9 * pnorm_both() is taken and adapted from R-4.0.2, originally licensed under
10 * GPLv2, which is compatible with BlosSOM. We whole-heartedly thank the R
11 * project for maintaining a quality library of mathematical function
12 * implementations. Original licence and copyright follows:
13 *
14 * Mathlib : A C Library of Special Functions
15 * Copyright (C) 1998 Ross Ihaka
16 * Copyright (C) 2000-2013 The R Core Team
17 * Copyright (C) 2003 The R Foundation
18 *
19 * This program is free software; you can redistribute it and/or modify
20 * it under the terms of the GNU General Public License as published by
21 * the Free Software Foundation; either version 2 of the License, or
22 * (at your option) any later version.
23 *
24 * This program is distributed in the hope that it will be useful,
25 * but WITHOUT ANY WARRANTY; without even the implied warranty of
26 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
27 * GNU General Public License for more details.
28 *
29 * You should have received a copy of the GNU General Public License
30 * along with this program; if not, a copy is available at
31 * https://www.R-project.org/Licenses/
32 */
33
34static void
35pnorm_both(double x, double &cum, double &ccum)
36{
37#define SIXTEN 16
38#define M_1_SQRT_2PI 0.398942280401432677939946059934
39#define M_SQRT_32 5.656854249492380195206754896838
40
41 const static double a[5] = { 2.2352520354606839287,
42 161.02823106855587881,
43 1067.6894854603709582,
44 18154.981253343561249,
45 0.065682337918207449113 };
46 const static double b[4] = { 47.20258190468824187,
47 976.09855173777669322,
48 10260.932208618978205,
49 45507.789335026729956 };
50 const static double c[9] = { 0.39894151208813466764, 8.8831497943883759412,
51 93.506656132177855979, 597.27027639480026226,
52 2494.5375852903726711, 6848.1904505362823326,
53 11602.651437647350124, 9842.7148383839780218,
54 1.0765576773720192317e-8 };
55 const static double d[8] = { 22.266688044328115691, 235.38790178262499861,
56 1519.377599407554805, 6485.558298266760755,
57 18615.571640885098091, 34900.952721145977266,
58 38912.003286093271411, 19685.429676859990727 };
59 const static double p[6] = {
60 0.21589853405795699, 0.1274011611602473639, 0.022235277870649807,
61 0.001421619193227893466, 2.9112874951168792e-5, 0.02307344176494017303
62 };
63 const static double q[5] = { 1.28426009614491121,
64 0.468238212480865118,
65 0.0659881378689285515,
66 0.00378239633202758244,
67 7.29751555083966205e-5 };
68
69 double xden, xnum, temp, del, eps, xsq, y;
70 int i;
71
72 eps = DBL_EPSILON * 0.5;
73
74 constexpr int lower = 1;
75 constexpr int upper = 0;
76
77 y = fabs(x);
78 if (y <= 0.67448975) {
79 if (y > eps) {
80 xsq = x * x;
81 xnum = a[4] * xsq;
82 xden = xsq;
83 for (i = 0; i < 3; ++i) {
84 xnum = (xnum + a[i]) * xsq;
85 xden = (xden + b[i]) * xsq;
86 }
87 } else
88 xnum = xden = 0.0;
89
90 temp = x * (xnum + a[3]) / (xden + b[3]);
91 if (lower)
92 cum = 0.5 + temp;
93 if (upper)
94 ccum = 0.5 - temp;
95 } else if (y <= M_SQRT_32) {
96 xnum = c[8] * y;
97 xden = y;
98 for (i = 0; i < 7; ++i) {
99 xnum = (xnum + c[i]) * y;
100 xden = (xden + d[i]) * y;
101 }
102 temp = (xnum + c[7]) / (xden + d[7]);
103
104#define do_del(X) \
105 xsq = trunc(X * SIXTEN) / SIXTEN; \
106 del = (X - xsq) * (X + xsq); \
107 cum = exp(-xsq * xsq * 0.5) * exp(-del * 0.5) * temp; \
108 ccum = 1.0 - cum;
109
110#define swap_tail \
111 if (x > 0.) { \
112 temp = cum; \
113 if (lower) \
114 cum = ccum; \
115 ccum = temp; \
116 }
117
118 do_del(y);
119 swap_tail;
120 }
121
122 else if ((lower && -37.5193 < x && x < 8.2924) ||
123 (upper && -8.2924 < x && x < 37.5193)) {
124
125 xsq = 1.0 / (x * x);
126 xnum = p[5] * xsq;
127 xden = xsq;
128 for (i = 0; i < 4; ++i) {
129 xnum = (xnum + p[i]) * xsq;
130 xden = (xden + q[i]) * xsq;
131 }
132 temp = xsq * (xnum + p[4]) / (xden + q[4]);
133 temp = (M_1_SQRT_2PI - temp) / y;
134
135 do_del(x);
136 swap_tail;
137 } else {
138 if (x > 0) {
139 cum = 1;
140 ccum = 0;
141 } else {
142 cum = 0;
143 ccum = 1;
144 }
145 }
146
147 if (cum < DBL_MIN)
148 cum = 0.;
149 if (ccum < DBL_MIN)
150 ccum = 0.;
151 return;
152}
153
154float
155pnormf(float x, float mean, float sd)
156{
157 double p = 0, cp = 0;
158
159 if (!std::isfinite(x) && mean == x)
160 return NAN;
161 if (sd <= 0) {
162 if (sd < 0)
163 return 0.5;
164 return (x < mean) ? 0 : 1;
165 }
166 p = (x - mean) / sd;
167 if (!std::isfinite(p))
168 return (x < mean) ? 0 : 1;
169 x = p;
170
171 pnorm_both(x, p, cp);
172
173 return p;
174}
#define swap_tail
#define M_1_SQRT_2PI
#define do_del(X)
#define M_SQRT_32
float pnormf(float x, float mean, float sd)
Definition: pnorm.cpp:155
static void pnorm_both(double x, double &cum, double &ccum)
The actual implementation of pnormf().
Definition: pnorm.cpp:35