Diffusion Equation in a Wedge with Mixed Boundary Conditions
In this example, we consider a diffusion equation on a wedge with angle $\alpha$ and mixed boundary conditions:
\[\begin{equation*} \begin{aligned} \pdv{u(r, \theta, t)}{t} &= \grad^2u(r,\theta,t), & 0<r<1,\,0<\theta<\alpha,\,t>0,\\[6pt] \pdv{u(r, 0, t)}{\theta} & = 0 & 0<r<1,\,t>0,\\[6pt] u(1, \theta, t) &= 0 & 0<\theta<\alpha,\,t>0,\\[6pt] \pdv{u(r,\alpha,t)}{\theta} & = 0 & 0<\theta<\alpha,\,t>0,\\[6pt] u(r, \theta, 0) &= f(r,\theta) & 0<r<1,\,0<\theta<\alpha, \end{aligned} \end{equation*}\]
where we take $f(r,\theta) = 1-r$ and $\alpha=\pi/4$.
Note that the PDE is provided in polar form, but Cartesian coordinates are assumed for the operators in our code. The conversion is easy, noting that the two Neumann conditions are just equations of the form $\grad u \vdot \vu n = 0$. Moreover, although the right-hand side of the PDE is given as a Laplacian, recall that $\grad^2 = \div\grad$, so we can write the PDE as $\partial u/\partial t + \div \vb q = 0$, where $\vb q = -\grad u$.
Let us now setup the problem. To define the geometry, we need to be careful that the Triangulation
recognises that we need to split the boundary into three parts, one part for each boundary condition. This is accomplished by providing a single vector for each part of the boundary as follows (and as described in DelaunayTriangulation.jl's documentation), where we also refine!
the mesh to get a better mesh. For the arc, we use the CircularArc
so that the mesh knows that it is triangulating a certain arc in that area.
using DelaunayTriangulation, FiniteVolumeMethod, ElasticArrays
α = π / 4
points = [(0.0, 0.0), (1.0, 0.0), (cos(α), sin(α))]
bottom_edge = [1, 2]
arc = CircularArc((1.0, 0.0), (cos(α), sin(α)), (0.0, 0.0))
upper_edge = [3, 1]
boundary_nodes = [bottom_edge, [arc], upper_edge]
tri = triangulate(points; boundary_nodes)
A = get_area(tri)
refine!(tri; max_area=1e-4A)
mesh = FVMGeometry(tri)
FVMGeometry with 8202 control volumes, 16040 triangles, and 24241 edges
This is the mesh we've constructed.
using CairoMakie
fig, ax, sc = triplot(tri)
fig
To confirm that the boundary is now in three parts, see:
get_boundary_nodes(tri)
3-element Vector{Vector{Int64}}:
[1, 6648, 3587, 6429, 3557, 5003, 3582, 535, 5006, 3579 … 4785, 203, 4760, 2736, 4784, 229, 6362, 4720, 5961, 2]
[2, 7091, 5468, 7419, 323, 4710, 8353, 221, 4706, 5298 … 2632, 6609, 322, 7380, 2231, 7248, 332, 7086, 2227, 3]
[3, 6616, 2215, 5359, 325, 6355, 2080, 6168, 308, 3288 … 5491, 534, 4990, 3589, 5000, 3556, 6428, 3586, 6647, 1]
We now need to define the boundary conditions. For this, we need to provide Tuple
s, where the i
th element of the Tuple
s refers to the i
th part of the boundary. The boundary conditions are thus:
lower_bc = arc_bc = upper_bc = (x, y, t, u, p) -> zero(u)
types = (Neumann, Dirichlet, Neumann)
BCs = BoundaryConditions(mesh, (lower_bc, arc_bc, upper_bc), types)
BoundaryConditions with 3 boundary conditions with types (Neumann, Dirichlet, Neumann)
Now we can define the PDE. We use the reaction-diffusion formulation, specifying the diffusion function as a constant.
f = (x, y) -> 1 - sqrt(x^2 + y^2)
D = (x, y, t, u, p) -> one(u)
initial_condition = [f(x, y) for (x, y) in DelaunayTriangulation.each_point(tri)]
final_time = 0.1
prob = FVMProblem(mesh, BCs; diffusion_function=D, initial_condition, final_time)
FVMProblem with 8202 nodes and time span (0.0, 0.1)
If you did want to use the flux formulation, you would need to provide
flux = (x, y, t, α, β, γ, p) -> (-α, -β)
#9 (generic function with 1 method)
which replaces u
with αx + βy + γ
so that we approximate $\grad u$ by $(\alpha,\beta)^{\mkern-1.5mu\mathsf{T}}$, and the negative is needed since $\vb q = -\grad u$.
We now solve the problem. We provide the solver for this problem. In my experience, I've found that TRBDF2(linsolve=KLUFactorization())
typically has the best performance for these problems.
using OrdinaryDiffEq, LinearSolve
sol = solve(prob, TRBDF2(linsolve=KLUFactorization()), saveat=0.01, parallel=Val(false))
retcode: Success
Interpolation: 1st order linear
t: 11-element Vector{Float64}:
0.0
0.01
0.02
0.03
0.04
0.05
0.06
0.07
0.08
0.09
0.1
u: 11-element Vector{Vector{Float64}}:
[1.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.5, 0.5, 0.2742608203909225, 0.25 … 0.07054525783529475, 0.32706835895764463, 0.37379427597203063, 0.20586900044003154, 0.2667451754925648, 0.14446073092610567, 0.2334962426876055, 0.7982378261948451, 0.12186065790155398, 0.22764124346876302]
[0.8217393862454233, 0.0, 0.0, 0.0, 0.0, 0.0, 0.47952625852179054, 0.4795248285410618, 0.2742608203909224, 0.23680391940144735 … 0.06407271774859011, 0.31209285203961384, 0.3576292329451442, 0.19373832031074756, 0.25316717852736836, 0.13422181967381108, 0.22068513653916977, 0.7424709922796584, 0.112540500995262, 0.21496831893118282]
[0.7468094286327989, 0.0, 0.0, 0.0, 0.0, 0.0, 0.45792740217833333, 0.457926106641875, 0.2742608203909225, 0.2246913815862142 … 0.06018946085857867, 0.297342166029556, 0.34129458204788843, 0.18331517308071174, 0.2404582055066338, 0.12652010982877132, 0.20918179974919598, 0.6890476144545409, 0.10595374996359407, 0.2036871803326995]
[0.6920153641095214, 0.0, 0.0, 0.0, 0.0, 0.0, 0.43545085454618204, 0.43544978918382243, 0.2742608203909225, 0.2136321181539309 … 0.05707463643563301, 0.2830446590349549, 0.32501707779362604, 0.17416183021782833, 0.22868784625768068, 0.1200815963567329, 0.1988297191367197, 0.642652611481857, 0.10052606333988594, 0.1935876496763704]
[0.6446116065596812, 0.0, 0.0, 0.0, 0.0, 0.0, 0.4129252357235237, 0.4129243011688625, 0.2742608203909225, 0.20314232286237124 … 0.05423520792792547, 0.26914350728162173, 0.30896379929215373, 0.16558516977452511, 0.21746625084493293, 0.11413781243561535, 0.1890576834944881, 0.6013265479001114, 0.09554156004561046, 0.1840696301413127]
[0.6029894027808588, 0.0, 0.0, 0.0, 0.0, 0.0, 0.3907758374351735, 0.39077500298120843, 0.27426082039092253, 0.19306882583008592 … 0.05157236538779525, 0.25560366345346713, 0.29321467352502695, 0.1574118193855062, 0.20665670116477466, 0.10852429125269578, 0.17970125371347387, 0.5641594469178337, 0.09084664736977457, 0.17496579372393548]
[0.5651036828780851, 0.0, 0.0, 0.0, 0.0, 0.0, 0.36949337260343534, 0.36949262410784406, 0.2742608203909225, 0.183310038370493 … 0.04900845440010177, 0.24247106170525345, 0.27795052392215613, 0.14950659608994127, 0.19617987973511927, 0.10310765783757855, 0.17064216714962327, 0.5298382149911381, 0.0863195513222838, 0.16615315737697472]
[0.5302581017339084, 0.0, 0.0, 0.0, 0.0, 0.0, 0.3491744758711103, 0.34917379955474576, 0.2742608203909225, 0.17386071523154137 … 0.04652787233071769, 0.22977845980879683, 0.2632315051262864, 0.1418509463491667, 0.1860374169975082, 0.09786379795296811, 0.1618693374121837, 0.4979610735900612, 0.08193759006467705, 0.15761879770979612]
[0.4982222964433546, 0.0, 0.0, 0.0, 0.0, 0.0, 0.3298202826588713, 0.3298196661948114, 0.2742608203909225, 0.16473292841303344 … 0.04413069858366479, 0.21754485876084378, 0.24907763017496543, 0.13445221957830206, 0.17624319348720155, 0.09279525420573283, 0.15339302305683694, 0.4683872079544743, 0.07770223997834844, 0.14937233593307958]
[0.46876590410144975, 0.0, 0.0, 0.0, 0.0, 0.0, 0.3114319283873914, 0.31143136090093865, 0.2742608203909225, 0.15593874991477172 … 0.04181701256296492, 0.2057892595581409, 0.23550891210574085, 0.12731776519246676, 0.16681108973945946, 0.08790456920274109, 0.14522348263926455, 0.4409758033242486, 0.07361497744469213, 0.1414233932575049]
[0.44165856180321983, 0.0, 0.0, 0.0, 0.0, 0.0, 0.2940105484773437, 0.29401002054602554, 0.2742608203909225, 0.14749025173655866 … 0.03958689367263985, 0.19453066319743476, 0.22254536395616048, 0.12045493260678033, 0.15775498628954196, 0.08319428555086124, 0.13737097471514823, 0.4155860449392552, 0.06967727884510226, 0.133781590893752]
using CairoMakie
fig = Figure(fontsize=38)
for (i, j) in zip(1:3, (1, 6, 11))
local ax
ax = Axis(fig[1, i], width=600, height=600,
xlabel="x", ylabel="y",
title="t = $(sol.t[j])",
titlealign=:left)
tricontourf!(ax, tri, sol.u[j], levels=0:0.01:1, colormap=:matter)
tightlimits!(ax)
end
resize_to_layout!(fig)
fig
Just the code
An uncommented version of this example is given below. You can view the source code for this file here.
using DelaunayTriangulation, FiniteVolumeMethod, ElasticArrays
α = π / 4
points = [(0.0, 0.0), (1.0, 0.0), (cos(α), sin(α))]
bottom_edge = [1, 2]
arc = CircularArc((1.0, 0.0), (cos(α), sin(α)), (0.0, 0.0))
upper_edge = [3, 1]
boundary_nodes = [bottom_edge, [arc], upper_edge]
tri = triangulate(points; boundary_nodes)
A = get_area(tri)
refine!(tri; max_area=1e-4A)
mesh = FVMGeometry(tri)
using CairoMakie
fig, ax, sc = triplot(tri)
fig
get_boundary_nodes(tri)
lower_bc = arc_bc = upper_bc = (x, y, t, u, p) -> zero(u)
types = (Neumann, Dirichlet, Neumann)
BCs = BoundaryConditions(mesh, (lower_bc, arc_bc, upper_bc), types)
f = (x, y) -> 1 - sqrt(x^2 + y^2)
D = (x, y, t, u, p) -> one(u)
initial_condition = [f(x, y) for (x, y) in DelaunayTriangulation.each_point(tri)]
final_time = 0.1
prob = FVMProblem(mesh, BCs; diffusion_function=D, initial_condition, final_time)
flux = (x, y, t, α, β, γ, p) -> (-α, -β)
using OrdinaryDiffEq, LinearSolve
sol = solve(prob, TRBDF2(linsolve=KLUFactorization()), saveat=0.01, parallel=Val(false))
using CairoMakie
fig = Figure(fontsize=38)
for (i, j) in zip(1:3, (1, 6, 11))
local ax
ax = Axis(fig[1, i], width=600, height=600,
xlabel="x", ylabel="y",
title="t = $(sol.t[j])",
titlealign=:left)
tricontourf!(ax, tri, sol.u[j], levels=0:0.01:1, colormap=:matter)
tightlimits!(ax)
end
resize_to_layout!(fig)
fig
This page was generated using Literate.jl.