Skip to content

Limiter

limiter

This module provides a class that implements limiting of functions defined on discontinuous function spaces. Users instantiate the class by providing relevant parameters and call the apply method to update the function.

VertexBasedP1DGLimiter(p1dg_space, clip_min=None, clip_max=None)

Bases: VertexBasedLimiter

Vertex-based limiter for P1DG tracer fields (Kuzmin, 2010).

Kuzmin, D. (2010). A vertex-based hierarchical slope limiter for p-adaptive discontinuous Galerkin methods. Journal of computational and applied mathematics, 233(12), 3077-3085.

Parameters:

Name Type Description Default
p1dg_space WithGeometry

UFL P1DG function space

required
clip_min Optional[float]

Minimal threshold to apply

None
clip_max Optional[float]

Maximal threshold to apply

None
Source code in g-adopt/gadopt/limiter.py
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
def __init__(
    self,
    p1dg_space: WithGeometry,
    clip_min: Optional[float] = None,
    clip_max: Optional[float] = None,
):
    assert_function_space(p1dg_space, ['Discontinuous Lagrange', 'DQ'], 1)

    self.is_vector = p1dg_space.value_size > 1
    if self.is_vector:
        p1dg_scalar_space = FunctionSpace(p1dg_space.mesh(), 'DG', 1)
        super(VertexBasedP1DGLimiter, self).__init__(p1dg_scalar_space)
    else:
        super(VertexBasedP1DGLimiter, self).__init__(p1dg_space)

    self.mesh = self.P0.mesh()
    self.dim = self.mesh.geometric_dimension()
    self.extruded = hasattr(self.mesh.ufl_cell(), 'sub_cells')
    assert not self.extruded or len(p1dg_space.ufl_element().sub_elements) > 0, \
        "Extruded mesh requires extruded function space"
    assert not self.extruded or all(e.variant() == 'equispaced' for e in p1dg_space.ufl_element().sub_elements), \
        "Extruded function space must be equivariant"

    self.clip_min = clip_min
    self.clip_max = clip_max

compute_bounds(field)

Re-computes min/max values of all neighbouring centroids.

Parameters:

Name Type Description Default
field Function

Firedrake function onto which the limiter is applied

required
Source code in g-adopt/gadopt/limiter.py
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
def compute_bounds(self, field: Function):
    """Re-computes min/max values of all neighbouring centroids.

    Arguments:
      field: Firedrake function onto which the limiter is applied

    """
    # Call general-purpose bound computation.
    super(VertexBasedP1DGLimiter, self).compute_bounds(field)

    # Add the average of lateral boundary facets to min/max fields
    # NOTE this just computes the arithmetic mean of nodal values on the facet,
    # which in general is not equivalent to the mean of the field over the bnd facet.
    # This is OK for P1DG triangles, but not exact for the extruded case (quad facets)
    from finat.finiteelementbase import entity_support_dofs

    if self.extruded:
        entity_dim = (self.dim-2, 1)  # get vertical facets
    else:
        entity_dim = self.dim-1
    boundary_dofs = entity_support_dofs(self.P1DG.finat_element, entity_dim)
    local_facet_nodes = np.array([boundary_dofs[e] for e in sorted(boundary_dofs.keys())])
    n_bnd_nodes = local_facet_nodes.shape[1]
    local_facet_idx = op2.Global(local_facet_nodes.shape, local_facet_nodes, dtype=np.int32, name='local_facet_idx')
    code = """
        void my_kernel(double *qmax, double *qmin, double *field, unsigned int *facet, unsigned int *local_facet_idx)
        {
            double face_mean = 0.0;
            for (int i = 0; i < %(nnodes)d; i++) {
                unsigned int idx = local_facet_idx[facet[0]*%(nnodes)d + i];
                face_mean += field[idx];
            }
            face_mean /= %(nnodes)d;
            for (int i = 0; i < %(nnodes)d; i++) {
                unsigned int idx = local_facet_idx[facet[0]*%(nnodes)d + i];
                qmax[idx] = fmax(qmax[idx], face_mean);
                qmin[idx] = fmin(qmin[idx], face_mean);
            }
        }"""
    bnd_kernel = op2.Kernel(code % {'nnodes': n_bnd_nodes}, 'my_kernel')
    op2.par_loop(bnd_kernel,
                 self.P1DG.mesh().exterior_facets.set,
                 self.max_field.dat(op2.MAX, self.max_field.exterior_facet_node_map()),
                 self.min_field.dat(op2.MIN, self.min_field.exterior_facet_node_map()),
                 field.dat(op2.READ, field.exterior_facet_node_map()),
                 self.P1DG.mesh().exterior_facets.local_facet_dat(op2.READ),
                 local_facet_idx(op2.READ))
    if self.extruded:
        # Add nodal values from surface/bottom boundaries
        # NOTE calling firedrake par_loop with measure=ds_t raises an error
        bottom_nodes = get_facet_mask(self.P1CG, 'bottom')
        top_nodes = get_facet_mask(self.P1CG, 'top')
        bottom_idx = op2.Global(len(bottom_nodes), bottom_nodes, dtype=np.int32, name='node_idx')
        top_idx = op2.Global(len(top_nodes), top_nodes, dtype=np.int32, name='node_idx')
        code = """
            void my_kernel(double *qmax, double *qmin, double *field, int *idx) {
                double face_mean = 0;
                for (int i=0; i<%(nnodes)d; i++) {
                    face_mean += field[idx[i]];
                }
                face_mean /= %(nnodes)d;
                for (int i=0; i<%(nnodes)d; i++) {
                    qmax[idx[i]] = fmax(qmax[idx[i]], face_mean);
                    qmin[idx[i]] = fmin(qmin[idx[i]], face_mean);
                }
            }"""
        kernel = op2.Kernel(code % {'nnodes': len(bottom_nodes)}, 'my_kernel')

        op2.par_loop(kernel, self.mesh.cell_set,
                     self.max_field.dat(op2.MAX, self.max_field.function_space().cell_node_map()),
                     self.min_field.dat(op2.MIN, self.min_field.function_space().cell_node_map()),
                     field.dat(op2.READ, field.function_space().cell_node_map()),
                     bottom_idx(op2.READ),
                     iteration_region=op2.ON_BOTTOM)

        op2.par_loop(kernel, self.mesh.cell_set,
                     self.max_field.dat(op2.MAX, self.max_field.function_space().cell_node_map()),
                     self.min_field.dat(op2.MIN, self.min_field.function_space().cell_node_map()),
                     field.dat(op2.READ, field.function_space().cell_node_map()),
                     top_idx(op2.READ),
                     iteration_region=op2.ON_TOP)

    if self.clip_min is not None:
        self.min_field.assign(max_value(self.min_field, self.clip_min))
    if self.clip_max is not None:
        self.max_field.assign(min_value(self.max_field, self.clip_max))

apply(field)

Applies the limiter on the given field (in place).

Parameters:

Name Type Description Default
field

Firedrake function onto which the limiter is applied

required
Source code in g-adopt/gadopt/limiter.py
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
def apply(self, field):
    """Applies the limiter on the given field (in place).

    Args:
      field: Firedrake function onto which the limiter is applied

    """
    with timed_stage('limiter'):

        if self.is_vector:
            tmp_func = self.P1DG.get_work_function()
            fs = field.function_space()
            for i in range(fs.value_size):
                tmp_func.dat.data_with_halos[:] = field.dat.data_with_halos[:, i]
                super(VertexBasedP1DGLimiter, self).apply(tmp_func)
                field.dat.data_with_halos[:, i] = tmp_func.dat.data_with_halos[:]
            self.P1DG.restore_work_function(tmp_func)
        else:
            super(VertexBasedP1DGLimiter, self).apply(field)

assert_function_space(fs, family, degree)

Checks the family and degree of the function space.

If the function space lies on an extruded mesh, checks both spaces of the outer product.

Parameters:

Name Type Description Default
fs WithGeometry

UFL function space

required
family str | list[str]

Name or names of the expected element families

required
degree int

Expected polynomial degree of the function space

required

Raises:

Type Description
AssertionError

The family and/or degree of the function space don't match the expected values

Source code in g-adopt/gadopt/limiter.py
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
def assert_function_space(
    fs: WithGeometry, family: str | list[str], degree: int
):
    """Checks the family and degree of the function space.

    If the function space lies on an extruded mesh, checks both spaces of the outer
    product.

    Arguments:
      fs: UFL function space
      family: Name or names of the expected element families
      degree: Expected polynomial degree of the function space

    Raises:
      AssertionError: The family and/or degree of the function
                      space don't match the expected values

    """
    fam_list = family
    if not isinstance(family, list):
        fam_list = [family]

    ufl_elem = fs.ufl_element()
    if isinstance(ufl_elem, VectorElement):
        ufl_elem = ufl_elem.sub_elements[0]

    if ufl_elem.family() == 'TensorProductElement':  # extruded mesh
        A, B = ufl_elem.sub_elements
        assert A.family() in fam_list, 'horizontal space must be one of {0:s}'.format(fam_list)
        assert B.family() in fam_list, 'vertical space must be {0:s}'.format(fam_list)
        assert A.degree() == degree, 'degree of horizontal space must be {0:d}'.format(degree)
        assert B.degree() == degree, 'degree of vertical space must be {0:d}'.format(degree)
    else:  # assume 2D mesh
        assert ufl_elem.family() in fam_list, 'function space must be one of {0:s}'.format(fam_list)
        assert ufl_elem.degree() == degree, 'degree of function space must be {0:d}'.format(degree)

get_extruded_base_element(ufl_element)

Gets the base element from an extruded element.

In case of a non-extruded mesh, returns the element itself.

Parameters:

Name Type Description Default
ufl_element FiniteElement

UFL element from which to extract the base element

required

Returns:

Type Description
FiniteElement

The base element, or the provided element for a non-extruded mesh.

Source code in g-adopt/gadopt/limiter.py
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
def get_extruded_base_element(ufl_element: FiniteElement) -> FiniteElement:
    """Gets the base element from an extruded element.

    In case of a non-extruded mesh, returns the element itself.

    Arguments:
      ufl_element: UFL element from which to extract the base element

    Returns:
      The base element, or the provided element for a non-extruded mesh.
    """
    if isinstance(ufl_element, HDivElement):
        ufl_element = ufl_element._element
    if isinstance(ufl_element, MixedElement):
        ufl_element = ufl_element.sub_elements[0]
    if isinstance(ufl_element, VectorElement):
        ufl_element = ufl_element.sub_elements[0]  # take the first component
    if isinstance(ufl_element, EnrichedElement):
        ufl_element = ufl_element._elements[0]
    return ufl_element

get_facet_mask(function_space, facet='bottom')

The meaning of top/bottom depends on the extrusion's direction. Here, we assume that the mesh has been extruded upwards (along the positive z axis).

Parameters:

Name Type Description Default
function_space WithGeometry

UFL function space

required
facet str

String specifying the facet ("bottom" or "top")

'bottom'

Returns:

Type Description
ndarray

The top/bottom nodes of extruded 3D elements.

Raises:

Type Description
AssertionError

The function space is not defined on an extruded mesh

Source code in g-adopt/gadopt/limiter.py
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
def get_facet_mask(
    function_space: WithGeometry, facet: str = 'bottom'
) -> np.ndarray:
    """The meaning of top/bottom depends on the extrusion's direction. Here, we assume
    that the mesh has been extruded upwards (along the positive z axis).

    Arguments:
      function_space: UFL function space
      facet: String specifying the facet ("bottom" or "top")

    Returns:
      The top/bottom nodes of extruded 3D elements.

    Raises:
      AssertionError: The function space is not defined on an extruded mesh

    """

    # get base element
    elem = get_extruded_base_element(function_space.ufl_element())
    assert isinstance(elem, TensorProductElement), \
        f'function space must be defined on an extruded 3D mesh: {elem}'
    # figure out number of nodes in sub elements
    h_elt, v_elt = function_space.finat_element.factors
    nb_nodes_h = h_elt.space_dimension()
    nb_nodes_v = v_elt.space_dimension()
    # compute top/bottom facet indices
    # extruded dimension is the inner loop in index
    # on interval elements, the end points are the first two dofs
    offset = 0 if facet == 'bottom' else 1
    indices = np.arange(nb_nodes_h)*nb_nodes_v + offset
    return indices