Source code for ufl.cell

"""Types for representing a cell."""

# Copyright (C) 2008-2016 Martin Sandve Alnæs
#
# This file is part of UFL (https://www.fenicsproject.org)
#
# SPDX-License-Identifier:    LGPL-3.0-or-later

from __future__ import annotations
import functools
import numbers
import typing
import weakref

from ufl.core.ufl_type import UFLObject
from abc import abstractmethod


__all_classes__ = ["AbstractCell", "Cell", "TensorProductCell"]


[docs]class AbstractCell(UFLObject): """A base class for all cells."""
[docs] @abstractmethod def topological_dimension(self) -> int: """Return the dimension of the topology of this cell."""
[docs] @abstractmethod def geometric_dimension(self) -> int: """Return the dimension of the geometry of this cell."""
[docs] @abstractmethod def is_simplex(self) -> bool: """Return True if this is a simplex cell."""
[docs] @abstractmethod def has_simplex_facets(self) -> bool: """Return True if all the facets of this cell are simplex cells."""
@abstractmethod def _lt(self, other) -> bool: """Define an arbitrarily chosen but fixed sort order for all instances of this type with the same dimensions."""
[docs] @abstractmethod def num_sub_entities(self, dim: int) -> int: """Get the number of sub-entities of the given dimension."""
[docs] @abstractmethod def sub_entities(self, dim: int) -> typing.Tuple[AbstractCell, ...]: """Get the sub-entities of the given dimension."""
[docs] @abstractmethod def sub_entity_types(self, dim: int) -> typing.Tuple[AbstractCell, ...]: """Get the unique sub-entity types of the given dimension."""
[docs] @abstractmethod def cellname(self) -> str: """Return the cellname of the cell."""
[docs] @abstractmethod def reconstruct(self, **kwargs: typing.Any) -> Cell: """Reconstruct this cell, overwriting properties by those in kwargs."""
def __lt__(self, other: AbstractCell) -> bool: """Define an arbitrarily chosen but fixed sort order for all cells.""" if type(self) is type(other): s = (self.geometric_dimension(), self.topological_dimension()) o = (other.geometric_dimension(), other.topological_dimension()) if s != o: return s < o return self._lt(other) else: if type(self).__name__ == type(other).__name__: raise ValueError("Cannot order cell types with the same name") return type(self).__name__ < type(other).__name__
[docs] def num_vertices(self) -> int: """Get the number of vertices.""" return self.num_sub_entities(0)
[docs] def num_edges(self) -> int: """Get the number of edges.""" return self.num_sub_entities(1)
[docs] def num_faces(self) -> int: """Get the number of faces.""" return self.num_sub_entities(2)
[docs] def num_facets(self) -> int: """Get the number of facets. Facets are entities of dimension tdim-1. """ tdim = self.topological_dimension() return self.num_sub_entities(tdim - 1)
[docs] def num_ridges(self) -> int: """Get the number of ridges. Ridges are entities of dimension tdim-2. """ tdim = self.topological_dimension() return self.num_sub_entities(tdim - 2)
[docs] def num_peaks(self) -> int: """Get the number of peaks. Peaks are entities of dimension tdim-3. """ tdim = self.topological_dimension() return self.num_sub_entities(tdim - 3)
[docs] def vertices(self) -> typing.Tuple[AbstractCell, ...]: """Get the vertices.""" return self.sub_entities(0)
[docs] def edges(self) -> typing.Tuple[AbstractCell, ...]: """Get the edges.""" return self.sub_entities(1)
[docs] def faces(self) -> typing.Tuple[AbstractCell, ...]: """Get the faces.""" return self.sub_entities(2)
[docs] def facets(self) -> typing.Tuple[AbstractCell, ...]: """Get the facets. Facets are entities of dimension tdim-1. """ tdim = self.topological_dimension() return self.sub_entities(tdim - 1)
[docs] def ridges(self) -> typing.Tuple[AbstractCell, ...]: """Get the ridges. Ridges are entities of dimension tdim-2. """ tdim = self.topological_dimension() return self.sub_entities(tdim - 2)
[docs] def peaks(self) -> typing.Tuple[AbstractCell, ...]: """Get the peaks. Peaks are entities of dimension tdim-3. """ tdim = self.topological_dimension() return self.sub_entities(tdim - 3)
[docs] def vertex_types(self) -> typing.Tuple[AbstractCell, ...]: """Get the unique vertices types.""" return self.sub_entity_types(0)
[docs] def edge_types(self) -> typing.Tuple[AbstractCell, ...]: """Get the unique edge types.""" return self.sub_entity_types(1)
[docs] def face_types(self) -> typing.Tuple[AbstractCell, ...]: """Get the unique face types.""" return self.sub_entity_types(2)
[docs] def facet_types(self) -> typing.Tuple[AbstractCell, ...]: """Get the unique facet types. Facets are entities of dimension tdim-1. """ tdim = self.topological_dimension() return self.sub_entity_types(tdim - 1)
[docs] def ridge_types(self) -> typing.Tuple[AbstractCell, ...]: """Get the unique ridge types. Ridges are entities of dimension tdim-2. """ tdim = self.topological_dimension() return self.sub_entity_types(tdim - 2)
[docs] def peak_types(self) -> typing.Tuple[AbstractCell, ...]: """Get the unique peak types. Peaks are entities of dimension tdim-3. """ tdim = self.topological_dimension() return self.sub_entity_types(tdim - 3)
_sub_entity_celltypes = { "vertex": [("vertex", )], "interval": [tuple("vertex" for i in range(2)), ("interval", )], "triangle": [tuple("vertex" for i in range(3)), tuple("interval" for i in range(3)), ("triangle", )], "quadrilateral": [tuple("vertex" for i in range(4)), tuple("interval" for i in range(4)), ("quadrilateral", )], "tetrahedron": [tuple("vertex" for i in range(4)), tuple("interval" for i in range(6)), tuple("triangle" for i in range(4)), ("tetrahedron", )], "hexahedron": [tuple("vertex" for i in range(8)), tuple("interval" for i in range(12)), tuple("quadrilateral" for i in range(6)), ("hexahedron", )], "prism": [tuple("vertex" for i in range(6)), tuple("interval" for i in range(9)), ("triangle", "quadrilateral", "quadrilateral", "quadrilateral", "triangle"), ("prism", )], "pyramid": [tuple("vertex" for i in range(5)), tuple("interval" for i in range(8)), ("quadrilateral", "triangle", "triangle", "triangle", "triangle"), ("pyramid", )], "pentatope": [tuple("vertex" for i in range(5)), tuple("interval" for i in range(10)), tuple("triangle" for i in range(10)), tuple("tetrahedron" for i in range(5)), ("pentatope", )], "tesseract": [tuple("vertex" for i in range(16)), tuple("interval" for i in range(32)), tuple("quadrilateral" for i in range(24)), tuple("hexahedron" for i in range(8)), ("tesseract", )], }
[docs]class Cell(AbstractCell): """Representation of a named finite element cell with known structure.""" __slots__ = ("_cellname", "_tdim", "_gdim", "_num_cell_entities", "_sub_entity_types", "_sub_entities", "_sub_entity_types") def __init__(self, cellname: str, geometric_dimension: typing.Optional[int] = None): """Initialise. Args: cellname: Name of the cell geometric_dimension: Geometric dimension """ if cellname not in _sub_entity_celltypes: raise ValueError(f"Unsupported cell type: {cellname}") self._sub_entity_celltypes = _sub_entity_celltypes[cellname] self._cellname = cellname self._tdim = len(self._sub_entity_celltypes) - 1 self._gdim = self._tdim if geometric_dimension is None else geometric_dimension self._num_cell_entities = [len(i) for i in self._sub_entity_celltypes] self._sub_entities = [tuple(Cell(t, self._gdim) for t in se_types) for se_types in self._sub_entity_celltypes[:-1]] self._sub_entity_types = [tuple(set(i)) for i in self._sub_entities] self._sub_entities.append((weakref.proxy(self), )) self._sub_entity_types.append((weakref.proxy(self), )) if not isinstance(self._gdim, numbers.Integral): raise ValueError("Expecting integer geometric_dimension.") if not isinstance(self._tdim, numbers.Integral): raise ValueError("Expecting integer topological_dimension.") if self._tdim > self._gdim: raise ValueError("Topological dimension cannot be larger than geometric dimension.")
[docs] def topological_dimension(self) -> int: """Return the dimension of the topology of this cell.""" return self._tdim
[docs] def geometric_dimension(self) -> int: """Return the dimension of the geometry of this cell.""" return self._gdim
[docs] def is_simplex(self) -> bool: """Return True if this is a simplex cell.""" return self._cellname in ["vertex", "interval", "triangle", "tetrahedron"]
[docs] def has_simplex_facets(self) -> bool: """Return True if all the facets of this cell are simplex cells.""" return self._cellname in ["interval", "triangle", "quadrilateral", "tetrahedron"]
[docs] def num_sub_entities(self, dim: int) -> int: """Get the number of sub-entities of the given dimension.""" if dim < 0: return 0 try: return self._num_cell_entities[dim] except IndexError: return 0
[docs] def sub_entities(self, dim: int) -> typing.Tuple[AbstractCell, ...]: """Get the sub-entities of the given dimension.""" if dim < 0: return () try: return self._sub_entities[dim] except IndexError: return ()
[docs] def sub_entity_types(self, dim: int) -> typing.Tuple[AbstractCell, ...]: """Get the unique sub-entity types of the given dimension.""" if dim < 0: return () try: return self._sub_entity_types[dim] except IndexError: return ()
def _lt(self, other) -> bool: return self._cellname < other._cellname
[docs] def cellname(self) -> str: """Return the cellname of the cell.""" return self._cellname
def __str__(self) -> str: """Format as a string.""" if self._gdim == self._tdim: return self._cellname else: return f"{self._cellname}{self._gdim}D" def __repr__(self) -> str: """Representation.""" if self._gdim == self._tdim: return self._cellname else: return f"Cell({self._cellname}, {self._gdim})" def _ufl_hash_data_(self) -> typing.Hashable: """UFL hash data.""" return (self._cellname, self._gdim)
[docs] def reconstruct(self, **kwargs: typing.Any) -> Cell: """Reconstruct this cell, overwriting properties by those in kwargs.""" gdim = self._gdim for key, value in kwargs.items(): if key == "geometric_dimension": gdim = value else: raise TypeError(f"reconstruct() got unexpected keyword argument '{key}'") return Cell(self._cellname, geometric_dimension=gdim)
[docs]class TensorProductCell(AbstractCell): """Tensor product cell.""" __slots__ = ("_cells", "_tdim", "_gdim") def __init__(self, *cells: Cell, geometric_dimension: typing.Optional[int] = None): """Initialise. Args: cells: Cells to take the tensor product of geometric_dimension: Geometric dimension """ self._cells = tuple(as_cell(cell) for cell in cells) self._tdim = sum([cell.topological_dimension() for cell in self._cells]) self._gdim = self._tdim if geometric_dimension is None else geometric_dimension if not isinstance(self._gdim, numbers.Integral): raise ValueError("Expecting integer geometric_dimension.") if not isinstance(self._tdim, numbers.Integral): raise ValueError("Expecting integer topological_dimension.") if self._tdim > self._gdim: raise ValueError("Topological dimension cannot be larger than geometric dimension.")
[docs] def sub_cells(self) -> typing.List[AbstractCell]: """Return list of cell factors.""" return self._cells
[docs] def topological_dimension(self) -> int: """Return the dimension of the topology of this cell.""" return self._tdim
[docs] def geometric_dimension(self) -> int: """Return the dimension of the geometry of this cell.""" return self._gdim
[docs] def is_simplex(self) -> bool: """Return True if this is a simplex cell.""" if len(self._cells) == 1: return self._cells[0].is_simplex() return False
[docs] def has_simplex_facets(self) -> bool: """Return True if all the facets of this cell are simplex cells.""" if len(self._cells) == 1: return self._cells[0].has_simplex_facets() if self._tdim == 1: return True return False
[docs] def num_sub_entities(self, dim: int) -> int: """Get the number of sub-entities of the given dimension.""" if dim < 0 or dim > self._tdim: return 0 if dim == 0: return functools.reduce(lambda x, y: x * y, [c.num_vertices() for c in self._cells]) if dim == self._tdim - 1: # Note: This is not the number of facets that the cell has, but I'm leaving it here for now # to not change past behaviour return sum(c.num_facets() for c in self._cells if c.topological_dimension() > 0) if dim == self._tdim: return 1 raise NotImplementedError(f"TensorProductCell.num_sub_entities({dim}) is not implemented.")
[docs] def sub_entities(self, dim: int) -> typing.Tuple[AbstractCell, ...]: """Get the sub-entities of the given dimension.""" if dim < 0 or dim > self._tdim: return [] if dim == 0: return [Cell("vertex", self._gdim) for i in range(self.num_sub_entities(0))] if dim == self._tdim: return [self] raise NotImplementedError(f"TensorProductCell.sub_entities({dim}) is not implemented.")
[docs] def sub_entity_types(self, dim: int) -> typing.Tuple[AbstractCell, ...]: """Get the unique sub-entity types of the given dimension.""" if dim < 0 or dim > self._tdim: return [] if dim == 0: return [Cell("vertex", self._gdim)] if dim == self._tdim: return [self] raise NotImplementedError(f"TensorProductCell.sub_entities({dim}) is not implemented.")
def _lt(self, other) -> bool: return self._ufl_hash_data_() < other._ufl_hash_data_()
[docs] def cellname(self) -> str: """Return the cellname of the cell.""" return " * ".join([cell.cellname() for cell in self._cells])
def __str__(self) -> str: """Format as a string.""" s = "TensorProductCell(" s += ", ".join(f"{c!r}" for c in self._cells) if self._tdim != self._gdim: s += f", geometric_dimension={self._gdim}" s += ")" return s def __repr__(self) -> str: """Representation.""" return str(self) def _ufl_hash_data_(self) -> typing.Hashable: """UFL hash data.""" return tuple(c._ufl_hash_data_() for c in self._cells) + (self._gdim,)
[docs] def reconstruct(self, **kwargs: typing.Any) -> Cell: """Reconstruct this cell, overwriting properties by those in kwargs.""" gdim = self._gdim for key, value in kwargs.items(): if key == "geometric_dimension": gdim = value else: raise TypeError(f"reconstruct() got unexpected keyword argument '{key}'") return TensorProductCell(*self._cells, geometric_dimension=gdim)
[docs]def simplex(topological_dimension: int, geometric_dimension: typing.Optional[int] = None): """Return a simplex cell of the given dimension.""" if topological_dimension == 0: return Cell("vertex", geometric_dimension) if topological_dimension == 1: return Cell("interval", geometric_dimension) if topological_dimension == 2: return Cell("triangle", geometric_dimension) if topological_dimension == 3: return Cell("tetrahedron", geometric_dimension) if topological_dimension == 4: return Cell("pentatope", geometric_dimension) raise ValueError(f"Unsupported topological dimension for simplex: {topological_dimension}")
[docs]def hypercube(topological_dimension, geometric_dimension=None): """Return a hypercube cell of the given dimension.""" if topological_dimension == 0: return Cell("vertex", geometric_dimension) if topological_dimension == 1: return Cell("interval", geometric_dimension) if topological_dimension == 2: return Cell("quadrilateral", geometric_dimension) if topological_dimension == 3: return Cell("hexahedron", geometric_dimension) if topological_dimension == 4: return Cell("tesseract", geometric_dimension) raise ValueError(f"Unsupported topological dimension for hypercube: {topological_dimension}")
[docs]def as_cell(cell: typing.Union[AbstractCell, str, typing.Tuple[AbstractCell, ...]]) -> AbstractCell: """Convert any valid object to a Cell or return cell if it is already a Cell. Allows an already valid cell, a known cellname string, or a tuple of cells for a product cell. """ if isinstance(cell, AbstractCell): return cell elif isinstance(cell, str): return Cell(cell) elif isinstance(cell, tuple): return TensorProductCell(cell) else: raise ValueError(f"Invalid cell {cell}.")