The information required to specify a linear PDE can be aggregated into a LinearProblem object. Aspects of using the LinearProblem object are described on this page
The LinearProblem constructor takes six arguments,
-
A Mesh, specifying the discrete geometry on which the problem is to be solved.
-
An Expr specifying the weak form of the equations
-
An Expr specifying the essential boundary conditions (if any). If the problem has no essential boundary conditions, then an empty expression should be given for this argument.
-
An Expr listing the test functions appearing in the equations. If there are multiple test functions, they should be passed through a List operator.
-
An Expr listing the unknown functions appearing in the equations. If there are multiple unknowns, they should be passed through a List operator.
-
A TSFExtended::VectorType<double> object, specifying the type of low-level linear algebra objects to be used.
Sundance can handle an arbitrarily large number of coupled weak equations and unknown field variables. Multiple weak equations can be written in two equivalent ways: as a List of equations, or as a sum of integrals. Because each equation is written in terms of its own test function, and the test functions are linearly independent, a sum of two weak equations is equivalent to enumerating the two equations in a list.
For example, consider solving the coupled system of equations
on
with boundary conditions
The Galerkin weak form of the equation is
We can write the pieces of this weak form in Sundance code as
Expr uEqn = Integral(interior, (dx*uHat)*(dx*u) + uHat*v, quad2);
Expr vEqn = Integral(interior, (dx*vHat)*(dx*v) + x*vHat, quad2);
and give it to the linear problem as either
uEqn + vEqn
or
List(uEqn, vEqn).
Likewise, the essential BCs can either be added or listed. Internally, a list of equations in transformed into a sum so the difference is purely one of taste at the user level.
A LinearProblem assembles the matrix

and vector

that appear in the discrete form of the linear problem
To solve this system, a linear solver algorithm is required. See the section on linear solvers for information on setting up linear solvers.
A linear problem is most easily solved using the solve()
method of LinearProblem, for example,
Expr soln = myProb.solve(mySolver);
where
mySolver
is an object of type TSFExtended::LinearSolver. As can be seen, the solution is returned as a
Sundance expression object, and thus can be used in other equations.
After solving, the convergence status of the solver can be obtained by calling the LinearProblem::solveStatus() method.
The order in which equations and unknowns are written can make a difference in the performance of a linear solver, and in keeping with the overall design goal of flexibility,
Sundance gives you the ability to specify this ordering. In order to understand how Sundance's ordering specification works, let's look into how
Sundance decides unknown and equation numbering.
Given a mesh and a set of unknowns, the Sundance discretization engine will traverse the mesh one maximal cell at a time and find all unknowns associated with that cell and its facets. In a problem with multiple unknowns, say velocity, pressure, and temperature, there can be more than one unknown associated with a cell; if so, the unknowns are assigned in the order that their associated UnknownFunction objects are listed in the LinearProblem constructor. This scheme gives us two ways to control the unknown ordering:
-
Cell ordering specifies the order in which cells are encountered as the mesh is traversed.
-
Function ordering specifies the order in which different functions are listed within a single cell.
There are low-level hooks for alternate cell orderings in
Sundance, however, at this point we have only implemented the identity reordering, i.e., no reordering relative to the ordering defined by the mesh.
Function ordering is controlled by the order in which test or unknown functions appear in the linear problem constructor. For example, if
ux, uy,
and
p
are unknowns we can order them as
List(ux, uy, p)
, or as
List(p, ux, uy)
or any of the other permutations. A list with the desired ordering is given to the LinearProblem constructor, e.g.,
LinearProblem problem(mesh, eqn, bc, List(vx, vy, q), List(ux, uy, p), vecType);
Notice that the test functions need not have the same ordering as their corresponding unknowns: it is possible to do a nonsymmetric ordering such as
LinearProblem problem(mesh, eqn, bc, List(vx, vy, q), List(p, ux, uy), vecType);
Nonsymmetric reorderings are often useful in PDE-constrained optimization.
Segregation of variables is not yet implemented in this version of
Sundance.
Programming with
Sundance is usually done at a high level of abstraction; however, it is sometimes useful to get low-level information about the discrete form of a problem.
If you want to inspect the system matrix and vector, or want to use them for some purpose other than solving a linear system, it is possible to do so. The
getOperator()
and
getRHS()
methods of LinearProblem return the stiffness matrix and load vector, respectively. They are returned as TSF objects. You may sometimes want to access the concrete object referenced by a TSF object; if so, refer to the documentation for the TSF adaptors to your concrete type for details on how to extract the concrete object, e.g., an EPetra matrix, underlying a TSF object.
The degree-of-freedom (DOF) maps for the row and column spaces can be accessed through the rowMap() and colMap() methods of LinearProblem. These maps specify how cell and function indices are mapped into indices into the matrix and vector. This is sometimes useful for low-level debugging, or for making custom modifications to your matrix or vector.
The bcRows() method of LinearProblem returns an STL set of all rows that are marked as boundary condition rows and processed using the replace method described in the page on boundary conditions.
If you solve the system outside the LinearProblem::solve() method but still want to capture the result into a
Sundance expression object, you can do so by using the LinearProblem::formSolutionExpr() method.