Elliptic solvers
The elliptic problem for the pressure
The 3D non-hydrostatic pressure field is obtained by taking the divergence of the horizontal component of the momentum equations and invoking the vertical component to yield an elliptic Poisson equation for the non-hydrostatic kinematic pressure
\[ \begin{equation} \label{eq:poisson-pressure} \nabla^2 p_{NH} = \frac{\boldsymbol{\nabla} \boldsymbol{\cdot} \boldsymbol{v}^n}{\Delta t} + \boldsymbol{\nabla} \boldsymbol{\cdot} \boldsymbol{G}_{\boldsymbol{v}} \equiv \mathscr{F} \, , \end{equation}\]
along with homogenous Neumann boundary conditions $\boldsymbol{v} \cdot \boldsymbol{\hat{n}} = 0$ (Neumann on $p$ for wall-bounded directions and periodic otherwise) and where $\mathscr{F}$ denotes the source term for the Poisson equation.
For problems in which the hydrostatic approximation is invoked, the Poisson equation for pressure above only needs to be solved for the vertically integrated flow and the pressure field is a two dimensional term $p_S(x, y, t)$.
Direct method
Discretizing elliptic problems that can be solved via a classical separation-of-variables approach, such as Poisson's equation, results in a linear system of equations $M \boldsymbol{x} = \boldsymbol{y}$ where $M$ is a real symmetric matrix of block tridiagonal form. This allows for the matrix to be decomposed and solved efficiently, provided that the eigenvalues and eigenvectors of the blocks are known (§2) (Buzbee et al., 1970). In the case of Poisson's equation on a rectangle, Hockney (1965) has taken advantage of the fact that the fast Fourier transform can be used to perform the matrix multiplication steps resulting in an even more efficient method. Schumann and Sweet (1988) describe the implementation of such an algorithm for Poisson's equation on a staggered grid with Dirichlet, Neumann, and periodic boundary conditions.
The method can be explained easily by taking the Fourier transform of both sides of \eqref{eq:poisson-pressure} to yield
\[ \begin{equation} \label{eq:poisson-spectral} -(k_x^2 + k_y^2 + k_z^2) \widehat{p}_{NH} = \widehat{\mathscr{F}} \quad \implies \quad \widehat{p}_{NH} = - \frac{\widehat{\mathscr{F}}}{k_x^2 + k_y^2 + k_z^2} \, , \end{equation}\]
where $\widehat{\cdot}$ denotes the Fourier component. Here $k_x$, $k_y$, and $k_z$ are the wavenumbers. However, when solving the equation on a staggered grid we require a solution for $p_{NH}$ that is second-order accurate such that when when its Laplacian is computed, $\nabla^2 p_{NH}$ matches $\mathscr{F}$ to machine precision. This is crucial to ensure that the projection step in the fractional time-step works (see Time-stepping section and Fractional step method appendix). To do this, the wavenumbers are replaced by eigenvalues $\lambda^x$, $\lambda^y$, and $\lambda^z$ satisfying the discrete form of Poisson's equation with appropriate boundary conditions. Thus, Poisson's equation is diagonalized in Fourier space and the Fourier coefficients of the solution are easily solved for
\[\widehat{p}_{NH}(i, j, k) = - \frac{\widehat{\mathscr{F}}(i, j, k)}{\lambda^x_i + \lambda^y_j + \lambda^z_k} \, .\]
The eigenvalues are given by Schumann and Sweet (1988) and can also be tediously derived by plugging in the definition of the discrete Fourier transform into \eqref{eq:poisson-spectral}:
\[\begin{align} \lambda^x_i &= 4\frac{N_x^2}{L_x^2} \sin^2 \left [ \frac{(i-1) \pi}{N_x} \right ], \quad i=0, 1, \dots, N_x-1 \, , \\ \lambda^y_j &= 4\frac{N_y^2}{L_y^2} \sin^2 \left [ \frac{(j-1) \pi}{N_y} \right ], \quad j=0, 1, \dots, N_y-1 \, , \\ \lambda^z_k &= 4\frac{N_z^2}{L_z^2} \sin^2 \left [ \frac{(k-1) \pi}{2N_z} \right ], \quad k=0, 1, \dots, N_z-1 \, , \end{align}\]
where $\lambda^x$ and $\lambda^y$ correspond to periodic boundary conditions in the horizontal and $\lambda^z$ to Neumann boundary conditions in the vertical.
There is also an ambiguity in the solution to Poisson's equation as it's only defined up to a constant. To resolve this ambiguity we choose the solution with zero mean by setting the zeroth Fourier coefficient $p_{000}$ (corresponding to $k_x = k_y = k_z = 0$) to zero. This also has the added benefit of discarding the zero eigenvalue so we don't divide by it.
The Fast Fourier transforms are computed using FFTW.jl [(Frigo and Johnson, 1998) and (Frigo and Johnson, 2005)] on the CPU and using the cuFFT library on the GPU. Along wall-bounded dimensions, the cosine transform is used. In particular, as the transforms are performed on a staggered grid, DCT-II (REDFT10
) is used to perform the forward cosine transform and DCT-III (REDFT01
) is used to perform the inverse cosine transform.
Direct method with a vertically stretched grid
Using Fourier transforms for all three dimensions results in a method requiring $\mathcal{O}(N \log_2 N)$ operations where $N$ is the total number of grid points. This algorithm can be made even more efficient by solving a tridiagonal system along one of the dimensions and utilizing cyclic reduction. This results in the Fourier analysis cyclic reduction or $\text{FACR}(\ell)$ algorithm (with $\ell$ cyclic reduction steps) which requires only $\mathcal{O}(N \log_2\log_2 N)$ operations provided the optimal number of cyclic reduction steps is taken, which is $\ell = \log_2 \log_2 n$ where $n$ is the number of grid points in the cyclic reduction dimension. The FACR algorithm was first developed by Hockney (1969) and is well reviewed by Swarztrauber (1977) then further benchmarked and extended by Temperton (1979) and Temperton (1980).
Furthermore, the FACR algorithm removes the restriction that the grid is uniform in one of the dimensions so it can be utilized to implement a fast Poisson solver for vertically stretched grids if the cyclic reduction is applied in the along the vertical dimension.
Expanding $p_{NH}$ and $\mathscr{F}$ into Fourier modes along the $x$ and $y$ directions
\[p_{ijk} = \sum_{m=1}^{N_x} \sum_{n=1}^{N_y} \tilde{p}_{mnk} \; e^{-\mathrm{i} 2\pi i m / N_x} \; e^{-\mathrm{i} 2\pi j n / N_y} \, ,\]
and recalling that Fourier transforms do $\partial_x \rightarrow \mathrm{i} k_x$ and $\partial_y \rightarrow \mathrm{i} k_y$ we can write \eqref{eq:poisson-pressure} as
\[\sum_{m=1}^{N_x} \sum_{n=1}^{N_y} \left\lbrace \partial_z^2 \tilde{p}_{mnk} - (k_x^2 + k_y^2) \tilde{p}_{mnk} - \tilde{\mathscr{F}}_{mnk} \right\rbrace e^{-\mathrm{i} 2 \pi i m / N_x} e^{-\mathrm{i} 2 \pi j n / N_y} = 0 \, .\]
Discretizing the $\partial_z^2$ derivative and equating the term inside the brackets to zero we arrive at $N_x\times N_y$ symmetric tridiagonal systems of $N_z$ linear equations for the Fourier modes:
\[\frac{\tilde{p}_{mn, k-1}}{\Delta z^C_k} - \left\lbrace \frac{1}{\Delta z^C_k} + \frac{1}{\Delta z^C_{k+1}} + \Delta z^F_k (k_x^2 + k_y^2) \right\rbrace \tilde{p}_{mnk} + \frac{\tilde{p}_{mn, k+1}}{\Delta z^C_{k+1}} = \Delta z^F_k \tilde{\mathscr{F}}_{mnk} \, .\]
Cosine transforms on the GPU
Unfortunately cuFFT does not provide cosine transforms and so we must write our own fast cosine transforms for the GPU. We implemented the fast 1D and 2D cosine transforms described by Makhoul (1980) which compute it by applying the regular Fourier transform to a permuted version of the array.
In this section we will be using the DCT-II as the definition of the forward cosine transform for a real signal of length $N$
\[ \begin{equation} \label{eq:FCT} \text{DCT}(X): \quad Y_k = 2 \sum_{j=0}^{N-1} \cos \left[ \frac{\pi(j + \frac{1}{2})k}{N} \right] X_j \, , \end{equation}\]
and the DCT-III as the definition of the inverse cosine transform
\[ \begin{equation} \label{eq:IFCT} \text{IDCT}(X): \quad Y_k = X_0 + 2 \sum_{j=1}^{N-1} \cos \left[ \frac{\pi j (k + \frac{1}{2})}{N} \right] X_j \, , \end{equation} \]
and will use $\omega_M = e^{-2 \pi \mathrm{i} / M}$ to denote the $M^\text{th}$ root of unity, sometimes called the twiddle factors in the context of FFT algorithms.
1D fast cosine transform
To calculate \eqref{eq:FCT} using the fast Fourier transform, we first permute the input signal along the appropriate dimension by ordering the odd elements first followed by the even elements to produce a permuted signal
\[ X^\prime_n = \begin{cases} \displaystyle X_{2N}, \quad 0 \le n \le \left[ \frac{N-1}{2} \right] \, , \\ \displaystyle X_{2N - 2n - 1}, \quad \left[ \frac{N+1}{2} \right] \le n \le N-1 \, , \end{cases}\]
where $[a]$ indicates the integer part of $a$. This should produce, for example,
\[ \begin{equation} \label{eq:permutation} (a, b, c, d, e, f, g, h) \quad \rightarrow \quad (a, c, e, g, h, f, d, b) \, , \end{equation}\]
after which \eqref{eq:FCT} is computed using
\[ Y = \text{DCT}(X) = 2 \text{Re} \left\lbrace \omega_{4N}^k \text{FFT} \lbrace X^\prime \rbrace \right\rbrace \, .\]
1D fast inverse cosine transform
The inverse \eqref{eq:IFCT} can be computed using
\[ Y = \text{IDCT}(X) = \text{Re} \left\lbrace \omega_{4N}^{-k} \text{IFFT} \lbrace X \rbrace \right\rbrace \, ,\]
after which the inverse permutation of \eqref{eq:permutation} must be applied.
2D fast cosine transform
Unfortunately, the 1D algorithm cannot be applied dimension-wise so the 2D algorithm is more complicated. Thankfully, the permutation \eqref{eq:permutation} can be applied dimension-wise. The forward cosine transform for a real signal of length $N_1 \times N_2$ is then given by
\[Y_{k_1, k_2} = \text{DCT}(X_{n_1, n_2}) = 2 \text{Re} \left\lbrace \omega_{4N_1}^k \left( \omega_{4N_2}^k \tilde{X} + \omega_{4N_2}^{-k} \tilde{X}^- \right) \right\rbrace \, ,\]
where $\tilde{X} = \text{FFT}(X^\prime)$ and $\tilde{X}^-$ indicates that $\tilde{X}$ is indexed in reverse.
2D fast inverse cosine transform
The inverse can be computed using
\[Y_{k_1, k_2} = \text{IDCT}(X_{n_1, n_2}) = \frac{1}{4} \text{Re} \left\lbrace \omega_{4N_1}^{-k} \omega_{4N_2}^{-k} \left( \tilde{X} - M_1 M_2 \tilde{X}^{--} \right) - \mathrm{i} \left( M_1 \tilde{X}^{-+} + M_2 \tilde{X}^{+-} \right) \right\rbrace \, ,\]
where $\tilde{X} = \text{IFFT}(X)$ here, $\tilde{X}^{-+}$ is indexed in reverse along the first dimension, $\tilde{X}^{-+}$ along the second dimension, and $\tilde{X}^{--}$ along both. $M_1$ and $M_2$ are masks of lengths $N_1$ and $N_2$ respectively, both containing ones except at the first element where $M_0 = 0$. Afterwards, the inverse permutation of \eqref{eq:permutation} must be applied.
Due to the extra steps involved in calculating the cosine transform in 2D, running with two wall-bounded dimensions typically slows the model down by a factor of 2. Switching to the FACR algorithm may help here as a 2D cosine transform won't be necessary anymore.
Iterative Solvers
For problems with irregular grids the eigenvectors of the discrete Poisson operator are no longer simple Fourier series sines and cosines. This means discrete Fast Fourier Transforms can't be used to generate the projection of the equation right hand side onto eigenvectors. So an eigenvector based approach to solving the Poisson equation is not computationally efficient.
For problems with grids that are non uniform in multiple directions, we use instead a pre-conditioned conjugate gradient iterative solver. Such cases include curvilinear grids on the sphere and also telescoping cartesian grids that stretch along more than one dimension. There are two forms of the pressure operator in this approach. One is rigid lid form and one is an implicit free-surface form.
Rigid lid pressure operator
The rigid lid operator is based on the same continuous form as is used in the Direct Method solver.
Implicit free surface pressure operator
The implicit free surface solver solves for the free-surface, $\eta(x, y, t)$, in the vertically integrated continuity equation:
\[ \begin{equation} \label{eq:vertically-integrated-continuity} \partial_t \eta + \partial_x \left ( \int_{-H}^0 u \, \mathrm{d}z \right ) + \partial_y \left ( \int_{-H}^0 v \, \mathrm{d}z \right ) = M \, , \end{equation}\]
where $H(x, y)$ is the depth of the water column (to first order with respect to the free surface elevation) and $M$ is some surface volume flux (e.g., terms such as precipitation, evaporation and runoff); currently Oceananigans.jl assumes $M = 0$. Note that in deriving \eqref{eq:vertically-integrated-continuity}, we used the bottom boundary condition $w_{\rm bottom} = \boldsymbol{u}_{\rm bottom} \boldsymbol{\cdot} \boldsymbol{\nabla}_h H$.
To form a linear system that can be solved implicitly we recast the vertically-integrated continuity equation \eqref{eq:vertically-integrated-continuity} into a discrete integral form. The best way to do so is by starting from the discrete version of the continuity equation (in this case without any surface volume flux, $M = 0$)
\[ \begin{align} \label{eq:continuity-discrete} \delta_x (A_x u) + \delta_y (A_y v) + \delta_z (A_z w) = 0 \, , \end{align}\]
and summing it vertically to get:
\[ \begin{align} \label{eq:vertically-integrated-continuity-discrete} \delta_x \sum_k (A_x u) + \delta_y \sum_k (A_y v) + A_z \underbrace{w(k = N_z + 1)}_{w_{\rm top}} = 0 \, . \end{align}\]
In equations \eqref{eq:continuity-discrete} and \eqref{eq:vertically-integrated-continuity-discrete} and here after, we have abused notation and used, e.g., $u$ and $v$ to denote the volume averages over grid cells of the quantities $u$ and $v$ respectively. Using $w_{\rm top} = \partial_t \eta$ and being a bit more explicit on the locations the difference operators act on, \eqref{eq:vertically-integrated-continuity-discrete} becomes:
\[ \begin{equation} \label{eq:semi-discrete-integral-continuity} A_z \partial_t \eta + \delta_{x}^{caa} \sum_{k} (A_x u) + \delta_y^{aca} \sum_k (A_y v) = 0 \, . \end{equation}\]
We can now apply the velocity fractional step equation (discussed in the Time-stepping section) for the hydrostatic model:
\[ \begin{equation} \label{eq:hydrostatic-fractional-step} \boldsymbol{u}^{n+1} = \boldsymbol{u}^{\star} - g \Delta t \, \boldsymbol{\nabla} \eta^{n+1} \, . \end{equation}\]
We impose that the $n+1$-th time step velocity is consistent with \eqref{eq:semi-discrete-integral-continuity}
\[ \begin{equation} A_z \frac{\eta^{n+1} - \eta^{n}}{\Delta t} = - \delta_x^{caa} \sum_k (A_x u^{n+1}) - \delta_y^{aca} \sum_k (A_y v^{n+1}) \, . \end{equation}\]
Substituting $u^{n+1}$ and $v^{n+1}$ from the discrete form of the right-hand-side of \eqref{eq:hydrostatic-fractional-step} then gives us an implicit equation for $\eta^{n+1}$:
\[\begin{align} \delta_x^{caa}\sum_k A_x \partial_x^{faa} \eta^{n+1} & + \delta_y^{aca} \sum_k A_y \partial_y^{afa} \eta^{n+1} - \frac{1}{g \, \Delta t^2} A_z \eta^{n+1} = \nonumber \\ & = \frac{1}{g \, \Delta t} \left( \delta_x^{caa} \sum_k A_x u^{\star} + \delta_y^{aca} \sum_k A_y v^{\star} \right ) - \frac{1}{g \, \Delta t^{2}} A_z \eta^{n} \, . \label{eq:implicit-free-surface} \end{align}\]
The left-hand-side of \eqref{eq:implicit-free-surface} is nothing else than a linear operator acting on $\eta^{n+1}$. Formulated in this way, the linear operator is symmetric and therefore \eqref{eq:implicit-free-surface} can be solved using a preconditioned conjugate gradient algorithm.