Skip to content
Open
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
40 commits
Select commit Hold shift + click to select a range
581b9df
sumlog
cscherrer May 2, 2022
5725aa9
Update src/sumlog.jl
cscherrer May 2, 2022
77aa3d9
Update src/sumlog.jl
cscherrer May 2, 2022
88d6fb1
Update src/sumlog.jl
cscherrer May 2, 2022
9ecf589
Update src/sumlog.jl
cscherrer May 2, 2022
9db732f
Update src/sumlog.jl
cscherrer May 2, 2022
76533e1
fall-back method
cscherrer May 2, 2022
5747205
more tests
cscherrer May 2, 2022
0f5a927
bump version
cscherrer May 2, 2022
977723d
cast to floating point when possible
cscherrer May 2, 2022
4d488cd
docstring fixes
cscherrer May 2, 2022
afa5d94
performance fix
cscherrer May 2, 2022
e400483
inline _sumlog
cscherrer May 2, 2022
cc1aaac
qualify IrrationalConstants.logtwo
cscherrer May 3, 2022
07809b7
update comment
cscherrer May 3, 2022
16ee153
Update src/sumlog.jl
cscherrer May 3, 2022
1af518b
bugfix
cscherrer May 3, 2022
0eaf8d2
Make it work (and be fast) for Tuples and NamedTuples
cscherrer May 3, 2022
1f478d0
add sumlog to docs
cscherrer May 3, 2022
0807f7a
tests
cscherrer May 3, 2022
2a0004d
comment that `eltype` of a `Base.Generator` returns `Any`
cscherrer May 3, 2022
e5809d1
saturday
mcabbott May 7, 2022
eb1b524
Merge pull request #1 from mcabbott/iterate
cscherrer May 7, 2022
6fe8bb1
Update src/sumlog.jl
cscherrer May 7, 2022
0def97d
change to `logprod`
cscherrer May 10, 2022
3ad95b2
fix sign bit
cscherrer May 10, 2022
a0a9348
Update docs/src/index.md
cscherrer May 10, 2022
fa667ec
Update src/LogExpFunctions.jl
cscherrer May 10, 2022
989a111
Update src/logprod.jl
cscherrer May 10, 2022
a54a024
Update src/LogExpFunctions.jl
cscherrer May 10, 2022
dc48433
Update src/logprod.jl
cscherrer May 10, 2022
39ca989
cleaning up
cscherrer May 10, 2022
207fce2
Merge branch 'master' of https://github.com/cscherrer/LogExpFunctions.jl
cscherrer May 10, 2022
55d125e
Update src/logprod.jl
cscherrer May 10, 2022
bef4728
Update src/logprod.jl
cscherrer May 10, 2022
9572e48
Update src/logprod.jl
cscherrer May 10, 2022
3848848
Update src/logprod.jl
cscherrer May 10, 2022
c4c3e89
Update src/logprod.jl
cscherrer May 10, 2022
e0f410e
Update src/logprod.jl
cscherrer May 10, 2022
23b5bf1
Update src/logprod.jl
cscherrer May 11, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion src/LogExpFunctions.jl
Original file line number Diff line number Diff line change
Expand Up @@ -11,12 +11,13 @@ import LinearAlgebra

export xlogx, xlogy, xlog1py, xexpx, xexpy, logistic, logit, log1psq, log1pexp, log1mexp, log2mexp, logexpm1,
softplus, invsoftplus, log1pmx, logmxp1, logaddexp, logsubexp, logsumexp, logsumexp!, softmax,
softmax!, logcosh
softmax!, logcosh, sumlog

include("basicfuns.jl")
include("logsumexp.jl")
include("chainrules.jl")
include("inverse.jl")
include("with_logabsdet_jacobian.jl")
include("sumlog.jl")

end # module
36 changes: 36 additions & 0 deletions src/sumlog.jl
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
using IrrationalConstants: logtwo

"""
$(SIGNATURES)
Compute `sum(log.(X))`.
`sum(log.(X))` can be evaluated much more quickly as `sum(log, X)`. However,
this still requires computing `log` for each element of `X`.
`sumlog(X)` can be faster still, especially an the length of `X` increases.
This works by representing the `j`th element of `X` as `xⱼ = aⱼ * 2 ^ bⱼ`,
allowing us to write
∑ⱼ log(xⱼ) = log(∏ⱼ aⱼ) + log(2) * ∑ⱼ bⱼ
Since `log(2)` is constant, `sumlog` only requires a single `log` evaluation.
"""
function sumlog(x::AbstractArray{T}) where {T}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It seems this will fail for non-floating point types T such as Int, etc., BigFloat, and for complex numbers?

sig = one(T)
ex = zero(exponent(one(T)))
bound = floatmax(T) / 2
for xj in x
sig *= significand(xj)
ex += exponent(xj)

# Significands are in the rang [1,2), so multiplication will eventually overflow
if sig > bound
(a, b) = (significand(sig), exponent(sig))
sig = a
ex += b
end
end
log(sig) + logtwo * ex
end
1 change: 1 addition & 0 deletions test/runtests.jl
Original file line number Diff line number Diff line change
Expand Up @@ -14,3 +14,4 @@ include("basicfuns.jl")
include("chainrules.jl")
include("inverse.jl")
include("with_logabsdet_jacobian.jl")
include("sumlog.jl")
5 changes: 5 additions & 0 deletions test/sumlog.jl
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
@testset "sumlog" begin
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Some surprises:

julia> sumlog([1,2,-0.1,-0.2])
-3.2188758248682

julia> sumlog([1,2,NaN])
ERROR: DomainError with NaN:
Cannot be NaN or Inf.
Stacktrace:
  [1] (::Base.Math.var"#throw1#5")(x::Float64)
    @ Base.Math ./math.jl:845
  [2] exponent
    @ ./math.jl:848 [inlined]

julia> sumlog([-0.0])
ERROR: DomainError with -0.0:
Cannot be ±0.0.

julia> sum(log, -0.0)
-Inf

Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Calling Base.Math._exponent_finite_nonzero works around the NaN problem (although not for BigFloats).

Adding xj < 0 && Base.Math.throw_complex_domainerror(:log, xj) doesn't seem to cost much speed.

Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also, note that tests right now only test Float64

for x in [10 .* rand(1000), repeat([nextfloat(1.0)], 1000), repeat([prevfloat(2.0)], 1000)]
@test (@inferred sumlog(x)) sum(log, x)
end
end