This page was generated from unit-2.10-dualbasis/dualbasis.ipynb.

2.10 Dual basis functions

We use dual basis functions to define interpolation operators, define transfer operators between different finite element spaces, and auxiliar space preconditioners.

Canonical interpolation

The canonical finite element interpolation operator is defined by specifying the degrees of freedom. For low order methods these are typically nodal values, while for high order methods these are most often moments. For example, the interpolation of a function \(u\) onto the \(p^{th}\) order triangle is given by: find \(u_{hp} \in V_{hp}\) such that

\begin{eqnarray*} u_{hp} (V) & = & u(V) \quad \forall \text{ vertices } V \\ \int_E u_{hp} q & = & \int_E u q \quad \forall q \in P^{p-2}(E) \; \forall \text{ edges } E \\ \int_T u_{hp} q & = & \int_T u q \quad \forall q \in P^{p-3}(T) \; \forall \text{ triangles } T \end{eqnarray*}
[1]:
import netgen.gui
from ngsolve import *
from netgen.geom2d import unit_square
import matplotlib.pylab as plt
mesh = Mesh(unit_square.GenerateMesh(maxh=2))

The NGSolve 'Set' function does local projection, and simple averaging. In particular, this does not respect point values in mesh vertices.

[2]:
fes = H1(mesh, order=3, low_order_space=False)

func = x*x*x*x
gfu = GridFunction(fes)
gfu.Set(func)
Draw (gfu)
print (gfu.vec)
 -0.0223792
 0.991843
 0.977621
 -0.00815655
 3.34984
 2.19912
 3.34984
       2
 0.00660131
 3.06756e-15
 0.00660131
 -3.19744e-14
 3.34984
 -1.80088
 -2.17601
 1.82399


Most NGSolve finite element spaces provide now a "dual" operator, which delivers the moments (i.e. the dual space basis functions) instead of function values. The integrals over faces, edges and also vertices are defined by co-dimension 1 (=BND), co-dimension 2 (=BBND) or co-dimension 3 (=BBBND) integrals over the volume elements. We define a variational problem for canonical interpolaion:

[3]:
u,v = fes.TnT()
vdual = v.Operator("dual")

a = BilinearForm(fes)
a += u*vdual*dx + u*vdual*dx(element_vb=BND) + \
    u*vdual*dx(element_vb=BBND)
a.Assemble()

f = LinearForm(fes)
f += func*vdual*dx + func*vdual*dx(element_vb=BND) + \
    func*vdual*dx(element_vb=BBND)
f.Assemble()

# interpolation in vertices preserves values 0 and 1
gfu.vec.data = a.mat.Inverse() * f.vec
print (gfu.vec)
Draw (gfu)
       0
       1
       1
       0
     3.6
       2
     3.6
       2
       0
       0
 -6.93335e-32
 -1.66533e-15
     3.6
      -2
      -2
       2


The vertex degrees of freedom vanish for edge and element basis functions, and the edge degrees of freedom vanish for element basis functions, but not vise versa. Thus, the obtained matrix A is block-triangular:

[4]:
import scipy.sparse as sp
A = sp.csr_matrix(a.mat.CSR())
plt.rcParams['figure.figsize'] = (4,4)
plt.spy(A)
plt.show()
../../_images/i-tutorials_unit-2.10-dualbasis_dualbasis_8_0.png

We can use proper block Gauss-Seidel smoothing for solving with that block triangular matrix by blocking the dofs for the individual vertices, edges and elements. Since the NGSolve Gauss-Seidel smoother reorders the order of smoothing blocks for parallelization, we have to take care to first compute vertex values, then edge values, and finally element values by running three different Gauss-Seidel sweeps.

[5]:
vblocks = [fes.GetDofNrs(vertex) for vertex in mesh.vertices]
eblocks = [fes.GetDofNrs(edge) for edge in mesh.edges]
fblocks = [fes.GetDofNrs(face) for face in mesh.faces]

print (vblocks)
print (eblocks)
print (fblocks)

vinv = a.mat.CreateBlockSmoother(vblocks)
einv = a.mat.CreateBlockSmoother(eblocks)
finv = a.mat.CreateBlockSmoother(fblocks)

vinv.Smooth(gfu.vec, f.vec)
einv.Smooth(gfu.vec, f.vec)
finv.Smooth(gfu.vec, f.vec)
print (gfu.vec)
[(0,), (1,), (2,), (3,)]
[(4, 5), (6, 7), (8, 9), (10, 11), (12, 13)]
[(14,), (15,)]
       0
       1
       1
       0
     3.6
       2
     3.6
       2
       0
       0
 -6.93335e-32
 -1.66533e-15
     3.6
      -2
      -2
       2


Embedding Finite Element Spaces

This interpolation can be used to transform functions from one finite element space \(V_{src}\) to another one \(V_{dst}\). We use the dual space of the destination space:

\[\int_{node} u_{dst} v_{dual} = \int_{node} u_{src} v_{dual} \qquad \forall \, v_{dual} \; \forall \, \text{nodes}\]

The left hand side leads to a non-symmetric square matrix, the right hand side to a rectangular matrix.

As an example we implement the transformation from an vector valued \(H^1\) space into \(H(\operatorname{div})\):

[6]:
import netgen.gui
from ngsolve import *
from netgen.geom2d import unit_square

mesh = Mesh(unit_square.GenerateMesh(maxh=0.2))

fesh1 = VectorH1(mesh, order=2)
feshdiv = HDiv(mesh, order=2)

gfuh1 = GridFunction(fesh1)
gfuh1.Set ( (x*x,y*y) )

gfuhdiv = GridFunction(feshdiv, name="uhdiv")

Build the matrices, and use a direct solver:

[7]:
amixed = BilinearForm(trialspace=fesh1, testspace=feshdiv)
ahdiv = BilinearForm(feshdiv)

u,v = feshdiv.TnT()
vdual = v.Operator("dual")
uh1 = fesh1.TrialFunction()

dS = dx(element_boundary=True)
ahdiv += u*vdual * dx + u*vdual * dS
ahdiv.Assemble()

amixed += uh1*vdual*dx + uh1*vdual*dS
amixed.Assemble()

# build transformation operator:
transform = ahdiv.mat.Inverse() @ amixed.mat
gfuhdiv.vec.data = transform * gfuh1.vec

Draw (gfuh1)
Draw (gfuhdiv)

We implement a linear operator performing the fast conversion by Gauss-Seidel smoothing:

[8]:
class MyBlockInverse(BaseMatrix):
    def __init__ (self, mat, eblocks, fblocks):
        super(MyBlockInverse, self).__init__()
        self.mat = mat
        self.einv = mat.CreateBlockSmoother(eblocks)
        self.finv = mat.CreateBlockSmoother(fblocks)
        self.res = self.mat.CreateColVector()

    def CreateRowVector(self):
        return self.mat.CreateColVector()
    def CreateColVector(self):
        return self.mat.CreateRowVector()

    def Mult(self, x, y):
        # y[:] = 0
        # self.einv.Smooth(y,x)    #   y = y +  A_E^-1  (x - A y)
        # self.finv.Smooth(y,x)    #   y = y +  A_E^-1  (x - A y)

        # the same, but we see how to transpose that
        y.data = self.einv * x
        self.res.data = x - self.mat * y
        y.data += finv * self.res

    def MultTrans(self, x, y):
        y.data = self.finv.T * x
        self.res.data = x - self.mat.T * y
        y.data += einv.T * self.res


eblocks = [feshdiv.GetDofNrs(edge) for edge in mesh.edges]
fblocks = [feshdiv.GetDofNrs(face) for face in mesh.faces]

transform = MyBlockInverse(ahdiv.mat, eblocks, fblocks) @ amixed.mat
gfuhdiv.vec.data = transform * gfuh1.vec

Auxiliary Space Preconditioning

Nepomnyaschikh 91, Hiptmair-Xu 07, ….

Assume we have a complicated problem with some complicated discretization, and we have good preconditioners for a nodal finite element discretization for the Laplace operator. By auxiliary space preconditioning we can reuse the simple preconditioners for the complicated problems. It is simple, and works well in many cases.

As a simple example, we precondition a DG discretization by an \(H^1\) conforming method.

[9]:
mesh = Mesh(unit_square.GenerateMesh(maxh=0.1))

The DG discretization:

[10]:
order=4
fesDG = L2(mesh, order=order, dgjumps=True)
u,v = fesDG.TnT()
aDG = BilinearForm(fesDG)
jump_u = u-u.Other()
jump_v = v-v.Other()
n = specialcf.normal(2)
mean_dudn = 0.5*n * (grad(u)+grad(u.Other()))
mean_dvdn = 0.5*n * (grad(v)+grad(v.Other()))
alpha = 4
h = specialcf.mesh_size
aDG = BilinearForm(fesDG)
aDG += grad(u)*grad(v) * dx
aDG += alpha*order**2/h*jump_u*jump_v * dx(skeleton=True)
aDG += alpha*order**2/h*u*v * ds(skeleton=True)
aDG += (-mean_dudn*jump_v -mean_dvdn*jump_u) * dx(skeleton=True)
aDG += (-n*grad(u)*v-n*grad(v)*u)* ds(skeleton=True)
aDG.Assemble()

fDG = LinearForm(fesDG)
fDG += 1*v * dx
fDG.Assemble()
gfuDG = GridFunction(fesDG)

The auxiliary \(H^1\) discretization:

[11]:
fesH1 = H1(mesh, order=2, dirichlet=".*")
u,v = fesH1.TnT()
aH1 = BilinearForm(fesH1)
aH1 += grad(u)*grad(v)*dx
preH1 = Preconditioner(aH1, "bddc")
aH1.Assemble()
[12]:
transform = fesH1.ConvertL2Operator(fesDG)
pre = transform @ preH1.mat @ transform.T + aDG.mat.CreateSmoother()

solvers.CG(mat=aDG.mat, rhs=fDG.vec, sol=gfuDG.vec, pre=pre, \
           maxsteps=200)

Draw (gfuDG)
iteration 0 error = 0.2531562087169284
iteration 1 error = 0.03354069456506922
iteration 2 error = 0.008966767436643222
iteration 3 error = 0.004244873344653201
iteration 4 error = 0.0015376161568948906
iteration 5 error = 0.0010332060879173154
iteration 6 error = 0.0009244390324201271
iteration 7 error = 0.0011118047897439255
iteration 8 error = 0.0009222415374385427
iteration 9 error = 0.000726425525724842
iteration 10 error = 0.0005929804827760287
iteration 11 error = 0.00046809037979247595
iteration 12 error = 0.0003581060513104977
iteration 13 error = 0.0003379748411397421
iteration 14 error = 0.0003081861529672423
iteration 15 error = 0.00025710855222474535
iteration 16 error = 0.0002075844506679731
iteration 17 error = 0.00016301354842722032
iteration 18 error = 0.00011555145351080929
iteration 19 error = 8.887847926505388e-05
iteration 20 error = 7.557505071814767e-05
iteration 21 error = 7.004297233469325e-05
iteration 22 error = 5.5349264464184216e-05
iteration 23 error = 4.7853455576337604e-05
iteration 24 error = 3.999444092604901e-05
iteration 25 error = 3.2420221040142364e-05
iteration 26 error = 2.823496936071904e-05
iteration 27 error = 2.791857320142218e-05
iteration 28 error = 2.6159948412659085e-05
iteration 29 error = 2.2515444168376023e-05
iteration 30 error = 1.8959649652892614e-05
iteration 31 error = 1.6599252337777326e-05
iteration 32 error = 1.2858237814290458e-05
iteration 33 error = 1.0780280691368687e-05
iteration 34 error = 9.813263580153103e-06
iteration 35 error = 9.51199182611226e-06
iteration 36 error = 7.736168049596183e-06
iteration 37 error = 5.838573338932672e-06
iteration 38 error = 4.700205388534292e-06
iteration 39 error = 4.114894872877887e-06
iteration 40 error = 3.597118729436315e-06
iteration 41 error = 3.4378511827964535e-06
iteration 42 error = 3.1997135299481288e-06
iteration 43 error = 2.6052414477561223e-06
iteration 44 error = 1.9776554102659035e-06
iteration 45 error = 1.6368258923906027e-06
iteration 46 error = 1.504800632381481e-06
iteration 47 error = 1.5089149162355437e-06
iteration 48 error = 1.3005968755530861e-06
iteration 49 error = 1.0628548744884738e-06
iteration 50 error = 8.394651022909381e-07
iteration 51 error = 6.86953464180822e-07
iteration 52 error = 6.030181423318715e-07
iteration 53 error = 5.806640948949447e-07
iteration 54 error = 5.118882202775998e-07
iteration 55 error = 4.1975528849149484e-07
iteration 56 error = 3.312381519546124e-07
iteration 57 error = 2.76477518134174e-07
iteration 58 error = 2.3231157310469447e-07
iteration 59 error = 1.9364056475072084e-07
iteration 60 error = 1.774121005099061e-07
iteration 61 error = 1.4840266164541486e-07
iteration 62 error = 1.1565176087755733e-07
iteration 63 error = 9.189907462022656e-08
iteration 64 error = 7.161818851456794e-08
iteration 65 error = 6.351308763916282e-08
iteration 66 error = 6.096363548909816e-08
iteration 67 error = 5.222446498881792e-08
iteration 68 error = 4.107240046099957e-08
iteration 69 error = 3.2186007524339513e-08
iteration 70 error = 2.6409270430150457e-08
iteration 71 error = 2.1100144836576114e-08
iteration 72 error = 1.8071058894981936e-08
iteration 73 error = 1.6745611705290442e-08
iteration 74 error = 1.4527909695458872e-08
iteration 75 error = 1.1849568152383082e-08
iteration 76 error = 9.140246622125565e-09
iteration 77 error = 7.167827908693989e-09
iteration 78 error = 6.091382762809514e-09
iteration 79 error = 5.681159768950569e-09
iteration 80 error = 5.38807369768192e-09
iteration 81 error = 4.282851617136741e-09
iteration 82 error = 3.2723083204478825e-09
iteration 83 error = 2.5840034762406374e-09
iteration 84 error = 2.2288318134637605e-09
iteration 85 error = 2.0695617326455375e-09
iteration 86 error = 1.8760227083971243e-09
iteration 87 error = 1.556099490487781e-09
iteration 88 error = 1.1303080868186056e-09
iteration 89 error = 8.768035718636273e-10
iteration 90 error = 7.491352145308873e-10
iteration 91 error = 6.784946578407398e-10
iteration 92 error = 6.197424249374832e-10
iteration 93 error = 5.561135066964688e-10
iteration 94 error = 4.090770920839586e-10
iteration 95 error = 3.1573814819942226e-10
iteration 96 error = 2.586843088813737e-10
iteration 97 error = 2.48366427176674e-10
iteration 98 error = 2.338686832615801e-10
iteration 99 error = 1.9589892681960214e-10
iteration 100 error = 1.520759939264206e-10
iteration 101 error = 1.165730432962988e-10
iteration 102 error = 9.84180636855076e-11
iteration 103 error = 9.389707032159417e-11
iteration 104 error = 8.826884873399961e-11
iteration 105 error = 7.222731191990624e-11
iteration 106 error = 5.210463669944703e-11
iteration 107 error = 4.1119299093425053e-11
iteration 108 error = 3.5248704516769406e-11
iteration 109 error = 3.208291563202172e-11
iteration 110 error = 3.1137361456054615e-11
iteration 111 error = 2.6816413593969138e-11
iteration 112 error = 2.1037399390596527e-11
iteration 113 error = 1.5453275591536068e-11
iteration 114 error = 1.2412185675833363e-11
iteration 115 error = 1.0924184199350918e-11
iteration 116 error = 1.0777860480202433e-11
iteration 117 error = 9.770425921721386e-12
iteration 118 error = 7.572287686087435e-12
iteration 119 error = 6.004296522525905e-12
iteration 120 error = 4.6841171608504985e-12
iteration 121 error = 4.116683132325578e-12
iteration 122 error = 4.1009112986231555e-12
iteration 123 error = 3.8386053590974124e-12
iteration 124 error = 2.978307061217356e-12
iteration 125 error = 2.261529524623657e-12
iteration 126 error = 1.7824660385265565e-12
iteration 127 error = 1.5743631191875813e-12
iteration 128 error = 1.5063699318235388e-12
iteration 129 error = 1.3475285297671042e-12
iteration 130 error = 1.1488951089675946e-12
iteration 131 error = 9.033421755976531e-13
iteration 132 error = 7.178339795003351e-13
iteration 133 error = 6.096449748966742e-13
iteration 134 error = 5.726816013313327e-13
iteration 135 error = 5.432828603876235e-13
iteration 136 error = 4.903102495067912e-13
iteration 137 error = 4.073484205469351e-13
iteration 138 error = 3.3308329891153863e-13
iteration 139 error = 2.696616336593137e-13
iteration 140 error = 2.4035049428085196e-13
[ ]: