Skip to content

Commit

Permalink
bug fixes in matrix log (#32327)
Browse files Browse the repository at this point in the history
* bug fixes in matrix log

* patches to matrix log (#33245)

* patches to matrix log

Avoid integer overflow if `s > 63`.
Correct logic for `s == 0`.
Only use fancy divided difference formulae if eigenvalues are close - avoids dangerous roundoff error if they are in opposite sectors.

* add tests

(cherry picked from commit 318affa)
  • Loading branch information
stevengj authored and KristofferC committed Feb 22, 2020
1 parent f7279ba commit db0bc60
Show file tree
Hide file tree
Showing 8 changed files with 1,871 additions and 29 deletions.
50 changes: 21 additions & 29 deletions stdlib/LinearAlgebra/src/triangular.jl
Original file line number Diff line number Diff line change
Expand Up @@ -2138,32 +2138,14 @@ function log(A0::UpperTriangular{T}) where T<:BlasFloat
end

# Compute accurate superdiagonal of T
p = 1 / 2^s
for k = 1:n-1
Ak = A0[k,k]
Akp1 = A0[k+1,k+1]
Akp = Ak^p
Akp1p = Akp1^p
A[k,k] = Akp
A[k+1,k+1] = Akp1p
if Ak == Akp1
A[k,k+1] = p * A0[k,k+1] * Ak^(p-1)
elseif 2 * abs(Ak) < abs(Akp1) || 2 * abs(Akp1) < abs(Ak)
A[k,k+1] = A0[k,k+1] * (Akp1p - Akp) / (Akp1 - Ak)
else
logAk = log(Ak)
logAkp1 = log(Akp1)
w = atanh((Akp1 - Ak)/(Akp1 + Ak)) + im*pi*ceil((imag(logAkp1-logAk)-pi)/(2*pi))
dd = 2 * exp(p*(logAk+logAkp1)/2) * sinh(p*w) / (Akp1 - Ak)
A[k,k+1] = A0[k,k+1] * dd
end
end
blockpower!(A, A0, 0.5^s)

# Compute accurate diagonal of T
for i = 1:n
a = A0[i,i]
if s == 0
r = a - 1
A[i,i] = a - 1
continue
end
s0 = s
if angle(a) >= pi / 2
Expand Down Expand Up @@ -2200,7 +2182,7 @@ function log(A0::UpperTriangular{T}) where T<:BlasFloat
end

# Scale back
lmul!(2^s, Y)
lmul!(2.0^s, Y)

# Compute accurate diagonal and superdiagonal of log(T)
for k = 1:n-1
Expand All @@ -2212,11 +2194,16 @@ function log(A0::UpperTriangular{T}) where T<:BlasFloat
Y[k+1,k+1] = logAkp1
if Ak == Akp1
Y[k,k+1] = A0[k,k+1] / Ak
elseif 2 * abs(Ak) < abs(Akp1) || 2 * abs(Akp1) < abs(Ak)
elseif 2 * abs(Ak) < abs(Akp1) || 2 * abs(Akp1) < abs(Ak) || iszero(Akp1 + Ak)
Y[k,k+1] = A0[k,k+1] * (logAkp1 - logAk) / (Akp1 - Ak)
else
w = atanh((Akp1 - Ak)/(Akp1 + Ak) + im*pi*(ceil((imag(logAkp1-logAk) - pi)/(2*pi))))
Y[k,k+1] = 2 * A0[k,k+1] * w / (Akp1 - Ak)
z = (Akp1 - Ak)/(Akp1 + Ak)
if abs(z) > 1
Y[k,k+1] = A0[k,k+1] * (logAkp1 - logAk) / (Akp1 - Ak)
else
w = atanh(z) + im * pi * (unw(logAkp1-logAk) - unw(log1p(z)-log1p(-z)))
Y[k,k+1] = 2 * A0[k,k+1] * w / (Akp1 - Ak)
end
end
end

Expand Down Expand Up @@ -2363,14 +2350,19 @@ function blockpower!(A::UpperTriangular, A0::UpperTriangular, p)

if Ak == Akp1
A[k,k+1] = p * A0[k,k+1] * Ak^(p-1)
elseif 2 * abs(Ak) < abs(Akp1) || 2 * abs(Akp1) < abs(Ak)
elseif 2 * abs(Ak) < abs(Akp1) || 2 * abs(Akp1) < abs(Ak) || iszero(Akp1 + Ak)
A[k,k+1] = A0[k,k+1] * (Akp1p - Akp) / (Akp1 - Ak)
else
logAk = log(Ak)
logAkp1 = log(Akp1)
w = atanh((Akp1 - Ak)/(Akp1 + Ak)) + im * pi * unw(logAkp1-logAk)
dd = 2 * exp(p*(logAk+logAkp1)/2) * sinh(p*w) / (Akp1 - Ak);
A[k,k+1] = A0[k,k+1] * dd
z = (Akp1 - Ak)/(Akp1 + Ak)
if abs(z) > 1
A[k,k+1] = A0[k,k+1] * (Akp1p - Akp) / (Akp1 - Ak)
else
w = atanh(z) + im * pi * (unw(logAkp1-logAk) - unw(log1p(z)-log1p(-z)))
dd = 2 * exp(p*(logAk+logAkp1)/2) * sinh(p*w) / (Akp1 - Ak);
A[k,k+1] = A0[k,k+1] * dd
end
end
end
end
Expand Down
18 changes: 18 additions & 0 deletions stdlib/LinearAlgebra/test/dense.jl
Original file line number Diff line number Diff line change
Expand Up @@ -877,4 +877,22 @@ end
@test_broken inv(transpose(B))*transpose(B) I
end

@testset "Matrix log issue #32313" begin
for A in ([30 20; -50 -30], [10.0im 0; 0 -10.0im], randn(6,6))
@test exp(log(A)) A
end
end

@testset "Matrix log PR #33245" begin
###########
# Note: Test removed in backporting to 1.0 due to diagm(::Array{Complex{Float64},1}) not existing
###########
# edge case for divided difference
#A1 = triu(ones(3,3),1) + diagm([1.0, -2eps()-1im, -eps()+0.75im])
#@test exp(log(A1)) ≈ A1
# case where no sqrt is needed (s=0)
A2 = [1.01 0.01 0.01; 0 1.01 0.01; 0 0 1.01]
@test exp(log(A2)) A2
end

end # module TestDense
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
name = "Statistics"
uuid = "10745b16-79ce-11e8-11f9-7d13ad32a3b2"

[deps]
LinearAlgebra = "37e2e46d-f89d-539d-b4ee-838fcccc9c8e"
SparseArrays = "2f01184e-e22b-5df5-ae63-d93ebab69eaf"

[extras]
Random = "9a3f8284-a2c9-5f02-9a11-845980a1fd5c"
Test = "8dfed614-e22c-5e08-85e1-65c5234f0b40"

[targets]
test = ["Random", "Test"]
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
1
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
# Statistics

The Statistics module contains basic statistics functionality.

!!! note
To use any of the examples described below, run `using Statistics` and then the code from the example.

```@docs
Statistics.std
Statistics.stdm
Statistics.var
Statistics.varm
Statistics.cor
Statistics.cov
Statistics.mean!
Statistics.mean
Statistics.median!
Statistics.median
Statistics.middle
Statistics.quantile!
Statistics.quantile
```
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
1
Loading

0 comments on commit db0bc60

Please sign in to comment.