C++ Bindings

Common Type Aliases

As the lowest-level component of µSpectre, µGrid defines all commonly used type aliases and data structures used throuhgout the project. The most common aliases are described below, but it is worth having a look at the file grid_common.hh for details.

All mathematical calculations should use these types.

group Scalar types

Typedefs

typedef int Dim_t

A type alias for signed integers used for static dimensions.

This type alias is used to represent signed integers for static dimensions in the µGrid codebase. It is used for consistency throughout the code. It is also capable of representing -1, which is a requirement for Eigen.

typedef Eigen::Index Index_t

A type alias for Eigen::Index used for size-related values.

This type alias is used to represent size-related values in the µGrid codebase. It is used for consistency with Eigen and to handle large arrays that have more indices than can be counted with Dim_t. For example, arrays with dimensions 65536 × 65536 would overflow Dim_t, so Index_t is used instead.

using Uint = unsigned int

type to use in math for unsigned integers

using Int = int

type to use in math for signed integers

using Real = double

type to use in math for real numbers

using Complex = std::complex<Real>

type to use in math for complex numbers

While it is possible to use other types in principle, these are the ones for which all data structures are tested and which are known to work. Also, other µSpectre developers will expect and understand these types.

Dimensions are counted using the signed integer type muGrid::Dim_t. This is necessary because Eigen uses -1 to signify a dynamic number of dimensions.

The types muGrid::Rcoord_t and muGrid::Ccoord_t are used to represent real-valued coordinates and integer-valued coordinates (i.e., pixel- or cell-coordinates).

group Coordinate types

Typedefs

typedef std::array<Index_t, Dim> Ccoord_t

A type alias for cell coordinates.

This type alias represents cell coordinates, which are up to three integer numbers with a fixed dimension. The dimension is determined by the template parameter Dim. The coordinates are stored in a std::array of type Index_t.

Template Parameters:

Dim – The dimension of the cell coordinates. It should be between 1 and 3.

typedef std::array<Real, Dim> Rcoord_t

A type alias for real space coordinates.

This type alias represents real space coordinates, which are up to three floating point numbers with a fixed dimension. The dimension is determined by the template parameter Dim. The coordinates are stored in a std::array of type Real.

Template Parameters:

Dim – The dimension of the real space coordinates. It should be between 1 and 3.

These types are also used to define nb_grid_pts or spatial lengths for computational domains.

Field Types

The most important part of µGrid to understand is how it handles field data and access to it. By field we mean the discretisation of a mathematical field on the grid points, i.e., numerical data associated with all pixels/voxels of an FFT grid or a subset thereof. The numerical data can be scalar, vectorial, matricial, tensorial or a generic array of integer, real or complex values per pixel.

Fields that are defined on every pixel/voxel of a grid are called global fields while fields defined on a subset of pixels/voxels local fields. As an example, the strain field is a global field for any calculation (it exists in the entire domain), while for instance the field of an internal (state) variable of a material in a composite is only defined for the pixels that belong to that material.

There are several ways in which we interact with fields, and the same field might be interacted with in different ways by different parts of a problem. Let’s take the (global) strain field in a three-dimensional finite strain problem with 255 × 255 × 255 voxels as an example: the solver treats it as a long vector (of length 3² · 255³), the FFT sees it as a four-dimensional array of shape 255 × 255 × 255 × 3², and from the constitutive laws’ perspective, it is just a sequence of second-rank tensors (i.e., shape 255³ × 3 × 3).

In order to reconcile these different interpretations without copying data around, µGrid splits the concept of a field into three components:

  • Storage: The Field

    This refers managing the actual memory in which field data is held. For this, the storage abstraction needs to know the scalar type of data (Int, Real, Complex, e.t.c.), the number of pixels/voxels for which the field is defined, and the number of scalar components per pixel/voxel (e.g., 9 for a second-rank asymmetric tensor in a three-dimensional problem).

    µGrid’s abstraction for field data storage is the field represented by a child class of FieldBase<FieldCollection_t>, see Fields.

  • Representation: The FieldMap

    Meaning how to interpret the data at a given pixel/voxel (i.e., is it a vector, a matrix, …). This will also determine which mathematical operations can be performed on per-pixel/voxel data. The representation allows also to iterate over a field pixel/voxel by pixel/voxel.

    µGrid’s abstraction for field representations is the field map represented by a child class of FieldMap<FieldCollection_t, Scalar_t, NbComponents[, IsConst]>, see Field Maps.

  • Per-pixel/voxel access/iteration: The FieldCollection

    Given a pixel/voxel coordinate or index, the position of the associated pixel/voxel data is a function of the type of field (global or local). Since the determination procedure is identical for every field defined on the same domain, this ability (and the associated overhead) can be centralised into a manager of field collections.

    µGrid’s abstraction for field access and management is the field collection represented by the two classes LocalFieldCollection<Dim> and GlobalFieldCollection<Dim>, see Field Collections.

Fields

Fields are where the data is stored, so they are mainly distinguished by the scalar type they store (Int, Real or Complex), and the number of components (statically fixed size, or dynamic size).

The most commonly used fields are the statically sized ones, TensorField, MatrixField, and the ScalarField (which is really just a 1×1 matrix field).

Less commonly, we use the dynamically sized TypedField, but more on this later.

Fields have a protected constructor, which means that you cannot directly build a field object, instead you have to go through the factory function make_field<Field_t>(name, collection) (or make_statefield<Field_t>(name, collection) if you’re building a statefield, see State or History Variables) to create them and you only receive a reference to the built field. The field itself is stored in a std::unique_ptr which is registered in and managed by a field collection. This mechanism is meant to ensure that fields are not copied around or free’d so that field maps always remain valid and unambiguously linked to a field.

Fields give access to their bulk memory in form of an Eigen::Map. This is useful for instance for accessing the global strain, stress, and tangent moduli fields in the solver.

Field Maps

Field maps are light-weight resource handles (meaning they can be created and destroyed cheaply) that are iterable and provide direct per-pixel/voxel access to the data stored in the mapped field.

The choice of field map defines the type of reference you obtain when dereferencing an iterator or using the direct random acccess operator [].

Typically used field maps include:

All of these are fixed size (meaning their size is set at compile time) and therefore support fast linear algebra on the iterates. There is also a dynamically sized field map type, the TypedFieldMap which is useful for debugging and python bindings. It supports all the features of the fixed-size maps, but linear algebra on the iterates will be slow because it cannot be effectively vectorised.

Field Collections

Field collections come in two flavours; LocalFieldCollection<Dim> and GlobalFieldCollection<Dim> and are templated by the spatial dimension of the problem. They adhere to the interface defined by their common base class, FieldCollection. Both types are iterable (the iterates are the coordinates of the pixels/voxels for which the fields of the collection are defiened.

Global field collections need to be given the problem nb_grid_pts (i.e. the size of the grid) at initialisation, while local collections need to be filled with pixels/voxels through repeated calls to add_pixel(pixel). At initialisation, they derive their size from the number of pixels that have been added.

Fields (State Fields) are identified by their unique name (prefix) and can be retrieved in multiple ways:

Field &muGrid::FieldCollection::get_field(const std::string &unique_name)

returns a (base-type) reference to the field identified by unique_name. Throws a muGrid::FieldCollectionError if the field does not exist.

StateField &muGrid::FieldCollection::get_state_field(const std::string &unique_prefix)

returns a (base-type) reference to the state field identified by unique_prefix. Throws a muGrid::FieldCollectionError if the state field does not exist.

Given a pixel/voxel coordinate or index, the position of the associated pixel/voxel data is a function of the type of field (global or local). Since the determination procedure is identical for every field defined on the same domain, this ability (and the associated overhead) can be centralised into a manager of field collections.

µGrid’s abstraction for field access and management is the field collection represented by the two classes LocalFieldCollection<Dim> and LocalFieldCollection<Dim>, see Field Collections.

State or History Variables

Some fields hold state or history variables, i.e., such fields have a current value and one or more old values. This is particularly common for internal variables of inelastic materials (e.g., damage variables, plastic flow, e.t.c.). The straight-forward way of handling this situation is to define a current field, and one or more fields of the same type to hold old values. This approach has the disadvantages that it leads to a multitude of variables to keep track of, and that the values need to by cycled between the fields using a copy; this approach is both inefficient and error-prone.

µGrid addresses this situation with the state field abstraction. A state

field is an encapsulated container of fields in a single variable. It allows to access the current field values globally, and gives read-only access to old field values globally. Iterative per-pixel access is handled through state field maps which, similarly to the field map, allow to iterate though all pixels/voxels on which the field is defined, and the iterates give access to the current value at the pixel/voxel or read-only access to the old values.

Mapped Fields

Some fields are only ever going to be used by one entity (e.g., internal variables of a material). For these fields, the flexibility of the field/field collection/field map paradigm can be a burden. Mapped fields are an encapsulation of a field and a corresponding map into a single object, drastically reducing boilerplate code.