Python Bindings
###############
Fields
******
The µGrid library handles field quantities, i.e. scalar, vectors or tensors,
that vary in space. It supports only a uniform discretization of space. In the
language of µGrid, we call the discrete coordinates **pixels**. Each pixel
is associated with a physical position in space. µGrid supports fields in
two and three dimensional Cartesian grids. Note that a common name for a pixel
in three dimensions is a **voxel**, but we refer to it as a pixel throughout
this documentation.
Each pixel
can be subdivided into logical elements, for example into a number of
support points for numerical quadrature. These subdivisions are called
**sub-points**. Note that while µGrid understands the physical location of
each *pixel*, the *sub-points* are logical subdivisions and not associated
with any position within the pixel. Each *sub-point* carries the field quantity,
which can be a scalar, vector or any type of tensor. The field quantity is
called the **component**.
Multidimensional arrays
***********************
Each µGrid field has a representation in a multidimensional array. This
representation is used when accessing a field with Python via `numpy `_
or when writing the field to a file. The default representation has the shape
.. code-block:: Python
(components, sub-points, pixels)
As a concrete example, a second-rank tensor (for example the deformation
gradient) living on two quadrature points in three dimensions with a spatial
discretization of 11 x 12 grid points would have the following shape:
.. code-block:: Python
(3, 3, 2, 11, 12)
Note that the components are omitted if there is only a single component (i.e.
a scalar value), but the sub-points are always included even if there is only
a single sub-point.
Throughout the code, we call the field quantities **entries**. For example,
`nb_pixels` refers to the number of pixels while `nb_sub_pts` refers to the
number of sub-points per pixel. `nb_entries` is then the product of the two.
The above field has another multidimensional array representation that folds
the sub-point into the last dimension of the components. For the above example,
this means a multidimensional array of shape:
.. code-block:: Python
(3, 6, 11, 12)
Depending on the numerical use case, it can be useful to work with either
representation.
Field collections
*****************
In µGrid, fields are grouped into field collections. A field collection knows
about the spatial discretization of the fields, but each field can differ in number of
sub-points and components.
There are two kinds of field collections:
* A **global** collection that groups fields that have values at all pixels.
* A **local** collection that groups fields that have values only at a subset of pixels.
An example would be the elastic constants of a material that only exists at
certain locations in the domain.
The following example shows how to initialize a field collection and create scalar fields:
.. literalinclude:: ../../examples/field_collection.py
:language: python
The first argument to the constructor of `FieldCollection` is the spatial dimension.
Fields are then registered with the field accessor methods of the field collection,
e.g. `real_field` in the example above. Fields are *named*, and the name needs to be
unique. Accessing a field of the same name yield the same field object.
Note the `FieldCollection` additionally has register methods, e.g. `register_real_field`.
The different to `real_field` method is that the explicit registration of the field fails if
it already exists, while the accessor method `real_field` registers it if it does not exist
but returns the existing field if it does. You can also query a field with `get_field`,
which will raise an exception if the field does not exist.
Components
**********
In the above example, we registered a scalar field, which has one component. Vector or
tensor-valued field can be defined by specifying either simply a number of components or
the shape of the components explicitly. The following example shows how to create a
tensor-valued field that contains 2 x 2 matrices:
.. literalinclude:: ../../examples/components.py
:language: python
Sub-points
**********
A pixel can be subdivided into multiple sub-points, each of which holds a value (scalar,
vector or tensor) of the field. Examples for sub-points are elements or quadrature points
in a the finite element method.
Sub-points are named, e.g. common names would be `element` for subdivision into elements or
`quad` for quadrature points. The name of the subdivision must be specified when the field
is created.
The following example initializes a three-dimension grid with
a sub-division of each pixel into five elements:
.. literalinclude:: ../../examples/sub_points.py
:language: python
The example demonstrates two ways of accessing the field. The convenience accessor `p` (that
we also used in the above examples) and the new accessor `s`. Both yield a numpy array that is
a view on the underlying data, but with different shapes. The `s` accessor has the shape
`(components, sub-points, pixels)` and exposes the sub-points explicitly. The `p` accessor folds
the sub-points into the last dimension of the components.
numpy views
***********
The multidimensional array representation of a field is accessible via the
`array` method.
.. code-block:: Python
a = displacement_field.array(muGrid.IterUnit.SubPt)
yields the multidimensional array with the explicit sub-point dimension.
The pixel-representation can be obtained by
.. code-block:: Python
a = displacement_field.array(muGrid.IterUnit.Pixel)
Because those operations are used to frequently, there are shortcuts already
introduced in the examples above:
.. code-block:: Python
displacement_field.s # sub-point representation
displacement_field.p # pixel representation
The entries of the field occur as the first indices in the multidimensional
because a numerical code is typically vectorized of the spatial domain, i.e.
we carry out the same operation on each pixel but not on each component.
This means for the above displacement field, we can simply get the components
of the field from
.. code-block:: Python
ux, uy, uz = displacement_field.s
Each of the variable `ux`, `uy`, and `uz` is a three-dimensional array with shape
`(2, 11, 12)`. *numpy*'s
`broadcasting rules `_
make it simple to vectorize over pixels,
for example normalizing the displacement field could look like:
.. code-block:: Python
displacement_field.s /= np.sqrt(ux**2 + uy**2 + uz**2)
Note that the default storage order of the field is column-major, which means the
field components are stored next to each other in memory.
I/O
***
Fields can be written to disk in the `NetCDF `_ format.
µGrid uses `Unidata NetCDF `_ when build
with just serial capabilities and `PnetCDF `_ when build
with MPI enabled.
I/O is handled by the `FileIONetCDF` class. The following example shows how to write all fields
from a field collection to disk:
.. literalinclude:: ../../examples/io.py
:language: python
The file has the following structure (output of `ncdump -h`):
.. code-block::
netcdf example {
dimensions:
frame = UNLIMITED ; // (1 currently)
tensor_dim__strain-0 = 3 ;
tensor_dim__strain-1 = 3 ;
subpt__element-5 = 5 ;
nx = 11 ;
ny = 12 ;
nz = 13 ;
variables:
double strain(frame, tensor_dim__strain-0, tensor_dim__strain-1, subpt__element-5, nx, ny, nz) ;
strain:unit = "no unit provided" ;
// global attributes:
:creation_date = "01-06-2024 (d-m-Y)" ;
:creation_time = "23:02:06 (H:M:S)" ;
:last_modified_date = "01-06-2024 (d-m-Y)" ;
:last_modified_time = "23:02:06 (H:M:S)" ;
:muGrid_version_info = "µGrid version: 0.90.1+40-g5bc73b30\n",
"" ;
:muGrid_git_hash = "5bc73b305881ef837ce6598568d41eb3d0307c41" ;
:muGrid_description = "0.90.1+40-g5bc73b30" ;
:muGrid_git_branch_is_dirty = "false" ;
}