On Sun, Apr 08, 2007 at 11:27:47AM +0200, Garth N. Wells wrote:
>
>
> Anders Logg wrote:
> > On Sat, Apr 07, 2007 at 02:35:07PM +0200, Garth N. Wells wrote:
> >
> >> Any further thoughts on how to apply Dirichlet boundary conditions? I've
> >> been playing around with it, and with the new UFC interface I don't see
> >> how we can retain the simplicity of the old approach.
> >>
> >> Garth
> >
> > How about the following:
> >
> > - A boundary condition can be applied in one of two ways: either by
> > overloading eval() in the class BoundaryCondition, or by suppliying a
> > Function that defines the boundary condition.
> >
> > - In the first case, the eval() function looks the same as the eval()
> > method in Function, but instead of real we have BoundaryValue:
> >
> > void eval(BoundaryValue* values, const real* x);
> >
> > When using this method, one sets the values of dofs direcly, so for a
> > BDM element, one would specify the value of the normal component, not
> > the function value.
> >
>
> Don't we need a function like pointmap to do this? tabulate_coordinates
> will give the position, but we are still missing something that
> translates the component of a vector function (0, 1 or 2) to the
> appropriate dof (like pointmap did).
Ah, yes. I forgot about this already... Since it's not always possible
to create such a list (only for Lagrange), I would like to avoid it
(and preferrably also tabulate_coordinates) and just use evaluate_dof.
If you look in the generated code for PoissonSystem.form (one of the
FFC demos), we have the following code in evaluate_dof:
virtual double evaluate_dof(unsigned int i,
const ufc::function& f,
const ufc::cell& c) const
{
...
// Nodal coordinates on reference cell
static double X[6][2] = {{0, 0}, {1, 0}, {0, 1}, {0, 0}, {1, 0}, {0, 1}};
// Components for each dof
static unsigned int components[6] = {0, 0, 0, 1, 1, 1};
...
// Evaluate function at coordinates
f.evaluate(values, coordinates, c);
// Pick component for evaluation
return values[components[i]];
}
FFC knows that the element is a tensor-product Lagrange element so it
generates the components array, evaluates the function and picks the
dof from the appropriate component. It would be good if we didn't need
to worry about the components in DOLFIN.
So then my suggestion would be to skip the first option and require
that one always sets a boundary condition by a Function. This can be
very flexible. The Function can either be something user-defined (with
an overloaded eval) or a DiscreteFunction obtained as the solution
from a previous problem.
The problem is then that we need to be able to set boundary conditions
only on a certain sub domain of the mesh, but this should be right way
to do it anyway. The current coordinate-based approach (if x == 0
then...) has only been a temporary solution until we figured out how
to label sub domains (by using a MeshFunction<uint>).
Perhaps we could provide some functionality for defining sub domains
based on coordinates?
/Anders
> > - The class BoundaryValue remains the same as today. So if one does
> >
> > values[0] = 3.5;
> >
> > then the value is set. Otherwise it remains Neumann.
> >
> > - In the second case, one specifies that the boundary condition should
> > be u = g on the boundary for g a given Function. For a BDM element,
> > one would then say that the normal component of u should be the same
> > as the normal component of g on the boundary. Or do we want to say
> > that g should be the Dirichlet values of u, so that g would be a
> > scalar that returns the normal components in the case of BDM?
> >
>
> For more complicated elements, we could just impose Dirichlet boundary
> conditions weakly.
>
> Garth
Agree.
/Anders
_______________________________________________
DOLFIN-dev mailing list
DOLFIN-dev@xxxxxxxxxx
http://www.fenics.org/mailman/listinfo/dolfin-dev