Implementation#
Author: Jørgen Schartum Dokken
This implementation is an adaptation of the work in [LL16] to DOLFINx.
In this section, you will learn:
How to use the built-in meshes in DOLFINx
How to create a spatially varying Dirichlet boundary conditions on the whole domain boundary
How to define a weak formulation of your PDE
How to solve the resulting system of linear equations
How to visualize the solution using a variety of tools
How to compute the
error and the error at mesh vertices
Interactive tutorials#
Run the tutorial as Jupyter notebook in browser
As this book has been published as a Jupyter Book, each code can be run in your browser as a Jupyter notebook. To start such a notebook click the rocket symbol in the top right corner of the relevant tutorial.
The Poisson problem has so far featured a general domain
We use this fact to construct a quadratic function in
Inserting
regardless of the shape of the domain as long as we prescribe
For simplicity, we choose the domain to be a unit square
This simple but very powerful method for constructing test problems is called the method of manufactured solutions.
First pick a simple expression for the exact solution, plug into
the equation to obtain the right-hand side (source term
Note that in many cases, it can be hard to determine if the program works if it produces an error of size
However, in cases where we have a solution we know that should have no approximation error, we know that the solution should be produced to machine precision by the program.
A major difference between a traditional FEniCS code and a FEniCSx code, is that one is not advised to use the wildcard import. We will see this throughout this first example.
Generating simple meshes#
The next step is to define the discrete domain, the mesh. We do this by importing one of the built-in mesh generators. We will build a unit square mesh, i.e. a mesh spanning
from mpi4py import MPI
from dolfinx import mesh
domain = mesh.create_unit_square(MPI.COMM_WORLD, 8, 8, mesh.CellType.quadrilateral)
Note that in addition to give how many elements we would like to have in each direction.
We also have to supply the MPI-communicator.
This is to specify how we would like the program to behave in parallel.
If we supply MPI.COMM_WORLD
we create a single mesh, whose data is distributed over the number of processors we
would like to use. We can for instance run the program in parallel on two processors by using mpirun
, as:
mpirun -n 2 python3 t1.py
However, if we would like to create a separate mesh on each processor, we can use MPI.COMM_SELF
.
This is for instance useful if we run a small problem, and would like to run it with multiple parameters.
Defining the finite element function space#
Once the mesh has been created, we can create the finite element function space dolfinx.fem
module.
from dolfinx.fem import functionspace
V = functionspace(domain, ("Lagrange", 1))
from dolfinx import fem
uD = fem.Function(V)
uD.interpolate(lambda x: 1 + x[0]**2 + 2 * x[1]**2)
We now have the boundary data (and in this case the solution of
the finite element problem) represented in the discrete function space.
Next we would like to apply the boundary values to all degrees of freedom that are on the boundary of the discrete domain. We start by identifying the facets (line-segments) representing the outer boundary, using dolfinx.mesh.exterior_facet_indices
.
import numpy
# Create facet to cell connectivity required to determine boundary facets
tdim = domain.topology.dim
fdim = tdim - 1
domain.topology.create_connectivity(fdim, tdim)
boundary_facets = mesh.exterior_facet_indices(domain.topology)
For the current problem, as we are using the “Lagrange” 1 function space, the degrees of freedom are located at the vertices of each cell, thus each facet contains two degrees of freedom.
To find the local indices of these degrees of freedom, we use dolfinx.fem.locate_dofs_topological
, which takes in the function space, the dimension of entities in the mesh we would like to identify and the local entities.
Local ordering of degrees of freedom and mesh vertices
Many people expect there to be a 1-1 correspondence between the mesh coordinates and the coordinates of the degrees of freedom.
However, this is only true in the case of Lagrange
1 elements on a first order mesh. Therefore, in DOLFINx we use separate local numbering for the mesh coordinates and the dof coordinates. To obtain the local dof coordinates we can use V.tabulate_dof_coordinates()
, while the ordering of the local vertices can be obtained by mesh.geometry.x
.
With this data at hand, we can create the Dirichlet boundary condition
boundary_dofs = fem.locate_dofs_topological(V, fdim, boundary_facets)
bc = fem.dirichletbc(uD, boundary_dofs)
Defining the trial and test function#
In mathematics, we distinguish between trial and test spaces
We use the Unified Form Language (UFL) to specify the varitional formulations. See [AlnaesLOlgaard+14] for more details.
import ufl
u = ufl.TrialFunction(V)
v = ufl.TestFunction(V)
Defining the source term#
As the source term is constant over the domain, we use dolfinx.Constant
from dolfinx import default_scalar_type
f = fem.Constant(domain, default_scalar_type(-6))
Compilation speed-up
Instead of wrapping dolfinx.Constant
, we could simply define f=-6
.
However, if we would like to change this parameter later in the simulation, we would have to redefine our variational formulation. The dolfinx.Constant
allows us to update the value in f.value=5
. Additionally, by indicating that
Defining the variational problem#
As we now have defined all variables used to describe our variational problem, we can create the weak formulation
a = ufl.dot(ufl.grad(u), ufl.grad(v)) * ufl.dx
L = f * v * ufl.dx
Note that there is a very close correspondence between the Python syntax and the mathematical syntax
ufl.dx
, an integration measure over all cells of the mesh.
This is the key strength of FEniCSx: the formulas in the variational formulation translate directly to very similar Python code, a feature that makes it easy to specify and solve complicated PDE problems.
Expressing inner products#
The inner product ufl.dot(ufl.grad(u), ufl.grad(v))*ufl.dx
. The dot product in UFL computes the sum (contraction) over the last index of the first factor and first index of the second factor. In this case, both factors are tensors of rank one (vectors) and so the sum is just over the single index of both ufl.dot
use the function ufl.inner
. For vectors, ufl.dot
and ufl.inner
are equivalent.
Complex numbers
In DOLFINx, one can solve complex number problems by using an installation of PETSc using complex numbers.
For variational formulations with complex numbers, one cannot use ufl.dot
to compute inner products.
One has to use ufl.inner
, with the test-function as the second input argument for ufl.inner
.
See Running DOLFINx in complex mode for more information.
Forming and solving the linear system#
Having defined the finite element variational problem and boundary condition, we can create our dolfinx.fem.petsc.LinearProblem
, as class for solving
the variational problem: Find
from dolfinx.fem.petsc import LinearProblem
problem = LinearProblem(a, L, bcs=[bc], petsc_options={"ksp_type": "preonly", "pc_type": "lu"})
uh = problem.solve()
Using problem.solve()
we solve the linear system of equations and return a dolfinx.fem.Function
containing the solution.
Computing the error#
Finally, we want to compute the error to check the accuracy of the solution. We do this by comparing the finite element solution u
with the exact solution. We do this by interpolating the exact solution into the the
V2 = fem.functionspace(domain, ("Lagrange", 2))
uex = fem.Function(V2)
uex.interpolate(lambda x: 1 + x[0]**2 + 2 * x[1]**2)
We compute the error in two different ways. First, we compute the dolfinx.fem.assemble_scalar
to compute the scalar value. In DOLFINx, assemble_scalar
only assembles over the cells on the local process. This means that if we use 2 processes to solve our problem, we need to gather the solution to one (or all the processes.
We can do this with the MPI.allreduce
function.
L2_error = fem.form(ufl.inner(uh - uex, uh - uex) * ufl.dx)
error_local = fem.assemble_scalar(L2_error)
error_L2 = numpy.sqrt(domain.comm.allreduce(error_local, op=MPI.SUM))
Secondly, we compute the maximum error at any degree of freedom.
As the finite element function problem.solve()
we compute all the coefficients uh
.
However, as a second order function space has more dofs than a linear function space, we cannot compare these arrays directly.
As we allready have interpolated the exact solution into the first order space when creating the boundary condition, we can compare the maximum values at any degree of freedom of the approximation space.
error_max = numpy.max(numpy.abs(uD.x.array-uh.x.array))
# Only print the error on one process
if domain.comm.rank == 0:
print(f"Error_L2 : {error_L2:.2e}")
print(f"Error_max : {error_max:.2e}")
Error_L2 : 8.24e-03
Error_max : 3.55e-15
Plotting the mesh using pyvista#
We will visualizing the mesh using pyvista, an interface to the VTK toolkit.
We start by converting the mesh to a format that can be used with pyvista
.
To do this we use the function dolfinx.plot.vtk_mesh
. The first step is to create an unstructured grid that can be used by pyvista
.
We need to start a virtual framebuffer for plotting through docker containers. You can print the current backend and change it with pyvista.set_jupyter_backend(backend)
import pyvista
print(pyvista.global_theme.jupyter_backend)
html
from dolfinx import plot
pyvista.start_xvfb()
domain.topology.create_connectivity(tdim, tdim)
topology, cell_types, geometry = plot.vtk_mesh(domain, tdim)
grid = pyvista.UnstructuredGrid(topology, cell_types, geometry)
There are several backends that can be used with pyvista, and they have different benefits and drawbacks. See the pyvista documentation for more information and installation details. In this example and the rest of the tutorial we will use panel.
We can now use the pyvista.Plotter
to visualize the mesh. We visualize it by showing it in 2D and warped in 3D.
In the jupyter notebook environment, we use the default setting of pyvista.OFF_SCREEN=False
, which will render plots directly in the notebook.
plotter = pyvista.Plotter()
plotter.add_mesh(grid, show_edges=True)
plotter.view_xy()
if not pyvista.OFF_SCREEN:
plotter.show()
else:
figure = plotter.screenshot("fundamentals_mesh.png")
Plotting a function using pyvista#
We want to plot the solution uh
. As the function space used to defined the mesh is disconnected from the function space defining the mesh, we create a mesh based on the dof coordinates for the function space V
. We use dolfinx.plot.vtk_mesh
with the function space as input to create a mesh with mesh geometry based on the dof coordinates.
u_topology, u_cell_types, u_geometry = plot.vtk_mesh(V)
Next, we create the pyvista.UnstructuredGrid
and add the dof-values to the mesh.
u_grid = pyvista.UnstructuredGrid(u_topology, u_cell_types, u_geometry)
u_grid.point_data["u"] = uh.x.array.real
u_grid.set_active_scalars("u")
u_plotter = pyvista.Plotter()
u_plotter.add_mesh(u_grid, show_edges=True)
u_plotter.view_xy()
if not pyvista.OFF_SCREEN:
u_plotter.show()
We can also warp the mesh by scalar to make use of the 3D plotting.
warped = u_grid.warp_by_scalar()
plotter2 = pyvista.Plotter()
plotter2.add_mesh(warped, show_edges=True, show_scalar_bar=True)
if not pyvista.OFF_SCREEN:
plotter2.show()
External post-processing#
For post-processing outside the python code, it is suggested to save the solution to file using either dolfinx.io.VTXWriter
or dolfinx.io.XDMFFile
and using Paraview. This is especially suggested for 3D visualization.
from dolfinx import io
from pathlib import Path
results_folder = Path("results")
results_folder.mkdir(exist_ok=True, parents=True)
filename = results_folder / "fundamentals"
with io.VTXWriter(domain.comm, filename.with_suffix(".bp"), [uh]) as vtx:
vtx.write(0.0)
with io.XDMFFile(domain.comm, filename.with_suffix(".xdmf"), "w") as xdmf:
xdmf.write_mesh(domain)
xdmf.write_function(uh)
Martin S. Alnæs, Anders Logg, Kristian B. Ølgaard, Marie E. Rognes, and Garth N. Wells. Unified form language: a domain-specific language for weak formulations of partial differential equations. ACM Trans. Math. Softw., 2014. doi:10.1145/2566630.