Mantle convection modelling =========================== In all demos that we have considered so far, the equations that we have solved all involve a time derivative. Those are clearly *time-dependent* equations. However, time-dependent equations need not involve a time derivative. For example, they might include fields that vary with time. Examples of where this might happen are in continuous pressure projection approaches, ice sheet and glaciological modelling, and mantle convection modelling. In this demo, we illustrate how Goalie can be used to solve such problems. We consider the problem of a mantle convection in a 2D unit square domain. The governing equations and Firedrake implementation are based on the 2D Cartesian incompressible isoviscous case from :cite:`Davies:2022`. We refer the reader to the paper for a detailed description of the problem and implementation. Here we present the governing equations involving a Stokes system and an energy equation, which we solve for the velocity :math:`\mathbf{u}`, pressure :math:`p`, and temperature :math:`T`: .. math:: \begin{align} \nabla \cdot \mu \left[\nabla \mathbf{u} + (\nabla \mathbf{u})^T \right] - \nabla p + \mathrm{Ra}\,T\,\mathbf{k} &= 0, \\ \frac{\partial T}{\partial t} \cdot \mathbf{u}\cdot\nabla T - \nabla \cdot (\kappa\nabla T) &= 0, \end{align} where :math:`\mu`, :math:`\kappa`, and :math:`\mathrm{Ra}` are constant viscosity, thermal diffusivity, and Rayleigh number, respectively, and :math:`\mathbf{k}` is the unit vector in the direction opposite to gravity. Note that while the Stokes equations do not involve a time derivative, they are still time-dependent as they depend on the temperature field :math:`T`, which is time-dependent. As always, we begin by importing Firedrake and Goalie, and defining constants. :: from firedrake import * from goalie import * Ra, mu, kappa = Constant(1e4), Constant(1.0), Constant(1.0) k = Constant((0, 1)) The problem is solved simultaneously for the velocity :math:`\mathbf{u}` and pressure :math:`p` using a *mixed* formulation, which was introduced in a `previous demo on advection-diffusion reaction <./gray_scott.py.html>`__. :: fields = ["up", "T"] def get_function_spaces(mesh): V = VectorFunctionSpace(mesh, "CG", 2, name="velocity") W = FunctionSpace(mesh, "CG", 1, name="pressure") Z = MixedFunctionSpace([V, W], name="velocity-pressure") Q = FunctionSpace(mesh, "CG", 1, name="temperature") return {"up": Z, "T": Q} We must set initial conditions to solve the problem. Note that we define the initial condition for the mixed field ``"up"`` despite the equations not involving a time derivative. In this case, the prescribed initial condition should be understood as the *initial guess* for the solver. :: def get_initial_condition(mesh_seq): x, y = SpatialCoordinate(mesh_seq[0]) T_init = Function(mesh_seq.function_spaces["T"][0]) T_init.interpolate(1.0 - y + 0.05 * cos(pi * x) * sin(pi * y)) up_init = Function(mesh_seq.function_spaces["up"][0]) # Zero by default return {"up": up_init, "T": T_init} In the solver, weak forms for the Stokes and energy equations are defined as follows. Note that the mixed field ``"up"`` does not have a lagged term. :: def get_solver(mesh_seq): def solver(index): Z = mesh_seq.function_spaces["up"][index] Q = mesh_seq.function_spaces["T"][index] up = mesh_seq.fields["up"] u, p = split(up) T, T_ = mesh_seq.fields["T"] # Crank-Nicolson time discretisation for temperature Ttheta = 0.5 * (T + T_) # Variational problem for the Stokes equations v, w = TestFunctions(mesh_seq.function_spaces["up"][index]) stress = 2 * mu * sym(grad(u)) F_up = ( inner(grad(v), stress) * dx - div(v) * p * dx - (dot(v, k) * Ra * Ttheta) * dx ) F_up += -w * div(u) * dx # Continuity equation # Variational problem for the energy equation q = TestFunction(mesh_seq.function_spaces["T"][index]) F_T = ( q * (T - T_) / dt * dx + q * dot(u, grad(Ttheta)) * dx + dot(grad(q), kappa * grad(Ttheta)) * dx ) # Boundary IDs left, right, bottom, top = 1, 2, 3, 4 # Boundary conditions for velocity bcux = DirichletBC(Z.sub(0).sub(0), 0, sub_domain=(left, right)) bcuy = DirichletBC(Z.sub(0).sub(1), 0, sub_domain=(bottom, top)) bcs_up = [bcux, bcuy] # Boundary conditions for temperature bctb = DirichletBC(Q, 1.0, sub_domain=bottom) bctt = DirichletBC(Q, 0.0, sub_domain=top) bcs_T = [bctb, bctt] # Solver parameters dictionary solver_parameters = { "mat_type": "aij", "snes_type": "ksponly", "ksp_type": "preonly", "pc_type": "lu", "pc_factor_mat_solver_type": "mumps", } nlvp_up = NonlinearVariationalProblem(F_up, up, bcs=bcs_up) nlvs_up = NonlinearVariationalSolver( nlvp_up, solver_parameters=solver_parameters, ad_block_tag="up", ) nlvp_T = NonlinearVariationalProblem(F_T, T, bcs=bcs_T) nlvs_T = NonlinearVariationalSolver( nlvp_T, solver_parameters=solver_parameters, ad_block_tag="T", ) # Time integrate over the subinterval num_timesteps = mesh_seq.time_partition.num_timesteps_per_subinterval[index] for _ in range(num_timesteps): nlvs_up.solve() nlvs_T.solve() yield T_.assign(T) return solver We can now create a MeshSeq object and solve the forward problem. For demonstration purposes, we only consider two subintervals and a low number of timesteps. :: num_subintervals = 2 meshes = [UnitSquareMesh(32, 32) for _ in range(num_subintervals)] dt = 1e-3 num_timesteps = 40 end_time = dt * num_timesteps dt_per_export = [10 for _ in range(num_subintervals)] To account for the lack of time derivative in the Stokes equations, we use the ``field_types`` argument of the ``TimePartition`` object to specify that the ``"up"`` field is *steady* (i.e. without a time derivative) and that the ``T`` field is *unsteady* (i.e. involves a time derivative). The order in ``field_types`` must match the order of the fields in the ``fields`` list above. :: time_partition = TimePartition( end_time, num_subintervals, dt, fields, num_timesteps_per_export=dt_per_export, field_types=["steady", "unsteady"], ) mesh_seq = MeshSeq( time_partition, meshes, get_function_spaces=get_function_spaces, get_initial_condition=get_initial_condition, get_solver=get_solver, transfer_method="interpolate", ) solutions = mesh_seq.solve_forward() We can plot the temperature fields for exported timesteps using the in-built plotting function ``plot_snapshots``. :: fig, axes, tcs = plot_snapshots(solutions, time_partition, "T", "forward", levels=25) fig.savefig("mantle_convection.jpg") .. figure:: mantle_convection.jpg :figwidth: 90% :align: center This demo can also be accessed as a `Python script `__.