-
Notifications
You must be signed in to change notification settings - Fork 101
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Jacobi elliptic functions and complete elliptic integral of the first kind #79
base: master
Are you sure you want to change the base?
Changes from 10 commits
396fc72
863bd9b
68fff8a
36f2262
8f22ff7
3825f65
2660d16
de7d83b
6f0d23d
2d406dc
9f16249
dbc4bd4
3b34edd
47d4a55
0eee6b2
f4f00b9
613fd93
37e5999
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,234 @@ | ||
# References: | ||
# [1] Abramowitz, Stegun: Handbook of Mathematical Functions (1965) | ||
ettersi marked this conversation as resolved.
Show resolved
Hide resolved
|
||
# [2] ellipjc.m in Toby Driscoll's Schwarz-Christoffel Toolbox | ||
|
||
|
||
#------------------------------------------------ | ||
# Descending and Ascending Landen Transformation | ||
|
||
descstep(m) = m/(1+sqrt(1-m))^2 | ||
|
||
@generated function shrinkm(m,::Val{N}) where {N} | ||
# [1, Sec 16.12] | ||
quote | ||
f = one(m) | ||
Base.Cartesian.@nexprs $N i->begin | ||
k_i = descstep(m) | ||
m = k_i^2 | ||
f *= 1+k_i | ||
end | ||
return (Base.Cartesian.@ntuple $N k), f, m | ||
end | ||
end | ||
|
||
function ellipj_smallm(u,m) | ||
# [1, Sec 16.13] | ||
sn = sin(u) - m*(u-sin(u)*cos(u))*cos(u)/4 | ||
ettersi marked this conversation as resolved.
Show resolved
Hide resolved
|
||
cn = cos(u) + m*(u-sin(u)*cos(u))*sin(u)/4 | ||
dn = 1 - m*sin(u)^2/2; | ||
return sn,cn,dn | ||
end | ||
function ellipj_largem(u,m1) | ||
# [1, Sec 16.15] | ||
sn = tanh(u) + m1*(sinh(u)*cosh(u)-u)*sech(u)^2/4 | ||
cn = sech(u) - m1*(sinh(u)*cosh(u)-u)*tanh(u)*sech(u)/4 | ||
dn = sech(u) + m1*(sinh(u)*cosh(u)+u)*tanh(u)*sech(u)/4 | ||
return sn,cn,dn | ||
end | ||
|
||
@generated function ellipj_growm(sn,cn,dn, k::NTuple{N,<:Any}) where {N} | ||
# [1, Sec 16.12] | ||
quote | ||
Base.Cartesian.@nexprs $N i->begin | ||
kk = k[end-i+1] | ||
sn,cn,dn = (1+kk)*sn/(1+kk*sn^2), | ||
cn*dn/(1+kk*sn^2), | ||
(1-kk*sn^2)/(1+kk*sn^2) | ||
# ^ Use [1, 16.9.1]. Idea taken from [2] | ||
end | ||
return sn,cn,dn | ||
end | ||
end | ||
@generated function ellipj_shrinkm(sn,cn,dn, k::NTuple{N,<:Any}) where {N} | ||
# [1, Sec 16.14] | ||
quote | ||
Base.Cartesian.@nexprs $N i->begin | ||
kk = k[end-i+1] | ||
sn,cn,dn = (1+kk)*sn*cn/dn, | ||
(cn^2-kk*sn^2)/dn, # Use [1, 16.9.1] | ||
(cn^2+kk*sn^2)/dn # Use [1, 16.9.1] | ||
end | ||
return sn,cn,dn | ||
end | ||
end | ||
|
||
function ellipj_viasmallm(u,m,::Val{N}) where {N} | ||
k,f,m = shrinkm(m,Val{N}()) | ||
sn,cn,dn = ellipj_smallm(u/f,m) | ||
return ellipj_growm(sn,cn,dn,k) | ||
end | ||
function ellipj_vialargem(u,m,::Val{N}) where {N} | ||
k,f,m1 = shrinkm(1-m,Val{N}()) | ||
sn,cn,dn = ellipj_largem(u/f,m1) | ||
return ellipj_shrinkm(sn,cn,dn,k) | ||
end | ||
|
||
|
||
#---------------- | ||
# Pick algorithm | ||
|
||
Base.@pure puresqrt(x) = sqrt(x) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Isn't There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Actually, it isn't for There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It also isn't for the
But I agree, I should have done
With this, I get
Just out of curiosity, is there any fundamental obstacle to do this in Base, and possible for all numerical functions? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Actually, maybe it would be better to just do
Then |
||
Base.@pure function nsteps(m,ε) | ||
i = 0 | ||
while abs(m) > ε | ||
m = descstep(m)^2 | ||
i += 1 | ||
end | ||
return i | ||
end | ||
Base.@pure nsteps(ε,::Type{<:Real}) = nsteps(0.5,ε) # Guarantees convergence in [-1,0.5] | ||
Base.@pure nsteps(ε,::Type{<:Complex}) = nsteps(0.5+sqrt(3)/2im,ε) # This is heuristic. | ||
Comment on lines
+101
to
+110
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I still don't think the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. These functions aren’t pure anyways... There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Right, this needs to be looked into. |
||
function ellipj_dispatch(u,m) | ||
T = promote_type(typeof(u),typeof(m)) | ||
ε = puresqrt(eps(real(typeof(m)))) | ||
N = nsteps(ε,typeof(m)) | ||
if abs(m) <= 1 && real(m) <= 0.5 | ||
return ellipj_viasmallm(u,m, Val{N}())::NTuple{3,T} | ||
elseif abs(1-m) <= 1 | ||
return ellipj_vialargem(u,m, Val{N}())::NTuple{3,T} | ||
elseif imag(m) == 0 && real(m) < 0 | ||
# [1, Sec 16.10] | ||
sn,cn,dn = ellipj_dispatch(u*sqrt(1-m),-m/(1-m)) | ||
return sn/(dn*sqrt(1-m)), cn/dn, 1/dn | ||
else | ||
# [1, Sec 16.11] | ||
sn,cn,dn = ellipj_dispatch(u*sqrt(m),1/m) | ||
return sn/sqrt(m), dn, cn | ||
end | ||
end | ||
|
||
|
||
#----------------------------------- | ||
# Type promotion and special values | ||
|
||
function ellipj_check(u,m) | ||
if isfinite(u) && isfinite(m) | ||
return ellipj_dispatch(u,m) | ||
else | ||
T = promote_type(typeof(u),typeof(m)) | ||
return (T(NaN),T(NaN),T(NaN)) | ||
end | ||
end | ||
|
||
ellipj(u::Real,m::Real) = ellipj_check(promote(float(u),float(m))...) | ||
function ellipj(u::Complex,m::Real) | ||
T = promote_type(real.(typeof.(float.((u,m))))...) | ||
return ellipj_check(convert(Complex{T},u), convert(T,m)) | ||
end | ||
ellipj(u,m::Complex) = ellipj_check(promote(float(u),float(m))...) | ||
|
||
""" | ||
ellipj(u,m) -> sn,cn,dn | ||
|
||
Jacobi elliptic functions `sn`, `cn` and `dn`. | ||
|
||
Convenience function `jpq(u,m)` with `p,q ∈ {s,c,d,n}` are also | ||
provided, but this function is more efficient if more than one elliptic | ||
function with the same arguments is required. | ||
""" | ||
function ellipj end | ||
|
||
|
||
#----------------------- | ||
# Convenience functions | ||
|
||
chars = ("s","c","d") | ||
for (i,p) in enumerate(chars) | ||
pn = Symbol("j"*p*"n") | ||
np = Symbol("jn"*p) | ||
@eval begin | ||
$pn(u,m) = ellipj(u,m)[$i] | ||
$np(u,m) = 1/$pn(u,m) | ||
end | ||
end | ||
for p in (chars...,"n") | ||
pp = Symbol("j"*p*p) | ||
@eval $pp(u,m) = one(promote_type(typeof.(float.((u,m)))...)) | ||
end | ||
|
||
for p in chars, q in chars | ||
p == q && continue | ||
pq = Symbol("j"*p*q) | ||
pn = Symbol(p*"n") | ||
qn = Symbol(q*"n") | ||
@eval begin | ||
function $pq(u::Number,m::Number) | ||
sn,cn,dn = ellipj(u,m) | ||
return $pn/$qn | ||
end | ||
end | ||
end | ||
|
||
for p in (chars...,"n"), q in (chars...,"n") | ||
pq = Symbol("j"*p*q) | ||
@eval begin | ||
""" | ||
$(string($pq))(u,m) | ||
|
||
Jacobi elliptic function `$($p)$($q)`. | ||
|
||
See also `ellipj(u,m)` if more than one Jacobi elliptic function | ||
with the same arguments is required. | ||
""" | ||
function $pq end | ||
end | ||
end | ||
|
||
|
||
#---------------------------------------------- | ||
# Complete elliptic integral of the first kind | ||
|
||
function iK_agm(m) | ||
# [1, Sec 17.6] | ||
T = typeof(m) | ||
m == 0 && return T(Inf) | ||
isnan(m) && return T(NaN) | ||
a,b = one(m),sqrt(m) | ||
while abs(a-b) > 2*eps(abs(a)) | ||
a,b = (a+b)/2,sqrt(a*b) | ||
end | ||
return T(π)/(2*a) # https://github.com/JuliaLang/julia/issues/26324 | ||
end | ||
iK(m) = iK_agm(float(m)) | ||
K(m::Real) = iK(1-m) | ||
function K(m::Complex) | ||
# Make sure we hit the "right" branch of sqrt if imag(m) == 0. | ||
# Here, "right" is defined as being consistent with mpmath. | ||
if imag(m) == 0 | ||
return iK(complex(1-real(m),imag(m))) | ||
else | ||
return iK(1-m) | ||
end | ||
end | ||
|
||
""" | ||
iK(m1) | ||
|
||
Evaluate `K(1-m1)` with better precision for small values of `m1`. | ||
""" | ||
function iK end | ||
|
||
doc""" | ||
K(m) | ||
|
||
Complete elliptic integral of the first kind. | ||
|
||
```math | ||
\begin{aligned} | ||
K(m) | ||
&= \int_0^1 \big((1-t^2)\,(1-mt^2)\big)^{-1/2} \, dt \\ | ||
&= \int_0^{π/2} (1-m \sin^2\theta)^{-1/2} \, d\theta | ||
\end{aligned} | ||
``` | ||
""" | ||
function K end |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What are all these exports?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah, I see, they're generated by macros. The names are a bit cryptic: are these standard? Maybe prefix them (
elliptic_jss
) or stick them in a module (Elliptic.jss
)?There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There are 16 Jacobi elliptic functions, namely all pairs of the letters scdn. Since
cd
is already taken, I decided to put aj
(for Jacobi) in front of the function to avoid name collision. Elliptic.jl solved this issue by putting these functions into a moduleJacobi
, but I'd sayjsn(u,m)
is more convenient thanJacobi.sn(u,m)
. Might be worth discussing, though.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ah ok, how about
jacobisn
? This matches nicely withbesselk
/besselj
, etc.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
jacobisn
possibly makes it hard to distinguish the functions, and it wouldn't really save much typing compared toJacobi.sn
. It might be worth to get input from someone who uses these functions regularly, though.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looking at other packages:
JacobiSN
ellipj
function.gsl_sf_elljac_e
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We might also want to rename the
K
andiK
functions. It's just too much of a hassle not to be able to useK
for anything else. For reference:EllipticK
ellipticK
ellipk
gsl_sf_ellint_Kcomp
I'd vote for
ellipK
, as the K is usually uppercase in mathematical notation. Definitely notellipticK
unless we renameellipj
.Also, I am a bit unsure about providing
iK
and how it should be named. If you look at the code, it definitely makes sense to provideiK
becauseK(m)
is really justiK(1-m)
and soK(1-m) = iK(1-(1-m))
which is just stupid. But AFAIK no other software package provides this, which means I might be doing something wrong. And the commonly used mathematical notation forK(1-m)
seems to be K', but obviously we can't use that so we need to think of something else.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'd still prefer to rename the functions to longer names.