from examples.cfd import plot_field, init_hat
import numpy as np
import sympy
%matplotlib inline
# Some variable declarations
= 101
nx = 101
ny = 80
nt = 1.
c = 2. / (nx - 1)
dx = 2. / (ny - 1)
dy = .2
sigma = sigma * dx dt
Example 2: Nonlinear convection in 2D
Following the initial convection tutorial with a single state variable \(u\), we will now look at non-linear convection (step 6 in the original). This brings one new crucial challenge: computing a pair of coupled equations and thus updating two time-dependent variables \(u\) and \(v\).
The full set of coupled equations is now
\[\begin{aligned} \frac{\partial u}{\partial t} + u \frac{\partial u}{\partial x} + v \frac{\partial u}{\partial y} = 0 \\ \\ \frac{\partial v}{\partial t} + u \frac{\partial v}{\partial x} + v \frac{\partial v}{\partial y} = 0\\ \end{aligned}\]and rearranging the discretized version gives us an expression for the update of both variables
\[\begin{aligned} u_{i,j}^{n+1} &= u_{i,j}^n - u_{i,j}^n \frac{\Delta t}{\Delta x} (u_{i,j}^n-u_{i-1,j}^n) - v_{i,j}^n \frac{\Delta t}{\Delta y} (u_{i,j}^n-u_{i,j-1}^n) \\ \\ v_{i,j}^{n+1} &= v_{i,j}^n - u_{i,j}^n \frac{\Delta t}{\Delta x} (v_{i,j}^n-v_{i-1,j}^n) - v_{i,j}^n \frac{\Delta t}{\Delta y} (v_{i,j}^n-v_{i,j-1}^n) \end{aligned}\]So, for starters we will re-create the original example run in pure NumPy array notation, before demonstrating the Devito version. Let’s start again with some utilities and parameters:
Let’s re-create the initial setup with a 2D “hat function”, but this time for two state variables.
#NBVAL_IGNORE_OUTPUT
# Allocate fields and assign initial conditions
= np.empty((nx, ny))
u = np.empty((nx, ny))
v
=u, dx=dx, dy=dy, value=2.)
init_hat(field=v, dx=dx, dy=dy, value=2.)
init_hat(field
plot_field(u)
Now we can create the two stencil expression for our two coupled equations according to the discretized equation above. We again use some simple Dirichlet boundary conditions to keep the values on all sides constant.
#NBVAL_IGNORE_OUTPUT
for n in range(nt + 1): ##loop across number of time steps
= u.copy()
un = v.copy()
vn 1:, 1:] = (un[1:, 1:] -
u[1:, 1:] * c * dt / dy * (un[1:, 1:] - un[1:, :-1])) -
(un[1:, 1:] * c * dt / dx * (un[1:, 1:] - un[:-1, 1:]))
vn[1:, 1:] = (vn[1:, 1:] -
v[1:, 1:] * c * dt / dy * (vn[1:, 1:] - vn[1:, :-1])) -
(un[1:, 1:] * c * dt / dx * (vn[1:, 1:] - vn[:-1, 1:]))
vn[
0, :] = 1
u[-1, :] = 1
u[0] = 1
u[:, -1] = 1
u[:,
0, :] = 1
v[-1, :] = 1
v[0] = 1
v[:, -1] = 1
v[:,
plot_field(u)
Excellent, we again get a wave that resembles the one from the oiginal examples.
Now we can set up our coupled problem in Devito. Let’s start by creating two initial state variables \(u\) and \(v\), as before, and initialising them with our “hat function.
#NBVAL_IGNORE_OUTPUT
from devito import Grid, TimeFunction
# First we need two time-dependent data fields, both initialized with the hat function
= Grid(shape=(nx, ny), extent=(2., 2.))
grid = TimeFunction(name='u', grid=grid)
u =u.data[0], dx=dx, dy=dy, value=2.)
init_hat(field
= TimeFunction(name='v', grid=grid)
v =v.data[0], dx=dx, dy=dy, value=2.)
init_hat(field
0]) plot_field(u.data[
Using the two TimeFunction
objects we can again derive our discretized equation, rearrange for the forward stencil point in time and define our variable update expression - only we have to do everything twice now! We again use forward differences for time via u.dt
and backward differences in space via u.dxl
and u.dyl
to match the original tutorial.
from devito import Eq, solve
= Eq(u.dt + u*u.dxl + v*u.dyl)
eq_u = Eq(v.dt + u*v.dxl + v*v.dyl)
eq_v
# We can use the same SymPy trick to generate two
# stencil expressions, one for each field update.
= solve(eq_u, u.forward)
stencil_u = solve(eq_v, v.forward)
stencil_v = Eq(u.forward, stencil_u, subdomain=grid.interior)
update_u = Eq(v.forward, stencil_v, subdomain=grid.interior)
update_v
print("U update:\n%s\n" % update_u)
print("V update:\n%s\n" % update_v)
U update:
Eq(u(t + dt, x, y), dt*(-u(t, x, y)*Derivative(u(t, x, y), x) - v(t, x, y)*Derivative(u(t, x, y), y) + u(t, x, y)/dt))
V update:
Eq(v(t + dt, x, y), dt*(-u(t, x, y)*Derivative(v(t, x, y), x) - v(t, x, y)*Derivative(v(t, x, y), y) + v(t, x, y)/dt))
We then set Dirichlet boundary conditions at all sides of the domain to \(1\).
= grid.dimensions
x, y = grid.stepping_dim
t = [Eq(u[t+1, 0, y], 1.)] # left
bc_u += [Eq(u[t+1, nx-1, y], 1.)] # right
bc_u += [Eq(u[t+1, x, ny-1], 1.)] # top
bc_u += [Eq(u[t+1, x, 0], 1.)] # bottom
bc_u = [Eq(v[t+1, 0, y], 1.)] # left
bc_v += [Eq(v[t+1, nx-1, y], 1.)] # right
bc_v += [Eq(v[t+1, x, ny-1], 1.)] # top
bc_v += [Eq(v[t+1, x, 0], 1.)] # bottom bc_v
And finally we can put it all together to build an operator and solve our coupled problem.
#NBVAL_IGNORE_OUTPUT
from devito import Operator
# Reset our data field and ICs
=u.data[0], dx=dx, dy=dy, value=2.)
init_hat(field=v.data[0], dx=dx, dy=dy, value=2.)
init_hat(field
= Operator([update_u, update_v] + bc_u + bc_v)
op =nt, dt=dt)
op(time
0]) plot_field(u.data[
Operator `Kernel` ran in 0.01 s
Excellent, we have now a scalar implementation of a convection problem, but this can be written as a single vectorial equation:
\(\frac{d U}{dt} + \nabla(U)U = 0\)
Let’s now use devito vectorial utilities and implement the vectorial equation
from devito import VectorTimeFunction, grad
= VectorTimeFunction(name='U', grid=grid)
U =U[0].data[0], dx=dx, dy=dy, value=2.)
init_hat(field=U[1].data[0], dx=dx, dy=dy, value=2.)
init_hat(field
1].data[0])
plot_field(U[
= Eq(U.dt + grad(U)*U) eq_u
We now have a vectorial equation. Unlike in the previous case, we do not need to play with left/right derivatives as the automated staggering of the vectorial function takes care of this.
eq_u
\(\displaystyle \left[\begin{matrix}U_x(t, x + h_x/2, y) \frac{\partial}{\partial x} U_x(t, x + h_x/2, y) + U_y(t, x, y + h_y/2) \frac{\partial}{\partial y} U_x(t, x + h_x/2, y) + \frac{\partial}{\partial t} U_x(t, x + h_x/2, y)\\U_x(t, x + h_x/2, y) \frac{\partial}{\partial x} U_y(t, x, y + h_y/2) + U_y(t, x, y + h_y/2) \frac{\partial}{\partial y} U_y(t, x, y + h_y/2) + \frac{\partial}{\partial t} U_y(t, x, y + h_y/2)\end{matrix}\right] = 0\)
Then we set the nboundary conditions
= grid.dimensions
x, y = grid.stepping_dim
t = [Eq(U[0][t+1, 0, y], 1.)] # left
bc_u += [Eq(U[0][t+1, nx-1, y], 1.)] # right
bc_u += [Eq(U[0][t+1, x, ny-1], 1.)] # top
bc_u += [Eq(U[0][t+1, x, 0], 1.)] # bottom
bc_u = [Eq(U[1][t+1, 0, y], 1.)] # left
bc_v += [Eq(U[1][t+1, nx-1, y], 1.)] # right
bc_v += [Eq(U[1][t+1, x, ny-1], 1.)] # top
bc_v += [Eq(U[1][t+1, x, 0], 1.)] # bottom bc_v
# We can use the same SymPy trick to generate two
# stencil expressions, one for each field update.
= solve(eq_u, U.forward)
stencil_U = Eq(U.forward, stencil_U, subdomain=grid.interior) update_U
And we have the updated (stencil) as a vectorial equation once again
update_U
\(\displaystyle \left[\begin{matrix}U_x(t + dt, x + h_x/2, y)\\U_y(t + dt, x, y + h_y/2)\end{matrix}\right] = \left[\begin{matrix}dt \left(- U_x(t, x + h_x/2, y) \frac{\partial}{\partial x} U_x(t, x + h_x/2, y) - U_y(t, x, y + h_y/2) \frac{\partial}{\partial y} U_x(t, x + h_x/2, y) + \frac{U_x(t, x + h_x/2, y)}{dt}\right)\\dt \left(- U_x(t, x + h_x/2, y) \frac{\partial}{\partial x} U_y(t, x, y + h_y/2) - U_y(t, x, y + h_y/2) \frac{\partial}{\partial y} U_y(t, x, y + h_y/2) + \frac{U_y(t, x, y + h_y/2)}{dt}\right)\end{matrix}\right]\)
We finally run the operator
#NBVAL_IGNORE_OUTPUT
= Operator([update_U] + bc_u + bc_v)
op =nt, dt=dt)
op(time
# The result is indeed the expected one.
0].data[0]) plot_field(U[
Operator `Kernel` ran in 0.01 s
from devito import norm
assert np.isclose(norm(u), norm(U[0]), rtol=1e-5, atol=0)