This page was generated from unit-6.1.2-surfacepde/surface_pdes.ipynb.
6.1.2 Surface PDEs¶
Surface Poisson equation¶
Homogeneous Dirichlet data¶
[1]:
from ngsolve import *
from netgen.csg import *
from netgen.meshing import MeshingStep
from ngsolve.webgui import Draw
Given a half-sphere we want to solve the surface Poisson equation
[2]:
geo = CSGeometry()
sphere = Sphere(Pnt(0,0,0), 1)
bot = Plane(Pnt(0,0,0), Vec(0,0,-1))
finitesphere = sphere * bot
geo.AddSurface(sphere, finitesphere.bc("surface"))
geo.NameEdge(sphere,bot, "bottom")
mesh = Mesh(geo.GenerateMesh(maxh=0.3))
mesh.Curve(2)
Draw(mesh)
Start Findpoints
main-solids: 1
Found points 4
Analyze spec points
Find edges
1 edges found
Check intersecting edges
CalcLocalH: 21 Points 0 Elements 0 Surface Elements
Start Findpoints
Analyze spec points
Find edges
1 edges found
Check intersecting edges
Start Findpoints
Analyze spec points
Find edges
1 edges found
Check intersecting edges
Surface 1 / 1
load internal triangle rules
Surface meshing done
0 illegal triangles
Optimize Surface
Edgeswapping, topological
Smoothing
Combine improve
Smoothing
Edgeswapping, metric
Smoothing
Combine improve
Smoothing
Edgeswapping, metric
Smoothing
Combine improve
Smoothing
175 elements, 99 points
SplitSeparateFaces
Remove Illegal Elements
Volume Optimization
CombineImprove
ImproveMesh
SplitImprove
ImproveMesh
SwapImprove
SwapImprove2
ImproveMesh
CombineImprove
ImproveMesh
SplitImprove
ImproveMesh
SwapImprove
SwapImprove2
ImproveMesh
CombineImprove
ImproveMesh
SplitImprove
ImproveMesh
SwapImprove
SwapImprove2
ImproveMesh
Update mesh topology
Update clusters
Curve elements, order = 2
Update clusters
Curving edges
Curving faces
[2]:
BaseWebGuiScene
Therefore, we define the usual \(H^1\) finite element space and use the dirichlet_bbnd flag to indicate the BBoundary on which the Dirichlet condtions are prescribed. The test- and trial-functions are gien as usual.
[3]:
fes = H1(mesh, order=2, dirichlet_bbnd="bottom")
u, v = fes.TnT()
print(fes.FreeDofs())
0: 00000000000000000000011111111111111111111111111111
50: 11111111111111111111111111111111111111111111111100
100: 11011011011011011011011011011011011011011011011011
150: 01101101111111111111111111111111111111111111111111
200: 11111111111111111111111111111111111111111111111111
250: 11111111111111111111111111111111111111111111111111
300: 11111111111111111111111111111111111111111111111111
350: 111111111111111111
For the (bi-)linear form we have to take care that we have to define the according integrators on the boundary and that the Trace operator has to be used to obtain the tangential/surface derivative
[4]:
a = BilinearForm(fes, symmetric=True)
a += grad(u).Trace()*grad(v).Trace()*ds
a.Assemble()
force = sin(x)*y*exp(z)
f = LinearForm(fes)
f += force*v*ds
f.Assemble()
assemble BND element 173/173
[4]:
<ngsolve.comp.LinearForm at 0x7fbf64061070>
assemble BND element 173/173
Solving is done as usual
[5]:
gfu = GridFunction(fes)
gfu.vec.data = a.mat.Inverse(fes.FreeDofs())*f.vec
Draw(gfu, mesh, "u")
call pardiso ... done
[5]:
BaseWebGuiScene
Inhomogeneous Dirichlet data¶
To solve the same problem with non-homogenous Dirichlet data, \(u=u_D\) on \(\Gamma_D\) the same technique as in the volume case is used, where we have to set a function on the BBoundary instead on the boundary
[6]:
gfu.Set(x, definedon=mesh.BBoundaries("bottom"))
r = f.vec.CreateVector()
r.data = f.vec - a.mat*gfu.vec
gfu.vec.data += a.mat.Inverse(fes.FreeDofs())*r
Draw(gfu)
setvalues element 21/21
call pardiso ... done
[6]:
BaseWebGuiScene
Finite element spaces for surfaces¶
We differ between two types of finite element spaces for surfaces. The first class consists of spaces, where the restriction of a 3D element to the surface leads to a valid 2D element of the same type: H1, HCurl, HCurlCurl, NumberSpace.
These spaces can directly be used, one has to take care using the Trace operator. Otherwise an exception is thrown during assembling.
[7]:
a += grad(u)*grad(v)*ds
a.Assemble()
---------------------------------------------------------------------------
NgException Traceback (most recent call last)
/tmp/ipykernel_4103/2227806119.py in <module>
----> 1 a += grad(u)*grad(v)*ds
2 a.Assemble()
NgException: Trialfunction does not support BND-forms, maybe a Trace() operator is missing, type = grad
The NumberSpace is an exception as it represents only a number, where no Trace operator has to be used.
The second class is given by
Space |
Surface Space |
---|---|
L2 |
SurfaceL2 |
HDiv |
HDivSurface |
HDivDiv |
HDivDivSurface |
FacetFESpace |
FacetSurface |
Here, a 2D reference element is mapped directly onto the surface. To be consistent, also here the Trace operator has to be used.
[ ]: