Skip to content

Commit 5d9b8a8

Browse files
Backport fix in #399 to 1.8 and release it as 1.8.5 (#400)
* Fix hangs in `beta_inc_inv` (alternative/extension of #396) (#399) * Limit number of Newton iterations in beta_inc_inv and allow much smaller starting values. * Adjust tolerance instead of number of iterations Co-authored-by: Andreas Noack <[email protected]> * Release 1.8.5 Co-authored-by: Andreas Noack <[email protected]>
1 parent 39e9d65 commit 5d9b8a8

File tree

3 files changed

+14
-20
lines changed

3 files changed

+14
-20
lines changed

Project.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
name = "SpecialFunctions"
22
uuid = "276daf66-3868-5448-9aa4-cd146d93841b"
3-
version = "1.8.4"
3+
version = "1.8.5"
44

55
[deps]
66
ChainRulesCore = "d360d2e6-b24c-11e9-a2a3-2a2ae2dbcce4"

src/beta_inc.jl

+7-19
Original file line numberDiff line numberDiff line change
@@ -934,7 +934,6 @@ function _beta_inc_inv(a::Float64, b::Float64, p::Float64, q::Float64=1-p)
934934
r = sqrt(-2*log(p))
935935
p_approx = r - @horner(r, 2.30753e+00, 0.27061e+00) / @horner(r, 1.0, .99229e+00, .04481e+00)
936936
fpu = floatmin(Float64)
937-
sae = log10(fpu)
938937
lb = logbeta(a, b)
939938

940939
if a > 1.0 && b > 1.0
@@ -968,12 +967,7 @@ function _beta_inc_inv(a::Float64, b::Float64, p::Float64, q::Float64=1-p)
968967
sq = 1.0
969968
prev = 1.0
970969

971-
if x < 0.0001
972-
x = 0.0001
973-
end
974-
if x > .9999
975-
x = .9999
976-
end
970+
x = clamp(x, fpu, prevfloat(1.0))
977971

978972
# This first argument was proposed in
979973
#
@@ -997,8 +991,8 @@ function _beta_inc_inv(a::Float64, b::Float64, p::Float64, q::Float64=1-p)
997991
# The idea with the -5.0/a^2 - 1.0/p^0.2 - 34.0 correction is to
998992
# use -2r - 2 (for 16 digits) for large values of a and p but use
999993
# a much smaller tolerance for small values of a and p.
1000-
iex = max(-5.0/a^2 - 1.0/p^0.2 - 34.0, sae)
1001-
acu = exp10(iex)
994+
iex = -5.0/a^2 - 1.0/p^0.2 - 34.0
995+
acu = max(exp10(iex), 10 * fpu) # 10 * fpu instead of fpu avoids hangs for small values
1002996

1003997
#iterate
1004998
while true
@@ -1012,21 +1006,15 @@ function _beta_inc_inv(a::Float64, b::Float64, p::Float64, q::Float64=1-p)
10121006
if p_approx * p_approx_prev <= 0.0
10131007
prev = max(sq, fpu)
10141008
end
1015-
g = 1.0
10161009

1017-
tx = x - g*p_approx
1018-
while true
1019-
adj = g*p_approx
1020-
sq = adj^2
1010+
adj = p_approx
1011+
tx = x - adj
1012+
while prev <= (sq = adj^2) || tx < 0.0 || tx > 1.0
1013+
adj /= 3.0
10211014
tx = x - adj
1022-
if (prev > sq && tx >= 0.0 && tx <= 1.0)
1023-
break
1024-
end
1025-
g /= 3.0
10261015
end
10271016

10281017
#check if current estimate is acceptable
1029-
10301018
if prev <= acu || p_approx^2 <= acu
10311019
x = tx
10321020
return (x, 1.0 - x)

test/beta_inc.jl

+6
Original file line numberDiff line numberDiff line change
@@ -299,4 +299,10 @@ end
299299
@test first(beta_inc_inv(1.0, 1.0, 1e-21)) 1e-21
300300
@test beta_inc_inv(1.0e30, 1.0, 0.49) == (1.0, 0.0)
301301
end
302+
303+
@testset "Avoid infinite loops" begin
304+
# See https://github.com/JuliaStats/StatsFuns.jl/issues/133#issuecomment-1069602721
305+
y = 2.0e-280
306+
@test beta_inc(2.0, 1.0, beta_inc_inv(2.0, 1.0, y, 1.0)[1])[1] y
307+
end
302308
end

0 commit comments

Comments
 (0)