From 31e83674423bb1482b397169ed533235d2c87014 Mon Sep 17 00:00:00 2001 From: ksagiyam Date: Fri, 5 Jul 2024 17:08:16 +0100 Subject: [PATCH 1/2] utility_meshes: enable PeriodicBoxMesh(..., hexahedral=True) --- firedrake/utility_meshes.py | 401 +++++++++++++---------- tests/regression/test_mesh_generation.py | 11 + 2 files changed, 243 insertions(+), 169 deletions(-) diff --git a/firedrake/utility_meshes.py b/firedrake/utility_meshes.py index b0b3b94490..62f27ebff9 100644 --- a/firedrake/utility_meshes.py +++ b/firedrake/utility_meshes.py @@ -1609,6 +1609,38 @@ def TensorBoxMesh( return m +def _firedrake_box_mesh_hexahedral_mark_boundaries(plex, nx, ny, nz, Lx, Ly, Lz): + plex.removeLabel(dmcommon.FACE_SETS_LABEL) + nvert = 4 # num. vertices on faect + # Apply boundary IDs + plex.createLabel(dmcommon.FACE_SETS_LABEL) + plex.markBoundaryFaces("boundary_faces") + coords = plex.getCoordinates() + coord_sec = plex.getCoordinateSection() + cdim = plex.getCoordinateDim() + assert cdim == 3 + if plex.getStratumSize("boundary_faces", 1) > 0: + boundary_faces = plex.getStratumIS("boundary_faces", 1).getIndices() + xtol = Lx / (2 * nx) + ytol = Ly / (2 * ny) + ztol = Lz / (2 * nz) + for face in boundary_faces: + face_coords = plex.vecGetClosure(coord_sec, coords, face) + if all([abs(face_coords[0 + cdim * i]) < xtol for i in range(nvert)]): + plex.setLabelValue(dmcommon.FACE_SETS_LABEL, face, 1) + if all([abs(face_coords[0 + cdim * i] - Lx) < xtol for i in range(nvert)]): + plex.setLabelValue(dmcommon.FACE_SETS_LABEL, face, 2) + if all([abs(face_coords[1 + cdim * i]) < ytol for i in range(nvert)]): + plex.setLabelValue(dmcommon.FACE_SETS_LABEL, face, 3) + if all([abs(face_coords[1 + cdim * i] - Ly) < ytol for i in range(nvert)]): + plex.setLabelValue(dmcommon.FACE_SETS_LABEL, face, 4) + if all([abs(face_coords[2 + cdim * i]) < ztol for i in range(nvert)]): + plex.setLabelValue(dmcommon.FACE_SETS_LABEL, face, 5) + if all([abs(face_coords[2 + cdim * i] - Lz) < ztol for i in range(nvert)]): + plex.setLabelValue(dmcommon.FACE_SETS_LABEL, face, 6) + plex.removeLabel("boundary_faces") + + @PETSc.Log.EventDecorator() def BoxMesh( nx, @@ -1657,46 +1689,15 @@ def BoxMesh( raise ValueError("Number of cells must be a postive integer") if hexahedral: plex = PETSc.DMPlex().createBoxMesh((nx, ny, nz), lower=(0., 0., 0.), upper=(Lx, Ly, Lz), simplex=False, periodic=False, interpolate=True, comm=comm) - plex.removeLabel(dmcommon.FACE_SETS_LABEL) - nvert = 4 # num. vertices on faect - - # Apply boundary IDs - plex.createLabel(dmcommon.FACE_SETS_LABEL) - plex.markBoundaryFaces("boundary_faces") - coords = plex.getCoordinates() - coord_sec = plex.getCoordinateSection() - cdim = plex.getCoordinateDim() - assert cdim == 3 - if plex.getStratumSize("boundary_faces", 1) > 0: - boundary_faces = plex.getStratumIS("boundary_faces", 1).getIndices() - xtol = Lx / (2 * nx) - ytol = Ly / (2 * ny) - ztol = Lz / (2 * nz) - for face in boundary_faces: - face_coords = plex.vecGetClosure(coord_sec, coords, face) - if all([abs(face_coords[0 + cdim * i]) < xtol for i in range(nvert)]): - plex.setLabelValue(dmcommon.FACE_SETS_LABEL, face, 1) - if all([abs(face_coords[0 + cdim * i] - Lx) < xtol for i in range(nvert)]): - plex.setLabelValue(dmcommon.FACE_SETS_LABEL, face, 2) - if all([abs(face_coords[1 + cdim * i]) < ytol for i in range(nvert)]): - plex.setLabelValue(dmcommon.FACE_SETS_LABEL, face, 3) - if all([abs(face_coords[1 + cdim * i] - Ly) < ytol for i in range(nvert)]): - plex.setLabelValue(dmcommon.FACE_SETS_LABEL, face, 4) - if all([abs(face_coords[2 + cdim * i]) < ztol for i in range(nvert)]): - plex.setLabelValue(dmcommon.FACE_SETS_LABEL, face, 5) - if all([abs(face_coords[2 + cdim * i] - Lz) < ztol for i in range(nvert)]): - plex.setLabelValue(dmcommon.FACE_SETS_LABEL, face, 6) - plex.removeLabel("boundary_faces") - m = mesh.Mesh( + _firedrake_box_mesh_hexahedral_mark_boundaries(plex, nx, ny, nz, Lx, Ly, Lz) + return mesh.Mesh( plex, reorder=reorder, distribution_parameters=distribution_parameters, name=name, distribution_name=distribution_name, permutation_name=permutation_name, - comm=comm, - ) - return m + comm=comm) else: xcoords = np.linspace(0, Lx, nx + 1, dtype=np.double) ycoords = np.linspace(0, Ly, ny + 1, dtype=np.double) @@ -1837,6 +1838,8 @@ def PeriodicBoxMesh( Lx, Ly, Lz, + directions=None, + hexahedral=False, reorder=None, distribution_parameters=None, comm=COMM_WORLD, @@ -1846,136 +1849,174 @@ def PeriodicBoxMesh( ): """Generate a periodic mesh of a 3D box. - :arg nx: The number of cells in the x direction - :arg ny: The number of cells in the y direction - :arg nz: The number of cells in the z direction - :arg Lx: The extent in the x direction - :arg Ly: The extent in the y direction - :arg Lz: The extent in the z direction - :kwarg reorder: (optional), should the mesh be reordered? - :kwarg distribution_parameters: options controlling mesh - distribution, see :func:`.Mesh` for details. - :kwarg comm: Optional communicator to build the mesh on. - :kwarg name: Optional name of the mesh. - :kwarg distribution_name: the name of parallel distribution used - when checkpointing; if `None`, the name is automatically - generated. - :kwarg permutation_name: the name of entity permutation (reordering) used - when checkpointing; if `None`, the name is automatically - generated. + Parameters + ---------- + nx : int + Number of cells in the x direction. + ny : int + Number of cells in the y direction. + nz : int + Number of cells in the z direction. + Lx : float + Extent in the x direction. + Ly : float + Extent in the y direction. + Lz : float + Extent in the z direction. + directions : list or tuple or None + Directions of periodicity; if `None`, the mesh is made periodic in all directions. + hexahedral : bool + Whether to make hexahedral mesh or not. + reorder : bool or None + Whether to reorder the mesh. + distribution_parameters : dict or None + Options controlling mesh distribution, see :func:`.Mesh` for details. + comm : + Communicator to build the mesh on. + name : str + Name of the mesh. + distribution_name : str or None + Name of parallel distribution used when checkpointing; + if `None`, the name is automatically generated. + permutation_name : str or None + Name of entity permutation (reordering) used when checkpointing; + if `None`, the name is automatically generated. + + Returns + ------- + MeshGeometry + The mesh. + """ for n in (nx, ny, nz): if n < 3: raise ValueError( "3D periodic meshes with fewer than 3 cells are not currently supported" ) + if hexahedral: + if directions is None: + directions = [True, True, True] + if len(directions) != 3: + raise ValueError(f"directions must have exactly dim ( = 3) elements : Got {directions}") + plex = PETSc.DMPlex().createBoxMesh((nx, ny, nz), lower=(0., 0., 0.), upper=(Lx, Ly, Lz), simplex=False, periodic=directions, interpolate=True, sparseLocalize=False, comm=comm) + _firedrake_box_mesh_hexahedral_mark_boundaries(plex, nx, ny, nz, Lx, Ly, Lz) + return mesh.Mesh( + plex, + reorder=reorder, + distribution_parameters=distribution_parameters, + name=name, + distribution_name=distribution_name, + permutation_name=permutation_name, + comm=comm) + else: + if directions is not None: + raise NotImplementedError("Can only specify directions with hexahedral = True") + xcoords = np.arange(0.0, Lx, Lx / nx, dtype=np.double) + ycoords = np.arange(0.0, Ly, Ly / ny, dtype=np.double) + zcoords = np.arange(0.0, Lz, Lz / nz, dtype=np.double) + coords = ( + np.asarray(np.meshgrid(xcoords, ycoords, zcoords)).swapaxes(0, 3).reshape(-1, 3) + ) + i, j, k = np.meshgrid( + np.arange(nx, dtype=np.int32), + np.arange(ny, dtype=np.int32), + np.arange(nz, dtype=np.int32), + ) + v0 = k * nx * ny + j * nx + i + v1 = k * nx * ny + j * nx + (i + 1) % nx + v2 = k * nx * ny + ((j + 1) % ny) * nx + i + v3 = k * nx * ny + ((j + 1) % ny) * nx + (i + 1) % nx + v4 = ((k + 1) % nz) * nx * ny + j * nx + i + v5 = ((k + 1) % nz) * nx * ny + j * nx + (i + 1) % nx + v6 = ((k + 1) % nz) * nx * ny + ((j + 1) % ny) * nx + i + v7 = ((k + 1) % nz) * nx * ny + ((j + 1) % ny) * nx + (i + 1) % nx - xcoords = np.arange(0.0, Lx, Lx / nx, dtype=np.double) - ycoords = np.arange(0.0, Ly, Ly / ny, dtype=np.double) - zcoords = np.arange(0.0, Lz, Lz / nz, dtype=np.double) - coords = ( - np.asarray(np.meshgrid(xcoords, ycoords, zcoords)).swapaxes(0, 3).reshape(-1, 3) - ) - i, j, k = np.meshgrid( - np.arange(nx, dtype=np.int32), - np.arange(ny, dtype=np.int32), - np.arange(nz, dtype=np.int32), - ) - v0 = k * nx * ny + j * nx + i - v1 = k * nx * ny + j * nx + (i + 1) % nx - v2 = k * nx * ny + ((j + 1) % ny) * nx + i - v3 = k * nx * ny + ((j + 1) % ny) * nx + (i + 1) % nx - v4 = ((k + 1) % nz) * nx * ny + j * nx + i - v5 = ((k + 1) % nz) * nx * ny + j * nx + (i + 1) % nx - v6 = ((k + 1) % nz) * nx * ny + ((j + 1) % ny) * nx + i - v7 = ((k + 1) % nz) * nx * ny + ((j + 1) % ny) * nx + (i + 1) % nx - - cells = [ - [v0, v1, v3, v7], - [v0, v1, v7, v5], - [v0, v5, v7, v4], - [v0, v3, v2, v7], - [v0, v6, v4, v7], - [v0, v2, v6, v7], - ] - cells = np.asarray(cells).reshape(-1, ny, nx, nz).swapaxes(0, 3).reshape(-1, 4) - plex = mesh.plex_from_cell_list( - 3, cells, coords, comm, mesh._generate_default_mesh_topology_name(name) - ) - m = mesh.Mesh( - plex, - reorder=reorder_noop, - distribution_parameters=distribution_parameters_no_overlap, - name=name, - distribution_name=distribution_name, - permutation_name=permutation_name, - comm=comm, - ) - - old_coordinates = m.coordinates - new_coordinates = Function( - VectorFunctionSpace( - m, FiniteElement("DG", tetrahedron, 1, variant="equispaced") - ), - name=mesh._generate_default_mesh_coordinates_name(name), - ) + cells = [ + [v0, v1, v3, v7], + [v0, v1, v7, v5], + [v0, v5, v7, v4], + [v0, v3, v2, v7], + [v0, v6, v4, v7], + [v0, v2, v6, v7], + ] + cells = np.asarray(cells).reshape(-1, ny, nx, nz).swapaxes(0, 3).reshape(-1, 4) + plex = mesh.plex_from_cell_list( + 3, cells, coords, comm, mesh._generate_default_mesh_topology_name(name) + ) + m = mesh.Mesh( + plex, + reorder=reorder_noop, + distribution_parameters=distribution_parameters_no_overlap, + name=name, + distribution_name=distribution_name, + permutation_name=permutation_name, + comm=comm, + ) - domain = "" - instructions = f""" - <{RealType}> x0 = real(old_coords[0, 0]) - <{RealType}> x1 = real(old_coords[1, 0]) - <{RealType}> x2 = real(old_coords[2, 0]) - <{RealType}> x3 = real(old_coords[3, 0]) - <{RealType}> x_max = fmax(fmax(fmax(x0, x1), x2), x3) - <{RealType}> y0 = real(old_coords[0, 1]) - <{RealType}> y1 = real(old_coords[1, 1]) - <{RealType}> y2 = real(old_coords[2, 1]) - <{RealType}> y3 = real(old_coords[3, 1]) - <{RealType}> y_max = fmax(fmax(fmax(y0, y1), y2), y3) - <{RealType}> z0 = real(old_coords[0, 2]) - <{RealType}> z1 = real(old_coords[1, 2]) - <{RealType}> z2 = real(old_coords[2, 2]) - <{RealType}> z3 = real(old_coords[3, 2]) - <{RealType}> z_max = fmax(fmax(fmax(z0, z1), z2), z3) - - new_coords[0, 0] = x_max+hx[0] if (x_max > real(1.5*hx[0]) and old_coords[0, 0] == 0.) else old_coords[0, 0] - new_coords[0, 1] = y_max+hy[0] if (y_max > real(1.5*hy[0]) and old_coords[0, 1] == 0.) else old_coords[0, 1] - new_coords[0, 2] = z_max+hz[0] if (z_max > real(1.5*hz[0]) and old_coords[0, 2] == 0.) else old_coords[0, 2] - - new_coords[1, 0] = x_max+hx[0] if (x_max > real(1.5*hx[0]) and old_coords[1, 0] == 0.) else old_coords[1, 0] - new_coords[1, 1] = y_max+hy[0] if (y_max > real(1.5*hy[0]) and old_coords[1, 1] == 0.) else old_coords[1, 1] - new_coords[1, 2] = z_max+hz[0] if (z_max > real(1.5*hz[0]) and old_coords[1, 2] == 0.) else old_coords[1, 2] - - new_coords[2, 0] = x_max+hx[0] if (x_max > real(1.5*hx[0]) and old_coords[2, 0] == 0.) else old_coords[2, 0] - new_coords[2, 1] = y_max+hy[0] if (y_max > real(1.5*hy[0]) and old_coords[2, 1] == 0.) else old_coords[2, 1] - new_coords[2, 2] = z_max+hz[0] if (z_max > real(1.5*hz[0]) and old_coords[2, 2] == 0.) else old_coords[2, 2] - - new_coords[3, 0] = x_max+hx[0] if (x_max > real(1.5*hx[0]) and old_coords[3, 0] == 0.) else old_coords[3, 0] - new_coords[3, 1] = y_max+hy[0] if (y_max > real(1.5*hy[0]) and old_coords[3, 1] == 0.) else old_coords[3, 1] - new_coords[3, 2] = z_max+hz[0] if (z_max > real(1.5*hz[0]) and old_coords[3, 2] == 0.) else old_coords[3, 2] - """ - hx = Constant(Lx / nx) - hy = Constant(Ly / ny) - hz = Constant(Lz / nz) + old_coordinates = m.coordinates + new_coordinates = Function( + VectorFunctionSpace( + m, FiniteElement("DG", tetrahedron, 1, variant="equispaced") + ), + name=mesh._generate_default_mesh_coordinates_name(name), + ) - par_loop( - (domain, instructions), - dx, - { - "new_coords": (new_coordinates, WRITE), - "old_coords": (old_coordinates, READ), - "hx": (hx, READ), - "hy": (hy, READ), - "hz": (hz, READ), - }, - ) - return _postprocess_periodic_mesh(new_coordinates, - comm, - distribution_parameters, - reorder, - name, - distribution_name, - permutation_name) + domain = "" + instructions = f""" + <{RealType}> x0 = real(old_coords[0, 0]) + <{RealType}> x1 = real(old_coords[1, 0]) + <{RealType}> x2 = real(old_coords[2, 0]) + <{RealType}> x3 = real(old_coords[3, 0]) + <{RealType}> x_max = fmax(fmax(fmax(x0, x1), x2), x3) + <{RealType}> y0 = real(old_coords[0, 1]) + <{RealType}> y1 = real(old_coords[1, 1]) + <{RealType}> y2 = real(old_coords[2, 1]) + <{RealType}> y3 = real(old_coords[3, 1]) + <{RealType}> y_max = fmax(fmax(fmax(y0, y1), y2), y3) + <{RealType}> z0 = real(old_coords[0, 2]) + <{RealType}> z1 = real(old_coords[1, 2]) + <{RealType}> z2 = real(old_coords[2, 2]) + <{RealType}> z3 = real(old_coords[3, 2]) + <{RealType}> z_max = fmax(fmax(fmax(z0, z1), z2), z3) + + new_coords[0, 0] = x_max+hx[0] if (x_max > real(1.5*hx[0]) and old_coords[0, 0] == 0.) else old_coords[0, 0] + new_coords[0, 1] = y_max+hy[0] if (y_max > real(1.5*hy[0]) and old_coords[0, 1] == 0.) else old_coords[0, 1] + new_coords[0, 2] = z_max+hz[0] if (z_max > real(1.5*hz[0]) and old_coords[0, 2] == 0.) else old_coords[0, 2] + + new_coords[1, 0] = x_max+hx[0] if (x_max > real(1.5*hx[0]) and old_coords[1, 0] == 0.) else old_coords[1, 0] + new_coords[1, 1] = y_max+hy[0] if (y_max > real(1.5*hy[0]) and old_coords[1, 1] == 0.) else old_coords[1, 1] + new_coords[1, 2] = z_max+hz[0] if (z_max > real(1.5*hz[0]) and old_coords[1, 2] == 0.) else old_coords[1, 2] + + new_coords[2, 0] = x_max+hx[0] if (x_max > real(1.5*hx[0]) and old_coords[2, 0] == 0.) else old_coords[2, 0] + new_coords[2, 1] = y_max+hy[0] if (y_max > real(1.5*hy[0]) and old_coords[2, 1] == 0.) else old_coords[2, 1] + new_coords[2, 2] = z_max+hz[0] if (z_max > real(1.5*hz[0]) and old_coords[2, 2] == 0.) else old_coords[2, 2] + + new_coords[3, 0] = x_max+hx[0] if (x_max > real(1.5*hx[0]) and old_coords[3, 0] == 0.) else old_coords[3, 0] + new_coords[3, 1] = y_max+hy[0] if (y_max > real(1.5*hy[0]) and old_coords[3, 1] == 0.) else old_coords[3, 1] + new_coords[3, 2] = z_max+hz[0] if (z_max > real(1.5*hz[0]) and old_coords[3, 2] == 0.) else old_coords[3, 2] + """ + hx = Constant(Lx / nx) + hy = Constant(Ly / ny) + hz = Constant(Lz / nz) + + par_loop( + (domain, instructions), + dx, + { + "new_coords": (new_coordinates, WRITE), + "old_coords": (old_coordinates, READ), + "hx": (hx, READ), + "hy": (hy, READ), + "hz": (hz, READ), + }, + ) + return _postprocess_periodic_mesh(new_coordinates, + comm, + distribution_parameters, + reorder, + name, + distribution_name, + permutation_name) @PETSc.Log.EventDecorator() @@ -1983,6 +2024,8 @@ def PeriodicUnitCubeMesh( nx, ny, nz, + directions=None, + hexahedral=False, reorder=None, distribution_parameters=None, comm=COMM_WORLD, @@ -1992,20 +2035,38 @@ def PeriodicUnitCubeMesh( ): """Generate a periodic mesh of a unit cube - :arg nx: The number of cells in the x direction - :arg ny: The number of cells in the y direction - :arg nz: The number of cells in the z direction - :kwarg reorder: (optional), should the mesh be reordered? - :kwarg distribution_parameters: options controlling mesh - distribution, see :func:`.Mesh` for details. - :kwarg comm: Optional communicator to build the mesh on. - :kwarg name: Optional name of the mesh. - :kwarg distribution_name: the name of parallel distribution used - when checkpointing; if `None`, the name is automatically - generated. - :kwarg permutation_name: the name of entity permutation (reordering) used - when checkpointing; if `None`, the name is automatically - generated. + Parameters + ---------- + nx : int + Number of cells in the x direction. + ny : int + Number of cells in the y direction. + nz : int + Number of cells in the z direction. + directions : list or tuple or None + Directions of periodicity; if `None`, the mesh is made periodic in all directions. + hexahedral : bool + Whether to make hexahedral mesh or not. + reorder : bool or None + Should the mesh be reordered? + distribution_parameters : dict or None + Options controlling mesh distribution, see :func:`.Mesh` for details. + comm : + Communicator to build the mesh on. + name : str + Name of the mesh. + distribution_name : str or None + Name of parallel distribution used when checkpointing; + if `None`, the name is automatically generated. + permutation_name : str or None + Name of entity permutation (reordering) used when checkpointing; + if `None`, the name is automatically generated. + + Returns + ------- + MeshGeometry + The mesh. + """ return PeriodicBoxMesh( nx, @@ -2014,6 +2075,8 @@ def PeriodicUnitCubeMesh( 1.0, 1.0, 1.0, + directions=directions, + hexahedral=hexahedral, reorder=reorder, distribution_parameters=distribution_parameters, comm=comm, diff --git a/tests/regression/test_mesh_generation.py b/tests/regression/test_mesh_generation.py index 637b0a1502..5e9ce27652 100644 --- a/tests/regression/test_mesh_generation.py +++ b/tests/regression/test_mesh_generation.py @@ -476,6 +476,17 @@ def test_boxmesh_kind(kind, num_cells): assert m.num_cells() == num_cells +@pytest.mark.parallel(nprocs=2) +def test_periodic_unit_cube_hex(): + mesh = PeriodicUnitCubeMesh(8, 8, 8, directions=[True, True, False], hexahedral=True) + x, y, z = SpatialCoordinate(mesh) + V = FunctionSpace(mesh, "CG", 3) + expr = (1 - x) * x + (1 - y) * y + z + f = Function(V).interpolate(expr) + error = assemble((f - expr) ** 2 * dx) + assert error < 1.e-30 + + @pytest.mark.parallel(nprocs=4) def test_split_comm_dm_mesh(): nspace = 2 From 2f847b8af603c8e432bb616f7b3edef19c05c6d1 Mon Sep 17 00:00:00 2001 From: ksagiyam Date: Fri, 5 Jul 2024 17:09:02 +0100 Subject: [PATCH 2/2] DROP BEFORE MERGE --- .github/workflows/build.yml | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index 4f61ab2d9b..2e4a4bc841 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -56,12 +56,12 @@ jobs: rm -rf firedrake_venv - name: Build Firedrake run: | + unset PETSC_DIR SLEPC_DIR cd .. # Linting should ignore unquoted shell variable $COMPLEX # shellcheck disable=SC2086 ./firedrake/scripts/firedrake-install \ $COMPLEX \ - --honour-petsc-dir \ --mpicc="$MPICH_DIR"/mpicc \ --mpicxx="$MPICH_DIR"/mpicxx \ --mpif90="$MPICH_DIR"/mpif90 \ @@ -82,6 +82,7 @@ jobs: --install defcon \ --install gadopt \ --install asQ \ + --package-branch petsc ksagiyam/merge_upstream \ || (cat firedrake-install.log && /bin/false) - name: Install test dependencies run: | @@ -92,6 +93,7 @@ jobs: python -m pip list - name: Test Firedrake run: | + unset PETSC_DIR SLEPC_DIR . ../firedrake_venv/bin/activate echo OMP_NUM_THREADS is "$OMP_NUM_THREADS" echo OPENBLAS_NUM_THREADS is "$OPENBLAS_NUM_THREADS" @@ -107,6 +109,7 @@ jobs: - name: Test pyadjoint if: ${{ matrix.scalar-type == 'real' }} run: | + unset PETSC_DIR SLEPC_DIR . ../firedrake_venv/bin/activate cd ../firedrake_venv/src/pyadjoint python -m pytest \