-
Notifications
You must be signed in to change notification settings - Fork 5
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
13 changed files
with
997 additions
and
316 deletions.
There are no files selected for viewing
Large diffs are not rendered by default.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,122 @@ | ||
# Rosenbrock methods | ||
|
||
In this page, we introduce Rosenbrock-type methods to solve ordinary | ||
differential equations. In doing do, we roughly follow Chapter IV.7 of "Solving | ||
Ordinary Differential Equations II" by Hairer and Wanner. (Beware, notation is | ||
not the same.). In this spirit, let us introduce the need for Rosenbrock-type | ||
methods in the same way as Hairer and Wanner, by quoting Rosenbrock himself: | ||
|
||
> When the functions are non-linear, implicit equations can in general be solved | ||
> only by iteration. This is a severe drawback, as it adds to the problem of | ||
> stability, that of convergence of the iterative process. An alternative, which | ||
> avoids this difficulty, is ... | ||
Rosenbrock method! | ||
|
||
Before reading this page, we recommend reading the page on ODE solvers first | ||
[TODO: Add link] | ||
|
||
## Introduction to the formalism | ||
|
||
Let us consider an ordinary differential equation of the form | ||
|
||
$$ \frac{d}{dt}u(t) = T(u(t), t)\,,$$ | ||
|
||
where $u$ is the state, $T$ is the tendency, | ||
and $t$ the time. For the sake of simplicity, let us ignore the explicit time | ||
dependency in the tendency (we will get back to that at the end). | ||
|
||
The simplest way to introduce the Rosenbrock method is to start from a | ||
diagonally implicit Runge-Kutta scheme (see page on DIRK). In an implicit | ||
Runge-Kutta method with $s$ stages and tableau $a, b, c$, the updated value | ||
$u_1$ for a step of size $\Delta t$ is obtained starting from the known value $u_0$ | ||
with | ||
|
||
$$ u_1 = u_0 + \sum_{i=1}^s b_i k_i\,, $$ | ||
with | ||
$$ k_i = \Delta t T ( u_0 + \sum_{j=1}^{i-1}\alpha_{ij}k_j + \alpha_{ii} k_i)\,. $$ | ||
$\alpha_{ij}$, $b_i$ are carefully chosen coefficients. | ||
|
||
Rosenbrock's idea consists in linearizing $T$. This operation can be interpreted | ||
as taking one iteration for Netwon solver and yields | ||
|
||
$$ k_i = \Delta t T(g_i) + \Delta t T'(g_i) \alpha_{ii} k_i $$ | ||
|
||
with $J(g_i)$ Jacobian of the tendency and with | ||
|
||
$$ g_i = u_0 + \sum_{j=1}^{i-1}\alpha_{ij} k_j $$ | ||
|
||
In Rosenbrock-type methods, the Jacobian $T'$ is replaced with linear combinations | ||
of the Jacobian evaluated at $u_0$, $J$: | ||
$$ T'(g_i) \alpha_{ii} k_i \approx J \sum_{j=1}^{i-1}\gamma_{ij} k_j + J \gamma_{ii} k_i\,,$$ | ||
with $\gamma_{ij}$ other coefficients that can be chosen. | ||
|
||
Now, each stage consists of solving a system of linear equations in $k_i$ with | ||
matrix $I - \Delta t \gamma_{ii}$: $$ (I - J \Delta t \gamma_{ii}) k_i = \Delta | ||
t T(g_i) + J \sum_{j=1}^{i-1}\gamma_{ij} k_j$$ for each $i$ in $1, \dots, s$. | ||
Once $k_i$ are known, $u_1$ can is easily computed and the process repeated for | ||
the next step. | ||
|
||
In practice, there are computational advantages at implementing a slight | ||
variation of what presented here. | ||
|
||
Let us define $\tilde{k}_{i} = \sum_{j=1}^{i} \gamma_{ij} k_j$. If the matrix | ||
$\gamma_{ij}$ is invertible, we can move freely from $k_i$ to $\tilde{k}_i$. A | ||
convenient step is to also define $C$, as | ||
|
||
$$ C = diag(\gamma_{11}^{-1}, \gamma_{22}^{-1}, \dots, \gamma_{ss}^{-1}) - \Gamma^{-1} $$ | ||
|
||
Which establishes that | ||
|
||
$$ k_i = \frac{\tilde{k}_i}{\gamma_{ii}} - \sum_{j=1}^{i -1} c_{ij} \tilde{k}_j$$ | ||
Substituting this, we obtain the following equations | ||
|
||
$$ (J \Delta t \gamma_{ii} - 1) \tilde{k}_i = - \Delta | ||
t \gamma_{ii} T(g_i) - \gamma_{ii} J \sum_{j=1}^{i-1}c_{ij} \tilde{k}_j \,, $$ | ||
|
||
$$ g_i = u_0 + \sum_{j=1}^{i-1}a_{ij} \tilde{k}_j \,,$$ | ||
|
||
$$ u_1 = u_0 + \sum_{j=1}^{s} m_j \tilde{k}_j\,,$$ | ||
with | ||
$$ (a_{ij}) = (\alpha_{ij}) \Gamma^{-1}\,, $$ and $$ (m_i) = (b_i) \Gamma^{-1} $$ | ||
|
||
Finally, small changes are required to add support for explicit time derivative: | ||
|
||
$$ (J \Delta t \gamma_{ii} - 1) \tilde{k}_i = - \Delta | ||
t \gamma_{ii} T( t_0 + \alpha_i \Delta t, g_i) - \gamma_{ii} J \sum_{j=1}^{i-1}c_{ij} \tilde{k}_j - \Delta | ||
t \gamma_{ii} \gamma_i \Delta | ||
t \frac{\partial T}{\partial t}(t_0, u_0) $$ | ||
|
||
where we defined | ||
|
||
$\alpha_{i} = \sum_{j=1}^{i-1}\alpha_{ij} $ | ||
|
||
$ \gamma _{i} = \sum_{j=1}^{i}\gamma_{ij} $ | ||
|
||
> :note: You may wonder, what is up with all these ugly minus signs? Why don't | ||
> we cancel them out? The reason for this is because we want to have the | ||
> prefactor $(J \Delta t \gamma_{ii} - 1)$. This is called `Wfact` in the | ||
> language of `DifferentialEquations.jl`. Most other algorithms specify this | ||
> quantity, so it is convenient to be consistent. | ||
## Implementation | ||
|
||
In `ClimaTimeSteppers.jl`, we implement the last equation presented in the | ||
previous section. Currently, the only Rosenbrock-type algorithm implemented is | ||
`SSPKnoth`. `SSPKnoth` is 3-stage, second-order accurate Rosenbrock with | ||
|
||
$$ \alpha = \begin{bmatrix} | ||
0 & 0 & 0 \\ | ||
1 & 0 & 0 \\ | ||
\frac{1}{4} & \frac{1}{4} & 0 \\ | ||
\end{bmatrix} $$ | ||
|
||
$$ \Gamma = \begin{bmatrix} | ||
1 & 0 & 0 \\ | ||
1 & 1 & 0 \\ | ||
-\frac{3}{4} & \frac{3}{4} & 1 \\ | ||
\end{bmatrix} $$ | ||
|
||
and $$ b = (\frac{1}{6}, \frac{1}{6}, \frac{2}{3}) $$. | ||
|
||
At each stage, the state is updated to $g_i$ and precomputed quantities are updated. Tendencies are computed and, unless the stage is the last, DSS is applied to the sum of all the tendencies. After the last stage, DSS is applied to the incremented state. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -51,6 +51,7 @@ HOMMEM1 | |
ARK2GKC | ||
ARK437L2SA1 | ||
ARK548L2SA2 | ||
SSPKnoth | ||
``` | ||
|
||
## Explicit Algorithm Names | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.