Hessian-based mesh adaptation for a 2D steady-state problem

Previous demos have covered the fundamental time partition and mesh sequence objects of Goalie, using them to solve PDEs over multiple meshes and perform goal-oriented error estimation. Here, we demonstrate how to use them for Hessian-based mesh adaptation for a steady-state problem in 2D.

It is recommended that you read the documentation on metric-based mesh adaptation before progressing with this demo.

In addition to importing from Firedrake and Goalie, we also import the mesh adaptation functionality from Animate, which can be found in animate.adapt.

from animate.adapt import adapt
from animate.metric import RiemannianMetric
from firedrake import *

from goalie import *

We again consider the “point discharge with diffusion” test case from the previous demo, approximating the tracer concentration \(c\) in \(\mathbb P1\) space.

field_names = ["c"]


def get_function_spaces(mesh):
    return {"c": FunctionSpace(mesh, "CG", 1)}


def source(mesh):
    x, y = SpatialCoordinate(mesh)
    x0, y0, r = 2, 5, 0.05606388
    return 100.0 * exp(-((x - x0) ** 2 + (y - y0) ** 2) / r**2)


def get_solver(mesh_seq):
    def solver(index):
        function_space = mesh_seq.function_spaces["c"][index]
        c = mesh_seq.fields["c"]
        h = CellSize(mesh_seq[index])
        S = source(mesh_seq[index])

        # Define constants
        R = FunctionSpace(mesh_seq[index], "R", 0)
        D = Function(R).assign(0.1)
        u_x = Function(R).assign(1.0)
        u_y = Function(R).assign(0.0)
        u = as_vector([u_x, u_y])

        # SUPG stabilisation parameter
        unorm = sqrt(dot(u, u))
        tau = 0.5 * h / unorm
        tau = min_value(tau, unorm * h / (6 * D))

        # Setup variational problem
        psi = TestFunction(function_space)
        psi = psi + tau * dot(u, grad(psi))
        F = (
            dot(u, grad(c)) * psi * dx
            + inner(D * grad(c), grad(psi)) * dx
            - S * psi * dx
        )
        bc = DirichletBC(function_space, 0, 1)

        solve(F == 0, c, bcs=bc, ad_block_tag="c")
        yield

    return solver

Take a relatively coarse initial mesh, a TimeInstant (since we have a steady-state problem), and put everything together in a MeshSeq.

mesh = RectangleMesh(50, 10, 50, 10)
time_partition = TimeInstant(field_names)
mesh_seq = MeshSeq(
    time_partition,
    mesh,
    get_function_spaces=get_function_spaces,
    get_solver=get_solver,
)

Give the initial mesh, we can plot it, solve the PDE on it, and plot the resulting solution field.

import matplotlib.pyplot as plt

fig, axes = plt.subplots(figsize=(10, 2))
interior_kw = {"linewidth": 0.5}
mesh_seq.plot(fig=fig, axes=axes, interior_kw=interior_kw)
axes.set_title("Initial mesh")
fig.savefig("point_discharge2d-mesh0.jpg")
plt.close()

solutions = mesh_seq.solve_forward()

plot_kwargs = {"levels": 50, "figsize": (10, 3), "cmap": "coolwarm"}
fig, axes, tcs = plot_snapshots(
    solutions,
    time_partition,
    "c",
    "forward",
    **plot_kwargs,
)
cbar = fig.colorbar(tcs[0][0], orientation="horizontal", pad=0.2)
axes.set_title("Forward solution on initial mesh")
fig.savefig("point_discharge2d-forward_init.jpg")
plt.close()
../_images/point_discharge2d-mesh0.jpg
../_images/point_discharge2d-forward_init.jpg

In order to perform metric-based mesh adaptation, we need to tell the mesh sequence how to define metrics and adapt its meshes. Since we have a steady-state problem, there is only ever one mesh, one solution field, and one metric, so this simplifies things significantly.

For this first example, we compute a metric by recovering the Hessian of the approximate solution field, and scaling it according to a desired metric complexity using \(L^p\) normalisation. The normalised metric is used to adapt the mesh, which we print the obtained metric complexity and DoF count of the adapted mesh. Since the solution is sought in \(\mathbb P1\) space, the DoF count is just the vertex count.

def adaptor(mesh_seq, solutions):
    c = solutions["c"]["forward"][0][0]

    # Define the Riemannian metric
    P1_tensor = TensorFunctionSpace(mesh_seq[0], "CG", 1)
    metric = RiemannianMetric(P1_tensor)

    # Recover the Hessian of the current solution
    metric.compute_hessian(c)

    # Ramp the target metric complexity from 400 to 1000 over the first few iterations
    base, target, iteration = 400, 1000, mesh_seq.fp_iteration
    mp = {"dm_plex_metric_target_complexity": ramp_complexity(base, target, iteration)}
    metric.set_parameters(mp)

    # Normalise the metric according to the target complexity and then adapt the mesh
    metric.normalise()
    complexity = metric.complexity()
    mesh_seq[0] = adapt(mesh_seq[0], metric)
    num_dofs = mesh_seq.count_vertices()[0]
    num_elem = mesh_seq.count_elements()[0]
    pyrint(
        f"{iteration + 1:2d}, complexity: {complexity:4.0f}"
        f", dofs: {num_dofs:4d}, elements: {num_elem:4d}"
    )

    # Plot each intermediate adapted mesh
    fig, axes = plt.subplots(figsize=(10, 2))
    mesh_seq.plot(fig=fig, axes=axes, interior_kw=interior_kw)
    axes.set_title(f"Mesh at iteration {iteration + 1}")
    fig.savefig(f"point_discharge2d-hessian_mesh{iteration + 1}.jpg")
    plt.close()

    # Check whether the target complexity has been (approximately) reached. If not,
    # return ``True`` to indicate that convergence checks should be skipped until the
    # next fixed point iteration.
    continue_unconditionally = complexity < 0.95 * target
    return [continue_unconditionally]

With the adaptor function defined, we can call the fixed point iteration method, which iteratively solves the PDE and calls the adaptor until one of the convergence criteria are met. To that end, we create a AdaptParameters object and set the element_rtol parameter to 0.005. This means that the fixed point iteration will terminate if the element count changes by less than 0.5% between iterations. As standard, we allow 35 iterations before establishing that the iteration is not going to converge.

params = AdaptParameters(
    {
        "element_rtol": 0.005,
        "maxiter": 35,
    }
)
solutions = mesh_seq.fixed_point_iteration(adaptor, parameters=params)

Mesh adaptation often gives slightly different results on difference machines. However, the output should look something like

 1, complexity:  433, dofs:  618, elements: 1161
 2, complexity:  630, dofs:  898, elements: 1725
 3, complexity:  825, dofs: 1128, elements: 2180
 4, complexity: 1023, dofs: 1336, elements: 2592
 5, complexity: 1020, dofs: 1354, elements: 2629
 6, complexity: 1022, dofs: 1362, elements: 2643
 7, complexity: 1022, dofs: 1356, elements: 2635
Element count converged after 7 iterations under relative tolerance 0.005.

We can plot the final mesh and the corresponding solution as follows.

fig, axes = plt.subplots(figsize=(10, 2))
mesh_seq.plot(fig=fig, axes=axes, interior_kw=interior_kw)
axes.set_title("Adapted mesh")
fig.savefig("point_discharge2d-hessian_mesh.jpg")
plt.close()

fig, axes, tcs = plot_snapshots(
    solutions,
    time_partition,
    "c",
    "forward",
    **plot_kwargs,
)
cbar = fig.colorbar(tcs[0][0], orientation="horizontal", pad=0.2)
axes.set_title("Forward solution on adapted mesh")
fig.savefig("point_discharge2d-forward_hessian_adapted.jpg")
plt.close()
../_images/point_discharge2d-hessian_mesh.jpg
../_images/point_discharge2d-forward_hessian_adapted.jpg

Notice how the adapted mesh has increased resolution in the regions of highest curvature in the solution field. Moreover, it is anisotropic, with the orientation of anisotropic elements aligned with the contours of the solution field.

In the next demo, we consider the same problem again, but using goal-oriented mesh adaptation.

This demo can also be accessed as a Python script.