from devito import Eq, TimeFunction, sqrt, Function, Operator, Grid, solve, ConditionalDimension
from matplotlib import pyplot as plt
import numpy as np
def ForwardOperator(etasave, eta, M, N, h, D, g, alpha, grid):
"""
Operator that solves the equations expressed above.
It computes and returns the discharge fluxes M, N and wave height eta from
the 2D Shallow water equation using the FTCS finite difference method.
Parameters
----------
eta : TimeFunction
The wave height field as a 2D array of floats.
M : TimeFunction
The discharge flux field in x-direction as a 2D array of floats.
N : TimeFunction
The discharge flux field in y-direction as a 2D array of floats.
h : Function
Bathymetry model as a 2D array of floats.
D : Function
Total thickness of the water column.
g : float
gravity acceleration.
alpha : float
Manning's roughness coefficient.
etasave : TimeFunction
Function that is sampled in a different interval than the normal propagation
and is responsible for saving the snapshots required for the following
animations.
"""
= np.finfo(grid.dtype).eps
eps
# Friction term expresses the loss of amplitude from the friction with the seafloor
= g * alpha**2 * sqrt(M**2 + N**2) / D**(7./3.)
frictionTerm
# System of equations
= Eq(eta.dt + M.dxc + N.dyc)
pde_eta = Eq(M.dt + (M**2/D).dxc + (M*N/D).dyc + g*D*eta.forward.dxc + frictionTerm*M)
pde_M = Eq(N.dt + (M.forward*N/D).dxc + (N**2/D).dyc + g*D*eta.forward.dyc + g * alpha**2 * sqrt(M.forward**2 + N**2) / D**(7./3.)*N)
pde_N
= solve(pde_eta, eta.forward)
stencil_eta = solve(pde_M, M.forward)
stencil_M = solve(pde_N, N.forward)
stencil_N
# Equations with the forward in time term isolated
= Eq(eta.forward, stencil_eta, subdomain=grid.interior)
update_eta = Eq(M.forward, stencil_M, subdomain=grid.interior)
update_M = Eq(N.forward, stencil_N, subdomain=grid.interior)
update_N = Eq(D, eta.forward + h)
eq_D
return Operator([update_eta, update_M, update_N, eq_D] + [Eq(etasave, eta)])
2D Shallow Water Equations 🌊
This notebook is a Devito reimplementation from this notebook. We will use an approximation to the Navier-Stokes equations - the 2D Shallow Water equations - in order to model the propagation of Tsunami events.
I’ve only made minor modifications to the text, in order to make it more compatible to the new coding format.
Author: Átila Saraiva Q. Soares.
Governing Equations
Starting from the continuity and momentum conservation equations, we want to model the following problem:
For a given bathymetry model, which can include a complex seafloor topography, we want to model the amplitudes, speed and interaction of waves at the seasurface. At a given point \((x,\; y)\), the thickness of the water column between the seafloor and undisturbed water surface is defined by the variable \(h\), while the wave amplitude is \(\eta\) and therefore the whole thickness of the water column \(D = h + \eta\).
Using appropriate boundary conditions at the water surface/seafloor, assuming that the horizontal wavelength of the modelled waves are much larger than the water depth and integrating the conservation of mass and momentum equations over the water column, we can derive the following equations to decribe wave propagation
\[\begin{equation} \begin{split} \frac{\partial \eta}{\partial t} &+ \frac{\partial M}{\partial x} + \frac{\partial N}{\partial y} = 0\\ \frac{\partial M}{\partial t} &+ \frac{\partial}{\partial x} \biggl(\frac{M^2}{D}\biggr) + \frac{\partial}{\partial y} \biggl(\frac{MN}{D}\biggr) + g D \frac{\partial \eta}{\partial x} + \frac{g \alpha^2}{D^{7/3}} M \sqrt{M^2+N^2} = 0\\ \frac{\partial N}{\partial t} &+ \frac{\partial}{\partial x} \biggl(\frac{MN}{D}\biggr) + \frac{\partial}{\partial y} \biggl(\frac{N^2}{D}\biggr) + g D \frac{\partial \eta}{\partial y} + \frac{g \alpha^2}{D^{7/3}} N \sqrt{M^2+N^2} = 0 \end{split} \tag{1} \end{equation}\]
known as 2D Shallow Water Equations (SWE). The derivation of these equations is beyond the scope of this notebook. Therefore, I refer to the Tsunami Modelling Handbook and the lecture Shallow Water Derivation and Applications by Christian Kühbacher for further details.
In Eq. (1) the discharge fluxes \(M,\; N\) in x- and y-direction, respectively are given by
\[\begin{equation} \begin{split} M & = \int_{-h}^\eta u dz = u(h+\eta) = uD\\ N & = \int_{-h}^\eta v dz = v(h+\eta) = vD\\ \end{split} \tag{2} \end{equation}\]
with the horizontal velocity components \(u,\;v\) in x- and y-direction, while \(g\) denotes the gravity acceleration. The terms \(\frac{g \alpha^2}{D^{7/3}} M \sqrt{M^2+N^2}\) and \(\frac{g \alpha^2}{D^{7/3}} N \sqrt{M^2+N^2}\) describe the influence of seafloor friction on the wave amplitude. \(\alpha\) denotes the Manning’s roughness which can be as small as 0.01 for neat cement or smooth metal up to 0.06 for very poor natural channels (see Tsunami modelling handbook).
The Shallow Water Equations can be applied to
- Tsunami prediction
- Atmospheric flow
- Storm surges
- Flows around structures
- Planetary flows
and easily extended to incorporate the effects of Coriolis forces, tides or wind stress.
Operator implementation
This is a simple function which returns the operator that solves this equation. The important part is the operator, which contains all the equations expressed above.
The following function transforms the saved wavefield snapshots and transform them into a video compatible with jupyter notebook.
from IPython.display import HTML, display
import matplotlib.pyplot as plt
import matplotlib.animation as animation
def snaps2video (eta, title):
= plt.subplots()
fig, ax = ax.imshow(eta.data[0, :, :].T, vmin=-1, vmax=1, cmap="seismic")
matrice
plt.colorbar(matrice)
'x')
plt.xlabel('z')
plt.ylabel(
plt.title(title)
def update(i):
matrice.set_array(eta.data[i, :, :].T)return matrice,
# Animation
= animation.FuncAnimation(fig, update, frames=nsnaps, interval=50, blit=True)
ani
plt.close(ani._fig) display(HTML(ani.to_html5_video()))
def plotDepthProfile (h, title):
= plt.subplots()
fig, ax = ax.imshow(h0)
matrice
plt.colorbar(matrice)'x')
plt.xlabel('z')
plt.ylabel(
plt.title(title) plt.show()
Example I: Tsunami in ocean with constant depth
After writing the Shallow_water_2D
code and all the required functions attached to it, we can define and run our first 2D Tsunami modelling run.
Let’s assume that the ocean model is $ L_x = 100; m$ in x-direction and \(L_y = 100\; m\) in y-direction. The model is discretized with \(nx=401\) gridpoints in x-direction and \(ny=401\) gridpoints in y-direction, respectively.
In this first modelling run, we assume a constant bathymetry \(h=50\;m\). The initial wave height field \(\eta_0\) is defined as a Gaussian at the center of the model, with a half-width of 10 m and an amplitude of 0.5 m. Regarding the initial discharge fluxes, we assume that
\[\begin{equation} \begin{split} M_0(x,y) &= 100 \eta_0(x,y)\\ N_0(x,y) &= 0\\ \end{split}\notag \end{equation}\]
In order to avoid the occurence of high frequency artifacts, when waves are interacting with the boundaries, boundary conditions will be avoided. Normally Dirichlet conditions are used for the M and N discharge fluxes, and Neumann for the wave height field. Those lead to significant boundary reflections which might be not realistic for a given problem.
Let’s assume the gravity \(g = 9.81\) and the Manning’s roughness coefficient \(\alpha = 0.025\) for the all the remaining examples.
= 100.0 # width of the mantle in the x direction []
Lx = 100.0 # thickness of the mantle in the y direction []
Ly = 401 # number of points in the x direction
nx = 401 # number of points in the y direction
ny = Lx / (nx - 1) # grid spacing in the x direction []
dx = Ly / (ny - 1) # grid spacing in the y direction []
dy = 9.81 # gravity acceleration [m/s^2]
g = 0.025 # friction coefficient for natural channels in good condition
alpha
# Maximum wave propagation time [s]
= 3.
Tmax = 1/4500.
dt = (int)(Tmax/dt)
nt print(dt, nt)
= np.linspace(0.0, Lx, num=nx)
x = np.linspace(0.0, Ly, num=ny)
y
# Define initial eta, M, N
= np.meshgrid(x,y) # coordinates X,Y required to define eta, h, M, N
X, Y
# Define constant ocean depth profile h = 50 m
= 50. * np.ones_like(X)
h0
# Define initial eta Gaussian distribution [m]
= 0.5 * np.exp(-((X-50)**2/10)-((Y-50)**2/10))
eta0
# Define initial M and N
= 100. * eta0
M0 = 0. * M0
N0 = eta0 + 50.
D0
= Grid(shape=(ny, nx), extent=(Ly, Lx), dtype=np.float32) grid
0.00022222222222222223 13500
# NBVAL_IGNORE_OUTPUT
= 400
nsnaps
# Defining symbolic functions
= TimeFunction(name='eta', grid=grid, space_order=2)
eta = TimeFunction(name='M', grid=grid, space_order=2)
M = TimeFunction(name='N', grid=grid, space_order=2)
N = Function(name='h', grid=grid)
h = Function(name='D', grid=grid)
D
# Inserting initial conditions
0] = eta0.copy()
eta.data[0] = M0.copy()
M.data[0] = N0.copy()
N.data[= eta0 + h0
D.data[:] = h0.copy()
h.data[:]
# Setting up function to save the snapshots
= round(nt / nsnaps)
factor print(f"The nt/nsnaps factor is {factor}")
= ConditionalDimension('t_sub', parent=grid.time_dim, factor=factor)
time_subsampled = TimeFunction(name='etasave', grid=grid, space_order=2,
etasave =nsnaps, time_dim=time_subsampled)
save
# Compile the operator
= ForwardOperator(etasave, eta, M, N, h, D, g, alpha, grid)
op
# Use the operator
apply(eta=eta, etasave=etasave, M=M, N=N, D=D, h=h, time=nt-2, dt=dt) op.
The nt/nsnaps factor is 34
Operator `Kernel` ran in 17.97 s
PerformanceSummary([(PerfKey(name='section0', rank=None),
PerfEntry(time=17.90456999999999, gflopss=0.0, gpointss=0.0, oi=0.0, ops=0, itershapes=[])),
(PerfKey(name='section1', rank=None),
PerfEntry(time=0.061892000000000044, gflopss=0.0, gpointss=0.0, oi=0.0, ops=0, itershapes=[]))])
# NBVAL_SKIP
"Modeling a tsunami in ocean with constant depth") snaps2video(etasave,
Let’s take a look on what the code looks like
# To look at the code, uncomment the line below
#print(op.ccode)
Example II: Two Tsunamis in ocean with constant depth
In example II, we will model the propagation and interaction of two Tsunamis, where the initial conditions for the wave height field \(\eta_0\) consists of Gaussian functions with opposite sign located at \((x_1,\;y_1) = (35\; m,\; 35\; m)\) and \((x_2,\;y_2) = (65\; m,\; 65\; m)\). All other modelling parameters remain the same:
# Define constant ocean depth profile h = 50 m
= 50 * np.ones_like(X)
h0
# Define initial Gaussian eta distribution [m]
= 0.5 * np.exp(-((X-35)**2/10)-((Y-35)**2/10)) # first Tsunami source
eta0 -= 0.5 * np.exp(-((X-65)**2/10)-((Y-65)**2/10)) # add second Tsunami source
eta0
# Define initial M and N
= 100. * eta0
M0 = 0. * M0
N0 = eta0 + h0
D0
# Maximum wave propagation time [s]
= 3.5
Tmax = 1/8000.
dt = (int)(Tmax/dt) nt
# NBVAL_IGNORE_OUTPUT
# Inserting initial conditions
0] = eta0.copy()
eta.data[0] = M0.copy()
M.data[0] = N0.copy()
N.data[= eta0 + h0
D.data[:] = h0.copy()
h.data[:]
# Setting up function to save the snapshots
= round(nt / nsnaps)
factor print(f"The nt/nsnaps factor is {factor}")
= ConditionalDimension('t_sub', parent=grid.time_dim, factor=factor)
time_subsampled = TimeFunction(name='etasave', grid=grid, space_order=2,
etasave =nsnaps, time_dim=time_subsampled)
save
# Compile the operator
= ForwardOperator(etasave, eta, M, N, h, D, g, alpha, grid)
op
# Use the operator. No need to recompile since the etasave factor is smaller
# than before
=eta, etasave=etasave, M=M, N=N, D=D, h=h, time=nt-2, dt=dt) op(eta
The nt/nsnaps factor is 70
Operator `Kernel` ran in 37.93 s
PerformanceSummary([(PerfKey(name='section0', rank=None),
PerfEntry(time=37.860470999999926, gflopss=0.0, gpointss=0.0, oi=0.0, ops=0, itershapes=[])),
(PerfKey(name='section1', rank=None),
PerfEntry(time=0.06278300000000006, gflopss=0.0, gpointss=0.0, oi=0.0, ops=0, itershapes=[]))])
# NBVAL_SKIP
"Modeling two tsunamis in ocean with constant depth") snaps2video(etasave,
Example III: Tsunami in an ocean with 1D \(\tanh\) depth variation
So far, so good, we can achieve stable and accurate modelling results. However, a constant bathymetry model is a little too simple. In this example, we assume that the bathymetry decreases with a \(\tanh\) function from the left to the right boundary in x-direction. Let ’s place a Tsunami source at \((x_1,\;y_1) = (30\; m,\; 50\; m)\) and see what ’s happening:
# Define depth profile h [m]
= 50 - 45 * np.tanh((X-70.)/8.)
h0
# Define initial eta [m]
= 0.5 * np.exp(-((X-30)**2/10)-((Y-50)**2/20))
eta0
# Define initial M and N
= 100. * eta0
M0 = 0. * M0
N0 = eta0 + h0
D0
# Maximum wave propagation time [s]
= 2.
Tmax = 1/4000.
dt = (int)(Tmax/dt) nt
Let’s take a look into how this depth profile \(h\) looks in the plot below.
# NBVAL_IGNORE_OUTPUT
"Tanh Depth Profile") plotDepthProfile(h,
# NBVAL_IGNORE_OUTPUT
# Inserting initial conditions
0] = eta0.copy()
eta.data[0] = M0.copy()
M.data[0] = N0.copy()
N.data[= eta0 + h0
D.data[:] = h0.copy()
h.data[:]
# Setting up function to save the snapshots
= round(nt / nsnaps)
factor print(f"The nt/nsnaps factor is {factor}")
= ConditionalDimension('t_sub', parent=grid.time_dim, factor=factor)
time_subsampled = TimeFunction(name='etasave', grid=grid, space_order=2,
etasave =nsnaps, time_dim=time_subsampled)
save
# Compile the operator
= ForwardOperator(etasave, eta, M, N, h, D, g, alpha, grid)
op
# Use the operator
=eta, etasave=etasave, M=M, N=N, D=D, h=h, time=nt-2, dt=dt) op(eta
The nt/nsnaps factor is 20
Operator `Kernel` ran in 10.70 s
PerformanceSummary([(PerfKey(name='section0', rank=None),
PerfEntry(time=10.634207999999974, gflopss=0.0, gpointss=0.0, oi=0.0, ops=0, itershapes=[])),
(PerfKey(name='section1', rank=None),
PerfEntry(time=0.06243400000000002, gflopss=0.0, gpointss=0.0, oi=0.0, ops=0, itershapes=[]))])
# NBVAL_SKIP
"Modeling a tsunami in an ocean with 1D Tanh depth variation") snaps2video(etasave,
Example IV: Tsunami in an ocean with a seamount
What happens if we have a constant bathymetry, except for a small seamount reaching 5 m below the water surface? Lets define the model parameters:
# Define constant ocean depth profile h = 50 m
= 50. * np.ones_like(X)
h0
# Adding seamount to seafloor topography
-= 45. * np.exp(-((X-50)**2/20)-((Y-50)**2/20))
h0
# Define initial eta [m]
= 0.5 * np.exp(-((X-30)**2/5)-((Y-50)**2/5))
eta0
# Define initial M and N
= 100. * eta0
M0 = 0. * M0
N0 = eta0 + h0
D0
# Maximum wave propagation time [s]
= 2.
Tmax = 1/8000.
dt = (int)(Tmax/dt) nt
Let’s take a look into how this depth profile \(h\) looks in the plot below.
# NBVAL_IGNORE_OUTPUT
"Seamount Depth Profile") plotDepthProfile(h,
# NBVAL_IGNORE_OUTPUT
# Inserting initial conditions
0] = eta0.copy()
eta.data[0] = M0.copy()
M.data[0] = N0.copy()
N.data[= eta0 + h0
D.data[:] = h0.copy()
h.data[:]
# Setting up function to save the snapshots
= round(nt / nsnaps)
factor print(f"The nt/nsnaps factor is {factor}")
= ConditionalDimension('t_sub', parent=grid.time_dim, factor=factor)
time_subsampled = TimeFunction(name='etasave', grid=grid, space_order=2,
etasave =nsnaps, time_dim=time_subsampled)
save
# Compile the operator
= ForwardOperator(etasave, eta, M, N, h, D, g, alpha, grid)
op
# Use the operator
=eta, etasave=etasave, M=M, N=N, D=D, h=h, time=nt-2, dt=dt) op(eta
The nt/nsnaps factor is 40
Operator `Kernel` ran in 21.54 s
PerformanceSummary([(PerfKey(name='section0', rank=None),
PerfEntry(time=21.467746999999985, gflopss=0.0, gpointss=0.0, oi=0.0, ops=0, itershapes=[])),
(PerfKey(name='section1', rank=None),
PerfEntry(time=0.06220900000000002, gflopss=0.0, gpointss=0.0, oi=0.0, ops=0, itershapes=[]))])
# NBVAL_SKIP
"Modeling a tsunami in an ocean with a seamount") snaps2video(etasave,
Example V: Tsunami in an ocean with random seafloor topography variations
Another modelling idea: what is the influence of the roughness of the seafloor topography. First, we add some random perturbations on the constant bathymetry model \(h_0\) by using the random.rand
function from the NumPy
library. To smooth the random perturbations, we apply the gaussian_filter
from the SciPy
library:
from scipy.ndimage import gaussian_filter
# Define constant ocean depth profile h = 30 m
= 30. * np.ones_like(X)
h0
# Add random seafloor perturbation of +- 5m
= 5. # perturbation amplitude
pert
102034)
np.random.seed(= 2.0 * (np.random.rand(ny, nx) - 0.5) * pert # create random number perturbations
r = gaussian_filter(r, sigma=16) # smooth random number perturbation
r = h0 * (1 + r) # add perturbations to constant seafloor
h0
# Define initial eta [m]
= 0.2 * np.exp(-((X-30)**2/5)-((Y-50)**2/5))
eta0
# Define initial M and N
= 100. * eta0
M0 = 0. * M0
N0 = eta0 + h0
D0
# Maximum wave propagation time [s]
= 3.
Tmax = 1/4000.
dt = (int)(Tmax/dt) nt
Let’s take a look into how this depth profile \(h\) looks in the plot below.
# NBVAL_IGNORE_OUTPUT
"Random seafloor Depth Profile") plotDepthProfile(h,
# NBVAL_IGNORE_OUTPUT
# Inserting initial conditions
0] = eta0.copy()
eta.data[0] = M0.copy()
M.data[0] = N0.copy()
N.data[= eta0 + h0
D.data[:] = h0.copy()
h.data[:]
# Setting up function to save the snapshots
= round(nt / nsnaps)
factor print(f"The nt/nsnaps factor is {factor}")
= ConditionalDimension('t_sub', parent=grid.time_dim, factor=factor)
time_subsampled = TimeFunction(name='etasave', grid=grid, space_order=2,
etasave =nsnaps, time_dim=time_subsampled)
save
# Compile the operator again
= ForwardOperator(etasave, eta, M, N, h, D, g, alpha, grid)
op
# Use the operator
=eta, etasave=etasave, M=M, N=N, D=D, h=h, time=nt-2, dt=dt) op(eta
The nt/nsnaps factor is 30
Operator `Kernel` ran in 16.40 s
PerformanceSummary([(PerfKey(name='section0', rank=None),
PerfEntry(time=16.326802999999938, gflopss=0.0, gpointss=0.0, oi=0.0, ops=0, itershapes=[])),
(PerfKey(name='section1', rank=None),
PerfEntry(time=0.06265700000000003, gflopss=0.0, gpointss=0.0, oi=0.0, ops=0, itershapes=[]))])
# NBVAL_SKIP
"Modeling a tsunami in an ocean with random seafloor topography variations") snaps2video(etasave,
Example VI: 2D circular dam break problem
As a final modelling example, let’s take a look at an (academic) engineering problem: a tsunami induced by the collapse of a circular dam in a lake with a constant bathymetry of 30 m. We only need to set the wave height in a circle with radius \(r_0 = 5\; m\) to \(\eta_0 = 0.5 \; m\) and to zero everywhere else. To avoid the occurence of high frequency artifacts in the wavefield, known as numerical grid dispersion, we apply a Gaussian filter to the initial wave height. To achieve a symmetric dam collapse, the initial discharge fluxes \(M_0,N_0\) are set to equal values.
# Define constant ocean depth profile h = 30 m
= 30. * np.ones_like(X)
h0
# Define initial eta [m]
= np.zeros_like(X)
eta0
# Define mask for circular dam location with radius r0
= 5.
r0 = np.where(np.sqrt((X-50)**2 + (Y-50)**2) <= r0)
mask
# Set wave height in dam to 0.5 m
= 0.5
eta0[mask]
# Smooth dam boundaries with gaussian filter
= gaussian_filter(eta0, sigma=8) # smooth random number perturbation
eta0
# Define initial M and N
= 1. * eta0
M0 = 1. * M0
N0 = eta0 + h0
D0
# Maximum wave propagation time [s]
= 3.
Tmax = 1/4000.
dt = (int)(Tmax/dt) nt
# NBVAL_IGNORE_OUTPUT
# Inserting initial conditions
0] = eta0.copy()
eta.data[0] = M0.copy()
M.data[0] = N0.copy()
N.data[= eta0 + h0
D.data[:] = h0.copy()
h.data[:]
# Setting up function to save the snapshots
= round(nt / nsnaps)
factor print(f"The nt/nsnaps factor is {factor}")
= ConditionalDimension('t_sub', parent=grid.time_dim, factor=factor)
time_subsampled = TimeFunction(name='etasave', grid=grid, space_order=2,
etasave =nsnaps, time_dim=time_subsampled)
save
# Compile the operator again
= ForwardOperator(etasave, eta, M, N, h, D, g, alpha, grid)
op
# Use the operator
=eta, etasave=etasave, M=M, N=N, D=D, h=h, time=nt-2, dt=dt) op(eta
The nt/nsnaps factor is 30
Operator `Kernel` ran in 16.72 s
PerformanceSummary([(PerfKey(name='section0', rank=None),
PerfEntry(time=16.65377599999995, gflopss=0.0, gpointss=0.0, oi=0.0, ops=0, itershapes=[])),
(PerfKey(name='section1', rank=None),
PerfEntry(time=0.06264100000000003, gflopss=0.0, gpointss=0.0, oi=0.0, ops=0, itershapes=[]))])
# NBVAL_SKIP
"Modeling a 2D circular dam break problem") snaps2video(etasave,
What we learned:
How to solve the 2D Shallow Water Equation using the FTCS finite difference scheme.
Propagation of (multiple) Tsunamis in an ocean with constant bathymetry.
Tsunamis reaching shallow waters near a coast will slow down, their wavelength decrease, while their wave height increases.
The influence of a seamount and roughness of the seafloor topoghraphy on the Tsunami wavefield.
Tsunami modelling for engineering problems, like the collapse of dams.