Meshes from external sources#
As DOLFINx works on numpy
-arrays it is quite straightforward to convert any mesh format into this structure.
DOLFINx has a gmsh
-interface, using the GMSH Python-API to read GMSH
-models or .msh
files.
We will start with creating a simple circular geometry (similar to the one used in the
Eshelby problem) using GMSH.
The Eshelby has an ellipsoid inclusion in the circular domain.
from mpi4py import MPI
import gmsh
import dolfinx
The first thing we do when using GMSH is to initialize it explicitly
gmsh.initialize()
A single GMSH instance can create multiple separate geometries, named models. We start by creating a model
gmsh.model.add("eshelby")
Generating parameterized geometries#
Next, we use the Open Cascade OCC backend to create a two disks, inner_disk
with radius R_i
and an outer_disk
with radius R_e
.
The disks will have a center at the origin and we can select an aspect ratio to make the inner disk elliptical
center = (0, 0, 0)
aspect_ratio = 0.5
R_i = 0.3
R_e = 0.8
inner_disk = gmsh.model.occ.addDisk(*center, R_i, aspect_ratio * R_i)
outer_disk = gmsh.model.occ.addDisk(*center, R_e, R_e)
What GMSH returns for each of the addDisk
calls is an integer, which internally represents a geometry
object of topological dimension two.
inner_disk=1
outer_disk=2
GMSH objects and mesh resolution
The disk objects has no concept of mesh resolution. This is controlled at a later stage using GMSH size field or meshing options, where a finer resolution will more closely approximate the disks. Two-dimensional meshing algorithms uses the parameterization of the circle to create the mesh.
Boolean operations of entities#
In GMSH, you can combine multiple geometries into a single mesh (by unions, intersections or differences).
We use the fragment
function to embed the boundary of the inner circle in the outer disk.
Now that we have created two parametric representations of a disk, we would like to create a combined surface, where each of the circular boundaries are included.
whole_domain, map_to_input = gmsh.model.occ.fragment([(2, outer_disk)], [(2, inner_disk)])
Info : [ 0%] Fragments
Info : [ 10%] Fragments
Info : [ 20%] Fragments
Info : [ 30%] Fragments
Info : [ 40%] Fragments - Performing Face-Face intersection
Info : [ 70%] Fragments
Info : [ 80%] Fragments - Making faces
Info : [ 90%] Fragments - Adding holes
We can inspect the first output of the fragment function
whole_domain=[(2, 1), (2, 2)]
We observe that we get a list of tuples. Each tuple (dim, obj_idx)
represents an OCC object of
dimension dim
and internal index obj_index
. We observe that we have two surfaces of dimension 2.
The second output of fragment is a tuple of such lists, where the first list contains objects
that are embedded in the first input object (outer_disk
), and the second list contains objects
that are embedded in the second input object (inner_disk
).
As the outer_disk
contains the inner disk, we observe that the first list and second list shares
an object
map_to_input[0]=[(2, 2), (2, 1)]
map_to_input[1]=[(2, 1)]
To make it possible for other functions in GMSH than those in the OCC module to modify
the domain
, we need to synchronize the model.
gmsh.model.occ.synchronize()
Physical groups#
In order to ensure that only the parts of the mesh that you would like to use is generated (and no overlapping cells),
you need to create Physical Groups
for all volumes, surfaces and curves you would like to have in your mesh.
We start by creating a physical marker for the inner circle.
The addPhysicalGroup
function takes three arguments:
The topological dimension of the entity
A list of GMSH objects that should be tagged
The marker that should be assigned to the tagged objects
Physical surfaces#
We start by marking the inner circle
circle_inner = [idx for (dim, idx) in map_to_input[1] if dim == 2]
Next, we known that the outer circle is the only remaining object of dimension 2
circle_outer = [idx for (dim, idx) in map_to_input[0] if dim == 2 and idx not in circle_inner]
circle_inner=[1]
circle_outer=[2]
We create a unique physical group for each surface
gmsh.model.addPhysicalGroup(2, circle_inner, tag=3)
gmsh.model.addPhysicalGroup(2, circle_outer, tag=7)
Physical curves#
Next, we would like to mark the external boundary, such that we can apply boundary conditions to it. We also want to mark the interface between the two domains
We start by getting the boundary of each of the surfaces.
Input to getBoundary
Note that the input format to getBoundary
is a list of tuples, similar to the output of fragment
.
However, as we have split the groups into to lists of just indices, we need to convert this back
to this format.
recursive
is a boolean. If True
would return the entities of lowest dimension (0), instead
of the dimension one lower than the input entities.
oriented
will multiply the object index with -1
if this gives a consistent orientation of the boundary.
inner_boundary = gmsh.model.getBoundary([(2, e) for e in circle_inner], recursive=False, oriented=False)
outer_boundary = gmsh.model.getBoundary([(2, e) for e in circle_outer], recursive=False, oriented=False)
We note that the outer circle now has a “donut” shape, i.e. it has two boundaries
inner_boundary=[(1, 2)]
outer_boundary=[(1, 1), (1, 2)]
We remove this boundary from the external boundaries before creating a physical group
interface = [idx for (dim, idx) in inner_boundary if dim == 1]
ext_boundary = [idx for (dim, idx) in outer_boundary if idx not in interface and dim == 1]
gmsh.model.addPhysicalGroup(1, interface, tag=12)
gmsh.model.addPhysicalGroup(1, ext_boundary, tag=15)
Next, we can generate the mesh with a given minimal resolution
gmsh.option.setNumber("Mesh.CharacteristicLengthMax", 0.2)
gmsh.model.mesh.generate(2)
Info : Meshing 1D...
Info : [ 0%] Meshing curve 1 (Ellipse)
Info : [ 60%] Meshing curve 2 (Ellipse)
Info : Done meshing 1D (Wall 0.00332466s, CPU 0.003916s)
Info : Meshing 2D...
Info : [ 0%] Meshing surface 1 (Plane, Frontal-Delaunay)
Info : [ 60%] Meshing surface 2 (Plane, Frontal-Delaunay)
Info : Done meshing 2D (Wall 0.00218758s, CPU 0.002281s)
Info : 90 nodes 188 elements
In GMSH we can choose to mesh first, second or third order meshes. We choose a 3rd order mesh for the circle.
gmsh.model.mesh.setOrder(3)
Info : Meshing order 3 (curvilinear on)...
Info : [ 0%] Meshing curve 1 order 3
Info : [ 30%] Meshing curve 2 order 3
Info : [ 60%] Meshing surface 1 order 3
Info : [ 80%] Meshing surface 2 order 3
Info : Surface mesh: worst distortion = 0.426319 (0 elements in ]0, 0.2]); worst gamma = 0.847892
Info : Done meshing order 3 (Wall 0.00221793s, CPU 0.003086s)
The mesh is generated, but not yet saved to disk. We could save it to disk using gmsh.write
command, or import it
directly into DOLFINx.
We choose the latter, and use dolfinx.io.gmshio.model_to_mesh
to convert the GMSH model to a DOLFINx mesh.
For this function we need to send in a few objects:
The
gmsh.model
instanceThe
MPI
communicator we want to distribute the mesh overThe
MPI
rank that holds the meshGMSH and MPI
GMSH does not have a concept of an MPI distributed mesh, and will generate a mesh on each process if used as above. Usually, one would call
gmsh.initialize()
on all processors, then add anif-else
clause that only generates the mesh on a single processor (usually rank 0).The geometrical dimension of the mesh when used in DOLFINx
Geometrical dimension of a mesh from GMSH
GMSH always generates three-dimensional points, which means that a 1D or 2D mesh could be interpreted as a manifold in a higher dimensional space. The
gdim
argument is used to specify the dimension of the mesh when used in DOLFINx.
With these inputs we can generate the mesh and the cell and facet markers.
circular_mesh, cell_marker, facet_marker = dolfinx.io.gmshio.model_to_mesh(gmsh.model, MPI.COMM_WORLD, 0, gdim=2)
We can now finalize GMSH (as we will not use it further in this section), and inspect the cell_marker
and facet_marker
.
gmsh.finalize()