121: 1D Poisson with point charge
Solve a Poisson equation
\[- \Delta u = 0\]
in $\Omega=(-1,1)$ with a point charge $Q$ at $x=0$.
module Example121_PoissonPointCharge1D
using Printf
using VoronoiFVM
using ExtendableGrids
using GridVisualize
function main(;
nref = 0, Plotter = nothing, verbose = false, unknown_storage = :sparse,
brea = false, assembly = :edgewise
)
# Create grid in (-1,1) refined around 0
hmax = 0.2 / 2.0^nref
hmin = 0.05 / 2.0^nref
X1 = geomspace(-1.0, 0.0, hmax, hmin)
X2 = geomspace(0.0, 1.0, hmin, hmax)
X = glue(X1, X2)
grid = simplexgrid(X)
# Edit default region numbers:
# additional boundary region 3 at 0.0
bfacemask!(grid, [0.0], [0.0], 3)
# Material 1 left of 0
cellmask!(grid, [-1.0], [0.0], 1)
# Material 2 right of 0
cellmask!(grid, [0.0], [1.0], 2)
Q::Float64 = 0.0
function flux!(f, u, edge, data)
f[1] = u[1, 1] - u[1, 2]
return nothing
end
function storage!(f, u, node, data)
f[1] = u[1]
return nothing
end
# Define boundary reaction defining charge
# Note that the term is written on the left hand side, therefore the - sign
function breaction!(f, u, node, data)
if node.region == 3
f[1] = -Q
end
return nothing
end
# Create physics
physics = VoronoiFVM.Physics(;
flux = flux!,
storage = storage!,
breaction = breaction!
)
# Create system
sys = VoronoiFVM.System(grid, physics; unknown_storage = :dense, assembly = assembly)
# put potential into both regions
enable_species!(sys, 1, [1, 2])
# Set boundary conditions
boundary_dirichlet!(sys, 1, 1, 1.0)
boundary_dirichlet!(sys, 1, 2, 0.0)
# Create a solution array
U = unknowns(sys)
U .= 0
# Create solver control info
control = VoronoiFVM.NewtonControl()
control.verbose = verbose
vis = GridVisualizer(; Plotter = Plotter)
# Solve and plot for several values of charge
for q in [0.1, 0.2, 0.4, 0.8, 1.6]
if brea
# Charge in reaction term
Q = q
else
# Charge as boundary condition
sys.boundary_values[1, 3] = q
end
U = solve(sys; inival = U, control)
# Plot data
scalarplot!(
vis, grid, U[1, :]; title = @sprintf("Q=%.2f", q), clear = true,
show = true
)
end
return sum(U)
end
using Test
function runtests()
testval = 20.254591679579015
@test main(; assembly = :edgewise) ≈ testval &&
main(; assembly = :cellwise) ≈ testval
return nothing
end
end
This page was generated using Literate.jl.