From e3081538f71c9eba1002bd996910e4be1f3dd498 Mon Sep 17 00:00:00 2001 From: benSepanski <ben_sepanski@alumni.baylor.edu> Date: Wed, 24 Jun 2020 16:31:23 -0500 Subject: [PATCH] Moved useful methods from mesh_topology.py into mesh.py --- meshmode/interop/firedrake/mesh.py | 158 +++++++++++++++++++++++++++++ 1 file changed, 158 insertions(+) create mode 100644 meshmode/interop/firedrake/mesh.py diff --git a/meshmode/interop/firedrake/mesh.py b/meshmode/interop/firedrake/mesh.py new file mode 100644 index 00000000..d067fb27 --- /dev/null +++ b/meshmode/interop/firedrake/mesh.py @@ -0,0 +1,158 @@ +__copyright__ = "Copyright (C) 2020 Benjamin Sepanski" + +__license__ = """ +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. +""" + +__doc__ = """ +.. autofunction:: get_firedrake_nodal_adjacency_group +.. autofunction:: get_firedrake_boundary_tags +""" + +from warnings import warn # noqa +import numpy as np + + +# {{{ functions to extract information from Mesh Topology + +def get_firedrake_nodal_adjacency_group(fdrake_mesh, cells_to_use=None): + """ + Create a nodal adjacency object + representing the nodal adjacency of :param:`fdrake_mesh` + + :param fdrake_mesh: A firedrake mesh (:class:`MeshTopology` + or :class:`MeshGeometry`) + :param cells_to_use: Either + + * *None*, in which case this argument is is ignored + * A numpy array of firedrake cell indices, in which case + any cell with indices not in the array :param:`cells_to_use` + is ignored. + This induces a new order on the cells. + The *i*th element in the returned :class:`NodalAdjacency` + object corresponds to the ``cells_to_use[i]``th cell + in the firedrake mesh. + + This feature has been used when only part of the mesh + needs to be converted, since firedrake has no concept + of a "sub-mesh". + + :return: A :class:`meshmode.mesh.NodalAdjacency` instance + representing the nodal adjacency of :param:`fdrake_mesh` + """ + mesh_topology = fdrake_mesh.topology + # TODO... not sure how to get around the private access + plex = mesh_topology._plex + + # dmplex cell Start/end and vertex Start/end. + c_start, c_end = plex.getHeightStratum(0) + v_start, v_end = plex.getDepthStratum(0) + + # TODO... not sure how to get around the private access + # This maps the dmplex index of a cell to its firedrake index + to_fd_id = np.vectorize(mesh_topology._cell_numbering.getOffset)( + np.arange(c_start, c_end, dtype=np.int32)) + + element_to_neighbors = {} + verts_checked = set() # dmplex ids of vertex checked + + # If using all cells, loop over them all + if cells_to_use is None: + range_ = range(c_start, c_end) + # Otherwise, just the ones you're using + else: + assert isinstance(cells_to_use, np.ndarray) + assert np.size(cells_to_use) == np.size(np.unique(cells_to_use)), \ + "cells_to_use must have unique values" + assert len(np.shape(cells_to_use)) == 1 and len(cells_to_use) > 0 + isin = np.isin(to_fd_id, cells_to_use) + range_ = np.arange(c_start, c_end, dtype=np.int32)[isin] + + # For each cell + for cell_id in range_: + # For each vertex touching the cell (that haven't already seen) + for vert_id in plex.getTransitiveClosure(cell_id)[0]: + if v_start <= vert_id < v_end and vert_id not in verts_checked: + verts_checked.add(vert_id) + cells = [] + # Record all cells touching that vertex + support = plex.getTransitiveClosure(vert_id, useCone=False)[0] + for other_cell_id in support: + if c_start <= other_cell_id < c_end: + cells.append(to_fd_id[other_cell_id - c_start]) + + # If only using some cells, clean out extraneous ones + # and relabel them to new id + cells = set(cells) + if cells_to_use is not None: + cells = set([cells_to_use[fd_ndx] for fd_ndx in cells + if fd_ndx in cells_to_use]) + + # mark cells as neighbors + for cell_one in cells: + element_to_neighbors.setdefault(cell_one, set()) + element_to_neighbors[cell_one] |= cells + + # Count the number of cells + if cells_to_use is None: + nelements = mesh_topology.num_cells() + else: + nelements = cells_to_use.shape[0] + + # Create neighbors_starts and neighbors + neighbors = [] + # FIXME : Is this the right integer type to choose? + neighbors_starts = np.zeros(nelements + 1, dtype=np.int32) + for iel in range(len(element_to_neighbors)): + elt_neighbors = element_to_neighbors[iel] + neighbors += list(elt_neighbors) + neighbors_starts[iel+1] = len(neighbors) + + neighbors = np.array(neighbors, dtype=np.int32) + + from meshmode.mesh import NodalAdjacency + return NodalAdjacency(neighbors_starts=neighbors_starts, + neighbors=neighbors) + + +def get_firedrake_boundary_tags(fdrake_mesh): + """ + Return a tuple of bdy tags as requested in + the construction of a :mod:`meshmode` :class:`Mesh` + + The tags used are :class:`meshmode.mesh.BTAG_ALL`, + :class:`meshmode.mesh.BTAG_REALLY_ALL`, and + any markers in the mesh topology's exterior facets + (see :attr:`firedrake.mesh.MeshTopology.exterior_facets.unique_markers`) + + :param fdrake_mesh: A :mod:`firedrake` :class:`MeshTopology` or + :class:`MeshGeometry` + + :return: A tuple of boundary tags + """ + from meshmode.mesh import BTAG_ALL, BTAG_REALLY_ALL + bdy_tags = [BTAG_ALL, BTAG_REALLY_ALL] + + unique_markers = fdrake_mesh.topology.exterior_facets.unique_markers + if unique_markers is not None: + bdy_tags += list(unique_markers) + + return tuple(bdy_tags) + +# }}} -- GitLab