Source code for firedrake.constant

import numpy as np
import ufl
import finat.ufl

from tsfc.ufl_utils import TSFCConstantMixin
from pyop2 import op2
from pyop2.exceptions import DataTypeError, DataValueError
from firedrake.petsc import PETSc
from firedrake.utils import ScalarType
from ufl.classes import all_ufl_classes, ufl_classes, terminal_classes
from ufl.core.ufl_type import UFLType
from ufl.corealg.multifunction import MultiFunction
from ufl.formatting.ufl2unicode import (
    Expression2UnicodeHandler, UC, subscript_number, PrecedenceRules,
    colorama,
)
from ufl.utils.counted import Counted


import firedrake.utils as utils
from firedrake.adjoint_utils.constant import ConstantMixin


__all__ = ['Constant']


def _create_dat(op2type, value, comm):
    if op2type is op2.Global and comm is None:
        raise ValueError("Attempted to create pyop2 Global with no communicator")

    data = np.array(value, dtype=ScalarType)
    shape = data.shape
    rank = len(shape)
    if rank == 0:
        dat = op2type(1, data, comm=comm)
    else:
        dat = op2type(shape, data, comm=comm)
    return dat, rank, shape


[docs] class Constant(ufl.constantvalue.ConstantValue, ConstantMixin, TSFCConstantMixin, Counted): """A "constant" coefficient A :class:`Constant` takes one value over the whole :func:`~.Mesh`. The advantage of using a :class:`Constant` in a form rather than a literal value is that the constant will be passed as an argument to the generated kernel which avoids the need to recompile the kernel if the form is assembled for a different value of the constant. :arg value: the value of the constant. May either be a scalar, an iterable of values (for a vector-valued constant), or an iterable of iterables (or numpy array with 2-dimensional shape) for a tensor-valued constant. :arg domain: an optional :func:`~.Mesh` on which the constant is defined. .. note:: If you intend to use this :class:`Constant` in a :class:`~ufl.form.Form` on its own you need to pass a :func:`~.Mesh` as the domain argument. """ _ufl_typecode_ = UFLType._ufl_num_typecodes_ _ufl_handler_name_ = "firedrake_constant" def __new__(cls, value, domain=None, name=None, count=None): if domain: # Avoid circular import from firedrake.function import Function from firedrake.functionspace import FunctionSpace import warnings warnings.warn( "Giving Constants a domain is no longer supported. Instead please " "create a Function in the Real space.", FutureWarning ) dat, rank, shape = _create_dat(op2.Global, value, domain._comm) if not isinstance(domain, ufl.AbstractDomain): cell = ufl.as_cell(domain) coordinate_element = finat.ufl.VectorElement("Lagrange", cell, 1, dim=cell.topological_dimension()) domain = ufl.Mesh(coordinate_element) cell = domain.ufl_cell() if rank == 0: element = finat.ufl.FiniteElement("R", cell, 0) elif rank == 1: element = finat.ufl.VectorElement("R", cell, 0, shape[0]) else: element = finat.ufl.TensorElement("R", cell, 0, shape=shape) R = FunctionSpace(domain, element, name="firedrake.Constant") return Function(R, val=dat).assign(value) else: return object.__new__(cls) @ConstantMixin._ad_annotate_init def __init__(self, value, domain=None, name=None, count=None): """""" # Init also called in mesh constructor, but constant can be built without mesh utils._init() self.dat, rank, self._ufl_shape = _create_dat(op2.Constant, value, None) super().__init__() Counted.__init__(self, count, Counted) self.name = name or f"constant_{self._count}" def __repr__(self): return f"Constant({self.dat.data_ro}, name='{self.name}', count={self._count})" def _ufl_signature_data_(self, renumbering): return (type(self).__name__, renumbering[self]) def __hash__(self): return hash((type(self), self.count())) def __eq__(self, other): return type(self) == type(other) and self.count() == other.count() @property def ufl_shape(self): return self._ufl_shape
[docs] def count(self): return self._count
[docs] @PETSc.Log.EventDecorator() def evaluate(self, x, mapping, component, index_values): """Return the evaluation of this :class:`Constant`. :arg x: The coordinate to evaluate at (ignored). :arg mapping: A mapping (ignored). :arg component: The requested component of the constant (may be ``None`` or ``()`` to obtain all components). :arg index_values: ignored. """ if component in ((), None): if self.ufl_shape == (): return self.dat.data_ro[0] return self.dat.data_ro return self.dat.data_ro[component]
[docs] def values(self): """Return a (flat) view of the value of the Constant.""" return self.dat.data_ro.reshape(-1)
[docs] def function_space(self): """Return a null function space.""" return None
@utils.cached_property def subfunctions(self): return (self,)
[docs] def split(self): import warnings warnings.warn("The .split() method is deprecated, please use the .subfunctions property instead", category=FutureWarning) return self.subfunctions
[docs] def cell_node_map(self, bcs=None): """Return a null cell to node map.""" if bcs is not None: raise RuntimeError("Can't apply boundary conditions to a Constant") return None
[docs] def interior_facet_node_map(self, bcs=None): """Return a null interior facet to node map.""" if bcs is not None: raise RuntimeError("Can't apply boundary conditions to a Constant") return None
[docs] def exterior_facet_node_map(self, bcs=None): """Return a null exterior facet to node map.""" if bcs is not None: raise RuntimeError("Can't apply boundary conditions to a Constant") return None
[docs] @PETSc.Log.EventDecorator() @ConstantMixin._ad_annotate_assign def assign(self, value): """Set the value of this constant. :arg value: A value of the appropriate shape""" try: self.dat.data = value return self except (DataTypeError, DataValueError) as e: raise ValueError(e)
def __iadd__(self, o): raise NotImplementedError("Augmented assignment to Constant not implemented") def __isub__(self, o): raise NotImplementedError("Augmented assignment to Constant not implemented") def __imul__(self, o): raise NotImplementedError("Augmented assignment to Constant not implemented") def __itruediv__(self, o): raise NotImplementedError("Augmented assignment to Constant not implemented") def __str__(self): return str(self.dat.data_ro)
# Unicode handler for Firedrake constants def _unicode_format_firedrake_constant(self, o): """Format a Firedrake constant.""" i = o.count() var = "C" if len(o.ufl_shape) == 1: var += UC.combining_right_arrow_above elif len(o.ufl_shape) > 1 and self.colorama_bold: var = f"{colorama.Style.BRIGHT}{var}{colorama.Style.RESET_ALL}" return f"{var}{subscript_number(i)}" # This monkey patches ufl2unicode support for Firedrake constants Expression2UnicodeHandler.firedrake_constant = _unicode_format_firedrake_constant # This is internally done in UFL by the ufl_type decorator, but we cannot # do the same here, because we want to use the class name Constant UFLType._ufl_num_typecodes_ += 1 UFLType._ufl_all_classes_.append(Constant) UFLType._ufl_all_handler_names_.add('firedrake_constant') UFLType._ufl_obj_init_counts_.append(0) UFLType._ufl_obj_del_counts_.append(0) # And doing the above does not append to these magic UFL variables... all_ufl_classes.add(Constant) ufl_classes.add(Constant) terminal_classes.add(Constant) # These caches need rebuilding for the new type to be registered MultiFunction._handlers_cache = {} ufl.formatting.ufl2unicode._precrules = PrecedenceRules()