Time integration methods

Methods from OrdinaryDiffEq.jl

Trixi.jl is compatible with the SciML ecosystem for ordinary differential equations. In particular, explicit Runge-Kutta methods from OrdinaryDiffEq.jl are tested extensively. Interesting classes of time integration schemes are

Some common options for solve from OrdinaryDiffEq.jl are the following. Further documentation can be found in the SciML docs.

  • If you use a fixed time step method like CarpenterKennedy2N54, you need to pass a time step as dt=.... If you use a StepsizeCallback, the value passed as dt=... is irrelevant since it will be overwritten by the StepsizeCallback. If you want to use an adaptive time step method such as SSPRK43 or RDPK3SpFSAL49 and still want to use CFL-based step size control via the StepsizeCallback, you need to pass the keyword argument adaptive=false to solve.
  • You should usually set save_everystep=false. Otherwise, OrdinaryDiffEq.jl will (try to) save the numerical solution after every time step in RAM (until you run out of memory or start to swap).
  • You can set the maximal number of time steps via maxiters=....
  • SSP methods and many low-storage methods from OrdinaryDiffEq.jl support stage_limiter!s and step_limiter!s, e.g., PositivityPreservingLimiterZhangShu and EntropyBoundedLimiter from Trixi.jl.
  • If you start Julia with multiple threads and want to use them also in the time integration method from OrdinaryDiffEq.jl, you need to pass the keyword argument thread=OrdinaryDiffEq.True() to the algorithm, e.g., RDPK3SpFSAL49(thread=OrdinaryDiffEq.True()) or CarpenterKennedy2N54(thread=OrdinaryDiffEq.True(), williamson_condition=false). For more information on using thread-based parallelism in Trixi.jl, please refer to Shared-memory parallelization with threads.
  • If you use error-based step size control (see also the section on error-based adaptive step sizes) together with MPI, you need to pass internalnorm=ode_norm and you should pass unstable_check=ode_unstable_check to OrdinaryDiffEq's solve, which are both included in ode_default_options.
Number of `rhs!` calls

If you use explicit Runge-Kutta methods from OrdinaryDiffEq.jl, the total number of rhs! calls can be (slightly) bigger than the number of steps times the number of stages, e.g. to allow for interpolation (dense output), root-finding for continuous callbacks, and error-based time step control. In general, you often should not need to worry about this if you use Trixi.jl.

Custom Optimized Schemes

Stabilized Explicit Runge-Kutta Methods

Optimized explicit schemes aim to maximize the timestep $\Delta t$ for a given simulation setup. Formally, this boils down to an optimization problem of the form

\[\underset{P_{p;S} \, \in \, \mathcal{P}_{p;S}}{\max} \Delta t \text{ such that } \big \vert P_{p;S}(\Delta t \lambda_m) \big \vert \leq 1, \quad m = 1 , \dots , M \tag{1}\]

where $p$ denotes the order of consistency of the scheme, $S$ is the number of stage evaluations and $M$ denotes the number of eigenvalues $\lambda_m$ of the Jacobian matrix $J \coloneqq \frac{\partial \boldsymbol F}{\partial \boldsymbol U}$ of the right-hand side of the semidiscretized PDE:

\[\dot{\boldsymbol U} = \boldsymbol F(\boldsymbol U) \tag{2} \: .\]

In particular, for $S > p$ the Runge-Kutta method includes some free coefficients which may be used to adapt the domain of absolute stability to the problem at hand. Since Trixi.jl supports exact computation of the Jacobian $J$ by means of automatic differentiation, we have access to the Jacobian of a given simulation setup. For small (say, up to roughly $10^4$ DoF) systems, the spectrum $\boldsymbol \sigma = \left \{ \lambda_m \right \}_{m=1, \dots, M}$ can be computed directly using LinearAlgebra.eigvals(J). For larger systems, we recommend the procedure outlined in section 4.1 of Doehring et al. (2024). This approach computes a reduced set of (estimated) eigenvalues $\widetilde{\boldsymbol \sigma}$ around the convex hull of the spectrum by means of the Arnoldi method.

The optimization problem (1) can be solved using the algorithms described in Ketcheson, Ahmadia (2012) for a moderate number of stages $S$ or Doehring, Gassner, Torrilhon (2024) for a large number of stages $S$. In Trixi.jl, the former approach is implemented by means of convex optimization using the Convex.jl package.

The resulting stability polynomial $P_{p;S}$ is then used to construct an optimized Runge-Kutta method. Trixi.jl implements the Paired-Explicit Runge-Kutta (PERK) method, a low-storage, multirate-ready method with optimized domain of absolute stability.

Paired-Explicit Runge-Kutta (PERK) Schemes

Paired-Explicit Runge-Kutta (PERK) or PairedExplicitRK schemes are an advanced class of numerical methods designed to efficiently solve ODEs. In the original publication, second-order schemes were introduced, which have been extended to third and fourth order in subsequent works.

By construction, PERK schemes are suited for integrating multirate systems, i.e., systems with varying characteristics speeds throughout the domain. Nevertheless, due to their optimized stability properties and low-storage nature, the PERK schemes are also highly efficient when applied as standalone methods. In Trixi.jl, the standalone PERK integrators are implemented such that all stages of the method are active.

Tutorial: Using PairedExplicitRK2

In this tutorial, we will demonstrate how you can use the second-order PERK time integrator. You need the packages Convex.jl and ECOS.jl, so be sure they are added to your environment.

First, you need to load the necessary packages:

using Convex, ECOS
using OrdinaryDiffEq
using Trixi

Then, define the ODE problem and the semidiscretization setup. For this example, we will use a simple advection problem.

# Define the mesh
cells_per_dimension = 100
coordinates_min = 0.0
coordinates_max = 1.0
mesh = StructuredMesh(cells_per_dimension,
                      coordinates_min, coordinates_max)

# Define the equation and initial condition
advection_velocity = 1.0
equations = LinearScalarAdvectionEquation1D(advection_velocity)

initial_condition = initial_condition_convergence_test

# Define the solver
solver = DGSEM(polydeg = 3, surface_flux = flux_lax_friedrichs)

# Define the semidiscretization
semi = SemidiscretizationHyperbolic(mesh,
                                    equations, initial_condition,
                                    solver)
┌──────────────────────────────────────────────────────────────────────────────────────────────────┐
│ SemidiscretizationHyperbolic                                                                     │
│ ════════════════════════════                                                                     │
│ #spatial dimensions: ………………………… 1                                                                │
│ mesh: ………………………………………………………………… StructuredMesh{1, Float64}                                       │
│ equations: …………………………………………………… LinearScalarAdvectionEquation1D                                  │
│ initial condition: ……………………………… initial_condition_convergence_test                               │
│ boundary conditions: ………………………… Trixi.BoundaryConditionPeriodic                                  │
│ source terms: …………………………………………… nothing                                                          │
│ solver: …………………………………………………………… DG                                                               │
│ total #DOFs per field: …………………… 400                                                              │
└──────────────────────────────────────────────────────────────────────────────────────────────────┘

After that, we will define the necessary callbacks for the simulation. Callbacks are used to perform actions at specific points during the integration process.

# Define some standard callbacks
summary_callback  = SummaryCallback()
alive_callback    = AliveCallback()
analysis_callback = AnalysisCallback(semi, interval = 200)
# For this optimized method we can use a relatively large CFL number
stepsize_callback = StepsizeCallback(cfl = 2.5)

# Create a CallbackSet to collect all callbacks
callbacks = CallbackSet(summary_callback,
                        alive_callback,
                        analysis_callback,
                        stepsize_callback)
CallbackSet{Tuple{}, Tuple{DiscreteCallback{typeof(Trixi.summary_callback), typeof(Trixi.summary_callback), Trixi.var"#initialize#1437"{Bool}, typeof(SciMLBase.FINALIZE_DEFAULT)}, DiscreteCallback{AliveCallback, AliveCallback, typeof(Trixi.initialize!), typeof(SciMLBase.FINALIZE_DEFAULT)}, DiscreteCallback{Trixi.var"#1443#1450"{Int64}, AnalysisCallback{Trixi.LobattoLegendreAnalyzer{Float64, 7, SVector{7, Float64}, Matrix{Float64}}, Tuple{typeof(Trixi.entropy_timederivative)}, SVector{1, Float64}, @NamedTuple{u_local::StrideArraysCore.StaticStrideArray{Float64, 2, (1, 2), Tuple{Static.StaticInt{1}, Static.StaticInt{7}}, Tuple{Nothing, Nothing}, Tuple{Static.StaticInt{1}, Static.StaticInt{1}}, 7}, x_local::StrideArraysCore.StaticStrideArray{Float64, 2, (1, 2), Tuple{Static.StaticInt{1}, Static.StaticInt{7}}, Tuple{Nothing, Nothing}, Tuple{Static.StaticInt{1}, Static.StaticInt{1}}, 7}, jacobian_local::StrideArraysCore.StaticStrideArray{Float64, 1, (1,), Tuple{Static.StaticInt{7}}, Tuple{Nothing}, Tuple{Static.StaticInt{1}}, 7}}}, typeof(Trixi.initialize!), typeof(SciMLBase.FINALIZE_DEFAULT)}, DiscreteCallback{StepsizeCallback{Float64}, StepsizeCallback{Float64}, typeof(Trixi.initialize!), typeof(SciMLBase.FINALIZE_DEFAULT)}}}((), (SummaryCallback, AliveCallback(alive_interval=0), DiscreteCallback{Trixi.var"#1443#1450"{Int64}, AnalysisCallback{Trixi.LobattoLegendreAnalyzer{Float64, 7, SVector{7, Float64}, Matrix{Float64}}, Tuple{typeof(Trixi.entropy_timederivative)}, SVector{1, Float64}, @NamedTuple{u_local::StrideArraysCore.StaticStrideArray{Float64, 2, (1, 2), Tuple{Static.StaticInt{1}, Static.StaticInt{7}}, Tuple{Nothing, Nothing}, Tuple{Static.StaticInt{1}, Static.StaticInt{1}}, 7}, x_local::StrideArraysCore.StaticStrideArray{Float64, 2, (1, 2), Tuple{Static.StaticInt{1}, Static.StaticInt{7}}, Tuple{Nothing, Nothing}, Tuple{Static.StaticInt{1}, Static.StaticInt{1}}, 7}, jacobian_local::StrideArraysCore.StaticStrideArray{Float64, 1, (1,), Tuple{Static.StaticInt{7}}, Tuple{Nothing}, Tuple{Static.StaticInt{1}}, 7}}}, typeof(Trixi.initialize!), typeof(SciMLBase.FINALIZE_DEFAULT)}(Trixi.var"#1443#1450"{Int64}(200), AnalysisCallback{Trixi.LobattoLegendreAnalyzer{Float64, 7, SVector{7, Float64}, Matrix{Float64}}, Tuple{typeof(Trixi.entropy_timederivative)}, SVector{1, Float64}, @NamedTuple{u_local::StrideArraysCore.StaticStrideArray{Float64, 2, (1, 2), Tuple{Static.StaticInt{1}, Static.StaticInt{7}}, Tuple{Nothing, Nothing}, Tuple{Static.StaticInt{1}, Static.StaticInt{1}}, 7}, x_local::StrideArraysCore.StaticStrideArray{Float64, 2, (1, 2), Tuple{Static.StaticInt{1}, Static.StaticInt{7}}, Tuple{Nothing, Nothing}, Tuple{Static.StaticInt{1}, Static.StaticInt{1}}, 7}, jacobian_local::StrideArraysCore.StaticStrideArray{Float64, 1, (1,), Tuple{Static.StaticInt{7}}, Tuple{Nothing}, Tuple{Static.StaticInt{1}}, 7}}}(0.0, 0.0, 0, 0.0, 200, false, "out", "analysis.dat", LobattoLegendreAnalyzer{Float64}(polydeg=6), [:l2_error, :linf_error], (Trixi.entropy_timederivative,), [0.0], (u_local = [0.0 0.0 … 0.0 0.0], x_local = [0.0 0.0 … 0.0 0.0], jacobian_local = [0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0])), Trixi.initialize!, SciMLBase.FINALIZE_DEFAULT, Bool[0, 0]), StepsizeCallback(cfl_number=2.5)))

Now, we define the ODE problem by specifying the time span over which the ODE will be solved. The tspan parameter is a tuple (t_start, t_end) that defines the start and end times for the simulation. The semidiscretize function is used to create the ODE problem from the simulation setup.

# Define the time span
tspan = (0.0, 1.0)

# Create ODE problem with time span from 0.0 to 1.0
ode = semidiscretize(semi, tspan)
ODEProblem with uType Vector{Float64} and tType Float64. In-place: true
timespan: (0.0, 1.0)
u0: 400-element Vector{Float64}:
 1.0
 1.0043415197116643
 1.01136541003923
 1.0157053795390643
 1.0157053795390643
 1.0200441649067613
 1.0270611235025782
 1.0313952597646567
 1.0313952597646567
 1.0357270289310565
 ⋮
 1.031395259764657
 1.0313952597646565
 1.027061123502578
 1.020044164906761
 1.0157053795390643
 1.015705379539064
 1.01136541003923
 1.0043415197116643
 1.0000000000000002

Next, we will construct the time integrator. In order to do this, you need the following components:

  • Number of stages: The number of stages $S$ in the Runge-Kutta method.

In this example, we use 6 stages.

  • Time span (tspan): A tuple (t_start, t_end) that defines the time span over which the ODE will be solved.

This defines the bounds for the bisection routine for the optimal timestep $\Delta t$ used in calculating the polynomial coefficients at optimization stage. This variable is already defined in step 5.

  • Semidiscretization (semi): The semidiscretization setup that includes the mesh, equations, initial condition, and solver. In this example, this variable is already defined in step 3.

In the background, we compute from semi the Jacobian $J$ evaluated at the initial condition using jacobian_ad_forward. This is then followed by the computation of the spectrum $\boldsymbol \sigma(J)$ using LinearAlgebra.eigvals. Equipped with the spectrum, the optimal stability polynomial is computed, from which the corresponding Runge-Kutta method is derived. Other constructors (if the coefficients $\boldsymbol{\alpha}$ of the stability polynomial are already available, or if a reduced spectrum $\widetilde{\boldsymbol{\sigma}}$ should be used) are discussed below.

# Construct second order-explicit Runge-Kutta method with 6 stages for given simulation setup (`semi`)
# `tspan` provides the bounds for the bisection routine that is used to calculate the maximum timestep.
ode_algorithm = Trixi.PairedExplicitRK2(6, tspan, semi)
Trixi.PairedExplicitRK2(6, [0.07844829670432756 0.10354700919663037 0.10796171666470256 0.06436285309436707; 0.12155170329567246 0.19645299080336961 0.29203828333529747 0.43563714690563293], [0.0, 0.1, 0.2, 0.3, 0.4, 0.5], 0.0, 1.0, 0.5, 0.005713800899684429)

With everything set up, you can now use Trixi.solve to solve the ODE problem. The solve function takes the ODE problem, the time integrator, and some options such as the time step (dt), whether to save every step (save_everystep), and the callbacks.

# Solve the ODE problem using PERK2
sol = Trixi.solve(ode, ode_algorithm,
                  dt = 1.0, # overwritten by `stepsize_callback`
                  save_everystep = false, callback = callbacks)
Trixi.TimeIntegratorSolution{Tuple{Float64, Float64}, Tuple{Vector{Float64}, Vector{Float64}}, ODEProblem{Vector{Float64}, Tuple{Float64, Float64}, true, SemidiscretizationHyperbolic{StructuredMesh{1, Float64}, LinearScalarAdvectionEquation1D{Float64}, typeof(initial_condition_convergence_test), Trixi.BoundaryConditionPeriodic, Nothing, DGSEM{LobattoLegendreBasis{Float64, 4, SVector{4, Float64}, Matrix{Float64}, Matrix{Float64}, Matrix{Float64}}, Trixi.LobattoLegendreMortarL2{Float64, 4, Matrix{Float64}, Matrix{Float64}}, SurfaceIntegralWeakForm{FluxLaxFriedrichs{typeof(max_abs_speed_naive)}}, VolumeIntegralWeakForm}, @NamedTuple{elements::Trixi.ElementContainer{1, Float64, Float64, 2, 3, 4}}}, ODEFunction{true, SciMLBase.FullSpecialize, typeof(Trixi.rhs!), LinearAlgebra.UniformScaling{Bool}, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, typeof(SciMLBase.DEFAULT_OBSERVED), Nothing, Nothing}, Base.Pairs{Symbol, Union{}, Tuple{}, @NamedTuple{}}, SciMLBase.StandardODEProblem}}((0.0, 1.0), ([1.0, 1.0043415197116643, 1.01136541003923, 1.0157053795390643, 1.0157053795390643, 1.0200441649067613, 1.0270611235025782, 1.0313952597646567, 1.0313952597646567, 1.0357270289310565  …  1.0357270289310565, 1.031395259764657, 1.0313952597646565, 1.027061123502578, 1.020044164906761, 1.0157053795390643, 1.015705379539064, 1.01136541003923, 1.0043415197116643, 1.0000000000000002], [-5.025587415119955e16, 3.974993904782206e15, -2.05007280153356e14, 3.4473429115291016e16, -2.5741750369884024e16, -4.654920877269446e15, -3.094268021793455e14, -3.842196774842362e16, 9.977556739132586e16, -1.4179725197049346e16  …  -5.027100469086303e15, -2.139842882748817e16, 3.403562084688463e16, 1.4173359905786425e15, -5.191256780739955e15, -1.848018013892012e16, 4.52373760175481e16, -1.466974896597132e15, 1.0507226656279805e15, 2.7975731506548172e16]), ODEProblem{Vector{Float64}, Tuple{Float64, Float64}, true, SemidiscretizationHyperbolic{StructuredMesh{1, Float64}, LinearScalarAdvectionEquation1D{Float64}, typeof(initial_condition_convergence_test), Trixi.BoundaryConditionPeriodic, Nothing, DGSEM{LobattoLegendreBasis{Float64, 4, SVector{4, Float64}, Matrix{Float64}, Matrix{Float64}, Matrix{Float64}}, Trixi.LobattoLegendreMortarL2{Float64, 4, Matrix{Float64}, Matrix{Float64}}, SurfaceIntegralWeakForm{FluxLaxFriedrichs{typeof(max_abs_speed_naive)}}, VolumeIntegralWeakForm}, @NamedTuple{elements::Trixi.ElementContainer{1, Float64, Float64, 2, 3, 4}}}, ODEFunction{true, SciMLBase.FullSpecialize, typeof(Trixi.rhs!), LinearAlgebra.UniformScaling{Bool}, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, typeof(SciMLBase.DEFAULT_OBSERVED), Nothing, Nothing}, Base.Pairs{Symbol, Union{}, Tuple{}, @NamedTuple{}}, SciMLBase.StandardODEProblem}(ODEFunction{true, SciMLBase.FullSpecialize, typeof(Trixi.rhs!), LinearAlgebra.UniformScaling{Bool}, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, Nothing, typeof(SciMLBase.DEFAULT_OBSERVED), Nothing, Nothing}(Trixi.rhs!, LinearAlgebra.UniformScaling{Bool}(true), nothing, nothing, nothing, nothing, nothing, nothing, nothing, nothing, nothing, nothing, nothing, nothing, nothing, nothing, SciMLBase.DEFAULT_OBSERVED, nothing, nothing), [1.0, 1.0043415197116643, 1.01136541003923, 1.0157053795390643, 1.0157053795390643, 1.0200441649067613, 1.0270611235025782, 1.0313952597646567, 1.0313952597646567, 1.0357270289310565  …  1.0357270289310565, 1.031395259764657, 1.0313952597646565, 1.027061123502578, 1.020044164906761, 1.0157053795390643, 1.015705379539064, 1.01136541003923, 1.0043415197116643, 1.0000000000000002], (0.0, 1.0), SemidiscretizationHyperbolic(StructuredMesh{1, Float64}, LinearScalarAdvectionEquation1D with one variable, initial_condition_convergence_test, boundary_condition_periodic, nothing, DG{Float64}(LobattoLegendreBasis{Float64}(polydeg=3), LobattoLegendreMortarL2{Float64}(polydeg=3), SurfaceIntegralWeakForm{FluxLaxFriedrichs{typeof(max_abs_speed_naive)}}(FluxLaxFriedrichs(max_abs_speed_naive)), VolumeIntegralWeakForm()), cache(elements)), Base.Pairs{Symbol, Union{}, Tuple{}, @NamedTuple{}}(), SciMLBase.StandardODEProblem()))
Advanced constructors:

There are two additional constructors for the PairedExplicitRK2 method besides the one taking in a semidiscretization semi:

  • PairedExplicitRK2(num_stages, base_path_monomial_coeffs::AbstractString) constructs a num_stages-stage method from the given optimal monomial_coeffs $\boldsymbol \alpha$.

These are expected to be present in the provided directory in the form of a gamma_<S>.txt file, where <S> is the number of stages num_stages. This constructor is useful when the optimal coefficients cannot be obtained using the optimization routine by Ketcheson and Ahmadia, possibly due to a large number of stages $S$.

  • PairedExplicitRK2(num_stages, tspan, eig_vals::Vector{ComplexF64}) constructs a num_stages-stage using the optimization approach by Ketcheson and Ahmadia for the (reduced) spectrum eig_vals.

The use-case for this constructor would be a large system, for which the computation of all eigenvalues is infeasible.

Automatic computation of a stable CFL Number

In the previous tutorial the CFL number was set manually to $2.5$. To avoid the manual trial-and-error process behind this, instantiations of AbstractPairedExplicitRK methods can automatically compute the stable CFL number for a given simulation setup using the Trixi.calculate_cfl function. When constructing the time integrator from a semidiscretization semi,

# Construct third-order paired-explicit Runge-Kutta method with 8 stages for given simulation setup.
ode_algorithm = Trixi.PairedExplicitRK3(8, tspan, semi)
Trixi.PairedExplicitRK3(8, [0.33273712758230917 0.49289754294489996 … 0.7501069018997653 0.3124732745250587; 0.06726287837815531 0.10710248089695798 … 0.2498930981002347 0.1875267254749413], [0.0, 0.20000000298023224, 0.4000000059604645, 0.6000000238418579, 0.800000011920929, 1.0, 1.0, 0.5], 0.008727616630494595)

the maximum timestep dt is stored by the ode_algorithm. This can then be used to compute the stable CFL number for the given simulation setup:

cfl_number = Trixi.calculate_cfl(ode_algorithm, ode)
3.4910466521985324

For nonlinear problems, the spectrum will in general change over the course of the simulation. Thus, it is often necessary to reduce the optimal cfl_number by a safety factor:

# For non-linear problems, the CFL number should be reduced by a safety factor
# as the spectrum changes (in general) over the course of a simulation
stepsize_callback = StepsizeCallback(cfl = 0.85 * cfl_number)
┌──────────────────────────────────────────────────────────────────────────────────────────────────┐
│ StepsizeCallback                                                                                 │
│ ════════════════                                                                                 │
│ CFL number: ………………………………………………… 2.9673896543687523                                               │
└──────────────────────────────────────────────────────────────────────────────────────────────────┘

If the optimal monomial coefficients are precomputed, the user needs to provide the obtained maximum timestep dt_opt from the optimization at construction stage. The corresponding constructor has signature

PairedExplicitRK3(num_stages, base_path_a_coeffs::AbstractString,
                  dt_opt = nothing; cS2 = 1.0f0)

Then, the stable CFL number can be computed as described above.

Currently implemented PERK methods

Single/Standalone methods