Skip to content
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

Tolerances New PR #161

Open
wants to merge 15 commits into
base: main
Choose a base branch
from
72 changes: 50 additions & 22 deletions src/BSeries.jl
Original file line number Diff line number Diff line change
Expand Up @@ -1748,16 +1748,20 @@ function renormalize!(series)
end

"""
energy_preserving_order(rk::RungeKuttaMethod, max_order)
energy_preserving_order(rk::RungeKuttaMethod, max_order; tol=nothing)

This function checks up to which order a Runge-Kutta method `rk` is
energy-preserving for Hamiltonian problems.
It requires a `max_order` so that it does not run forever if the order up to
which the method is energy-preserving is too big or infinite.
Keyword Arguments:
- `tol = nothing`: The absolute tolerance for energy preservation. Default value
is `tol = 100 * eps(V)` if `valtype(rk) == AbstractFloat`, and `tol = zero(V)`
for rational numbers.
Sondar74 marked this conversation as resolved.
Show resolved Hide resolved

See also [`is_energy_preserving`](@ref)
"""
function energy_preserving_order(rk::RungeKuttaMethod, max_order)
function energy_preserving_order(rk::RungeKuttaMethod, max_order; tol=nothing)
p = 0
not_energy_preserving = false
while not_energy_preserving == false
Expand All @@ -1766,7 +1770,7 @@ function energy_preserving_order(rk::RungeKuttaMethod, max_order)
return max_order
end
# Check energy preservation up to the given order
if is_energy_preserving(rk, p + 1) == false
if is_energy_preserving(rk, p + 1, tol=tol) == false
not_energy_preserving = true
end
p = p + 1
Expand All @@ -1775,23 +1779,31 @@ function energy_preserving_order(rk::RungeKuttaMethod, max_order)
end

"""
is_energy_preserving(rk::RungeKuttaMethod, order)::Bool
is_energy_preserving(rk::RungeKuttaMethod, order; tol=nothing)::Bool

This function checks whether the Runge-Kutta method `rk` is
energy-preserving for Hamiltonian systems up to a given `order`.
Keyword Arguments:
- `tol = nothing`: The absolute tolerance for energy preservation. Default value
is `tol = 100 * eps(V)` if `valtype(rk) == AbstractFloat`, and `tol = zero(V)`
for rational numbers.
"""
function is_energy_preserving(rk::RungeKuttaMethod, order)
function is_energy_preserving(rk::RungeKuttaMethod, order; tol=nothing)
series = bseries(rk, order)
return is_energy_preserving(series)
return is_energy_preserving(series, tol=tol)
end

"""
is_energy_preserving(series_integrator)::Bool
is_energy_preserving(series_integrator; tol=nothing)::Bool

This function checks whether the B-series `series_integrator` of a time
integration method is energy-preserving for Hamiltonian systems - up to the
[`order`](@ref) of `series_integrator`.

Keyword Arguments:
- `tol = nothing`: The absolute tolerance for energy preservation. Default value
is `tol = 100 * eps(V)` if `valtype(series_integrator) == AbstractFloat`, and `tol = zero(V)`
for rational numbers.

# References

This code is based on the Theorem 2 of
Expand All @@ -1800,13 +1812,16 @@ This code is based on the Theorem 2 of
Foundations of Computational Mathematics 10 (2010): 673-693.
[DOI: 10.1007/s10208-010-9073-1](https://link.springer.com/article/10.1007/s10208-010-9073-1)
"""
function is_energy_preserving(series_integrator)
function is_energy_preserving(series_integrator; tol=nothing)
# This method requires the B-series of a map as input
let t = first(keys(series_integrator))
@assert isempty(t)
@assert isone(series_integrator[t])
end

# tolerance
V = valtype(series_integrator)
tol = energy_preserving_default_tolerance(V, tol)
# Theorem 2 of Celledoni et al. (2010) requires working with the modified
# equation. The basic idea is to check whether the modified equation lies
# in a certain subspace spanned by energy-preserving pairs of trees.
Expand Down Expand Up @@ -1835,7 +1850,7 @@ function is_energy_preserving(series_integrator)
# for order less than 4 because the energy-preserving basis has only one element.
# The following function checks the energy-preserving condition up to order 4
# together.
if _is_energy_preserving_low_order(trees, coefficients) == false
if _is_energy_preserving_low_order(trees, coefficients, tol) == false
return false
end

Expand All @@ -1852,7 +1867,7 @@ function is_energy_preserving(series_integrator)
push!(same_order_trees, trees[j])
end
end
if _is_energy_preserving(same_order_trees, same_order_coeffs) == false
if _is_energy_preserving(same_order_trees, same_order_coeffs, tol) == false
return false
end
end
Expand All @@ -1864,7 +1879,7 @@ end
# and the set of 'coefficients' corresponding to a B-series is
# energy_preserving up to the 'uppermost_order', which we choose
# to be 4 for computational optimization
function _is_energy_preserving_low_order(trees, coefficients)
function _is_energy_preserving_low_order(trees, coefficients, tol)
# Set the limit up to which this function will work
uppermost_order = 4
same_order_trees = empty(trees)
Expand All @@ -1877,7 +1892,7 @@ function _is_energy_preserving_low_order(trees, coefficients)
push!(same_order_trees, t)
push!(same_order_coeffs, coefficients[index])
end
return _is_energy_preserving(same_order_trees, same_order_coeffs)
return _is_energy_preserving(same_order_trees, same_order_coeffs, tol)
end

# This function is the application of Theorem 2 of the paper
Expand All @@ -1886,7 +1901,7 @@ end
# of coefficients.
# Checks whether `trees` and `ocefficients` satisfify the energy_preserving
# condition.
function _is_energy_preserving(trees, coefficients)
function _is_energy_preserving(trees, coefficients, tol)
# TODO: `Float32` would also be nice to have. However, the default tolerance
# of the rank computation is based on `Float64`. Thus, it will usually
# not work with coefficients given only in 32 bit precision.
Expand All @@ -1897,13 +1912,13 @@ function _is_energy_preserving(trees, coefficients)
Rational{Int8}, Rational{Int16}, Rational{Int32}, Rational{Int64},
Rational{Int128}})
# These types support efficient computations in sparse matrices
_is_energy_preserving_sparse(trees, coefficients)
_is_energy_preserving_sparse(trees, coefficients, tol)
else
_is_energy_preserving_dense(trees, coefficients)
_is_energy_preserving_dense(trees, coefficients, tol)
end
end

function _is_energy_preserving_dense(trees, coefficients)
function _is_energy_preserving_dense(trees, coefficients, tol)
# For every tree, obtain the adjoint and check if it exists
length_coeff = length(trees)
# Provided that a tree can have several adjoints, for every tree `t`
Expand All @@ -1916,12 +1931,11 @@ function _is_energy_preserving_dense(trees, coefficients)
for (t_index, t) in enumerate(trees)
# We do not need to consider the empty tree
isempty(t) && continue

# Check whether the level sequence corresponds to a bushy tree.
# If so, we can exit early since the coefficients of bushy trees
# must be zero in the modified equation of energy-preserving methods.
if length(t) > 1 && is_bushy(t)
if !iszero(coefficients[t_index])
if !isapprox(coefficients[t_index], 0, atol = tol)
return false
end
else
Expand Down Expand Up @@ -1969,7 +1983,7 @@ end

# This method is specialized for coefficients that can be used efficiently in
# sparse arrays.
function _is_energy_preserving_sparse(trees, coefficients)
function _is_energy_preserving_sparse(trees, coefficients, tol)
# For every tree, obtain the adjoint and check if it exists.
# Provided that a tree can have several adjoints, for every tree `t`
# we need to check if there is a linear combination of the coefficients
Expand All @@ -1986,12 +2000,11 @@ function _is_energy_preserving_sparse(trees, coefficients)
for (t_index, t) in enumerate(trees)
# We do not need to consider the empty tree
isempty(t) && continue

# Check whether the level sequence corresponds to a bushy tree.
# If so, we can exit early since the coefficients of bushy trees
# must be zero in the modified equation of energy-preserving methods.
if length(t) > 1 && is_bushy(t)
if !iszero(coefficients[t_index])
if !isapprox(coefficients[t_index], 0, atol = tol)
return false
end
else
Expand Down Expand Up @@ -2218,4 +2231,19 @@ function equivalent_trees(tree)
return equivalent_trees_set
end

# This function calculates the default tolerance for 'is_energy_preserving'
function energy_preserving_default_tolerance(V, tol)
# We use `nothing` as default value
if tol === nothing
if V <: AbstractFloat
# For floating point numbers, we cannot expect exact calculations in general.
tol = 100 * eps(V)
else
# If something like rational numbers are used, we do not expect numerical errors.
tol = zero(V)
end
end
return tol
end

end # module
17 changes: 17 additions & 0 deletions test/runtests.jl
Original file line number Diff line number Diff line change
Expand Up @@ -2115,6 +2115,23 @@ using Aqua: Aqua
series = bseries(AverageVectorFieldMethod(BigFloat), 7)
# TODO: This test is currently broken and throws an error
@test_broken is_energy_preserving(series)

# Generated using RK-Opt
# References:
# Ketcheson et al., (2020). RK-Opt: A package for the design of numerical ODE solvers.
# Journal of Open Source Software, 5(54), 2514, https://doi.org/10.21105/joss.02514
A = [0.0000000000000000E+00 0.0000000000000000E+00 0.0000000000000000E+00 0.0000000000000000E+00
-2.0314781009861151E-02 0.0000000000000000E+00 0.0000000000000000E+00 0.0000000000000000E+00
1.0721250651661693E+01 -1.0078547105299583E+01 0.0000000000000000E+00 0.0000000000000000E+00
-4.1299845612430317E+01 4.0239538957660457E+01 2.0603066547698594E+00 0.0000000000000000E+00]
b = [2.0623108969503225E+00
-1.7306596124709785E+00
5.6957376564633877E-01
9.8774949874317328E-02 ]
rk = RungeKuttaMethod(A,b)
series = bseries(rk)
@test order_of_accuracy(series) == 4
@test energy_preserving_order(rk, 10) == 4
Sondar74 marked this conversation as resolved.
Show resolved Hide resolved
end

@testset "Symbolic coefficients" begin
Expand Down