begin
    import Pkg as _Pkg
    haskey(ENV, "PLUTO_PROJECT") && _Pkg.activate(ENV["PLUTO_PROJECT"])
    using Revise
    using VoronoiFVM
    using ExtendableGrids
    using GridVisualize
    using PlutoUI
    using HypertextLiteral
    using LinearAlgebra
    using LinearSolve
    using Test
    using CairoMakie
    CairoMakie.activate!(; type = "png", visible = false)
    GridVisualize.default_plotter!(CairoMakie)
end
CairoMakie

Interface conditions in 1D

Source

This notebooks discusses handling of internal interfaces with VoronoiFVM.jl.

Two subdomains

For a simple stationary diffusion equation with an interior interface, we discuss possible interface conditions between two subdomains.

Let \(\Omega=\Omega_1\cup\Omega_2\) where \(\Omega_1=(-1,0)\) and \(\Omega_2=(0,1)\). Let \(\Gamma_1={-1}\),\(\Gamma_2={1}\) and \(\Gamma_3={0}\).

Regard the following problem:

\(\begin{aligned} -\Delta u_1 &= 0 & \text{in}\quad \Omega_1\\ -\Delta u_2 &= 0 & \text{in}\quad \Omega_2\\ \end{aligned}\)

with exterior boundary conditions

\(u_1|_{\Gamma_1} = g_1\) and \(u_2|_{\Gamma_2} = g_2\)

For the interior boundary (interface) conditions we set

\(\nabla u_1|_{\Gamma_3}+f_1(u_1,u_2)=0\)

\(-\nabla u_2|_{\Gamma_3}+f_2(u_1,u_2)=0\)

where \(f_1\), \(f_2\) are discussed later.

Set up

Create a grid with two subdomins and an interface in the center.

nref = 2
2
begin
    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 = VoronoiFVM.Grid(X)

    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)
end;
gridplot(grid; legend = :rt, resolution = (600, 200))

For later use (plotting) extract the two subgrids from the grid

subgrid1 = subgrid(grid, [1]);
subgrid2 = subgrid(grid, [2]);

Define the diffusion flux for the two species in their respective subdomains

function flux!(f, u, edge, data)
    if edge.region == 1
        f[1] = u[1, 1] - u[1, 2]
    end
    if edge.region == 2
        f[2] = u[2, 1] - u[2, 2]
    end
    return nothing
end
flux! (generic function with 1 method)

Specify the outer boundary values.

const g_1 = 1.0
1.0
const g_2 = 0.1
0.1

Create the system. We pass the interface condition function as a parameter.

function make_system(breaction)
    physics = VoronoiFVM.Physics(; flux = flux!, breaction = breaction)

    ## Create system
    sys = VoronoiFVM.System(grid, physics; unknown_storage = :sparse)

    ##  Enable species in their respective subregions
    enable_species!(sys, 1, [1])
    enable_species!(sys, 2, [2])

    ## Set boundary conditions
    for ispec in 1:2
        boundary_dirichlet!(sys, ispec, 1, g_1)
        boundary_dirichlet!(sys, ispec, 2, g_2)
    end
    return sys
end
make_system (generic function with 1 method)

Stationary solution with zero initial value

function mysolve(sys)
    U = solve(sys)
    U1 = view(U[1, :], subgrid1)
    U2 = view(U[2, :], subgrid2)
    return U1, U2
end
mysolve (generic function with 1 method)

Plot the results

function plot(U1, U2; title = "")
    vis = GridVisualizer(; resolution = (600, 300))
    scalarplot!(
        vis,
        subgrid1,
        U1;
        clear = false,
        show = false,
        color = :green,
        label = "u1"
    )
    return scalarplot!(
        vis,
        subgrid2,
        U2;
        clear = false,
        show = true,
        color = :blue,
        label = "u2",
        legend = :rt,
        title = title,
        flimits = (-0.5, 1.5)
    )
end
plot (generic function with 1 method)

No interface reaction

This means we set \(f_1(u_1,u_2)=0\) and \(f_2(u_1,u_2)=0\).

function noreaction(f, u, node, data)
    return nothing
end
noreaction (generic function with 1 method)
system1 = make_system(noreaction);
plot(mysolve(system1)...)

The solution consists of two constants defined by the respective Dirichlet boundary conditions at the outer boundary.

Mass action law reaction \(u_1 \leftrightharpoons u_2\)

This is a rather general ansatz where we assume a backward-forward reaction between the two species meeting at the interface with reaction constants \(k_1\) and \(k_2\), respectively.

According to the mass action law, this translates to a reaction rate

\(r(u_1,u_2)=k_1u_1 - k_2u_2\)

and correspondingly

\(f_1(u_1,u_2)=r\)

\(f_2(u_1,u_2)=-r\)

Note, that \(f_i\) is monotonically increasing in \(u_i\) and monotonically decreasing in the respective other argument, leading to an M-Property of the overall discretization matrix.

Note that the "no reaction" case is just a special case where \(k_1,k_2=0\).

function mal_reaction(f, u, node, data)
    if node.region == 3
        react = k1 * u[1] - k2 * u[2]
        f[1] = react
        f[2] = -react
    end
    return nothing
end
mal_reaction (generic function with 1 method)
system2 = make_system(mal_reaction)
VoronoiFVM.System{Float64, Float64, Int32, Int64, SparseArrays.SparseMatrixCSC{Int32,
  Int32}}(  
  grid = ExtendableGrids.ExtendableGrid{Float64, Int32}(dim=1, nnodes=75, ncells=74,
  nbfaces=3),  
  physics = Physics(flux=flux!, storage=default_storage, breaction=mal_reaction, ),  
  num_species = 2)
begin
    const k1 = 0.1
    const k2 = 10
end
10

The back reaction is 100 times stronger than the forward reaction. This means that species 2 is consumed, creating species 1.

Penalty enforcing continuity

Setting \(k_1,k_2\) to a large number leads to another special case of the above reaction - similar to the penalty method to implement the Dirichlet boundary conditions, this lets the reaction equation dominate, which in this case forces \(u_1-u_2=0\) at the interface, and thus continuity.

function penalty_reaction(f, u, node, data)
    if node.region == 3
        react = 1.0e10 * (u[1] - u[2])
        f[1] = react
        f[2] = -react
    end
    return nothing
end
penalty_reaction (generic function with 1 method)
system3 = make_system(penalty_reaction);
plot(mysolve(system3)...)

Penalty enforcing fixed jump

Instead of enforcing continuity, one can enforce a fixed jump.

const jump = 0.2
0.2
function penalty_jump_reaction(f, u, node, data)
    if node.region == 3
        react = 1.0e10 * (u[1] - u[2] - jump)
        f[1] = react
        f[2] = -react
    end
    return nothing
end
penalty_jump_reaction (generic function with 1 method)
system3jump = make_system(penalty_jump_reaction);
plot(mysolve(system3jump)...)

Interface recombination

Here, we implement an annihilation reaction \(u_1 + u_2 \to \emptyset\) According to the mass action law, this is implemented via

\(r(u_1,u_2)=k_r u_1 u_2\)

\(f_1(u_1,u_2)=r\)

\(f_2(u_1,u_2)=r\)

function recombination(f, u, node, data)
    if node.region == 3
        react = k_r * (u[1] * u[2])
        f[1] = react
        f[2] = react
    end
    return nothing
end;
system4 = make_system(recombination);
const k_r = 1000
1000
plot(mysolve(system4)...)

Bot species are consumed at the interface.

Thin conductive interface layer

Let us assume that the interface is of thickness \(d\) which is however small with respect to \(\Omega\) that we want to derive an interface condition from the assumption of an exact continuous solution within the interface.

So let \(\Omega_I=(x_l,x_r)\) be the interface region where we have \(-\Delta u_I=0\) with values \(u_l\), \(u_r\) at the boundaries.

Then we have for the flux in the interface region, \(q_I=\nabla u = \frac1{d}(u_r - u_l)\)

Continuity of fluxes then gives \(f_1=q_I\) and \(f_2=-q_I\).

Continuity of \(u\) gives \(u_{1,I}=u_l, u_{2,I}=u_r\) This gives

\(r=q_I=\frac{1}{d}(u_1-u_{2})\)

\(f_1(u_1,v_1)=r\)

\(f_2(u_1,v_1)=-r\)

and therefore another special case of the mass action law condition.

const d = 1
1
function thinlayer(f, u, node, data)
    if node.region == 3
        react = (u[1] - u[2]) / d
        f[1] = react
        f[2] = -react
    end
    return nothing
end
thinlayer (generic function with 1 method)
system5 = make_system(thinlayer);
plot(mysolve(system5)...)

The solution looks very similar to the case of the jump condition, however here, the size of the jump is defined by the physics of the interface.

Multiple domains

From the above discussion it seems that discontinuous interface conditions can be formulated in a rather general way via linear or nonlinear robin boundary conditions for each of the adjacent discontinuous species. Technically, it is necessary to be able to access the adjacent bulk data.

In order to streamline the handling of multiple interfaces, we propose an API layer on top of the species handling of VoronoiFVM. We call these "meta species" "quantities".

We define a grid with N=6 subregions

N = 6
6
begin
    XX = collect(0:0.1:1)
    local xcoord = XX
    for i in 1:(N - 1)
        xcoord = glue(xcoord, XX .+ i)
    end
    grid2 = simplexgrid(xcoord)
    for i in 1:N
        cellmask!(grid2, [i - 1], [i], i)
    end
    for i in 1:(N - 1)
        bfacemask!(grid2, [i], [i], i + 2)
    end
end
gridplot(grid2; legend = :lt, resolution = (600, 200))

To work with quantities, we first introduce a new constructor call without the "physics" parameter:

system6 = VoronoiFVM.System(grid2)
VoronoiFVM.System{Float64, Float64, Int32, Int64, Matrix{Int32}}(
  grid = ExtendableGrids.ExtendableGrid{Float64, Int32}(dim=1, nnodes=61, ncells=60,
  nbfaces=7),  
  physics = Physics(storage=default_storage, ),  
  num_species = 0)

First, we introduce a continuous quantity which we name "cspec". Note that the "species number" can be assigned automatically if not given explicitly.

const cspec = ContinuousQuantity(system6, 1:N; ispec = 1)
ContinuousQuantity{Int32}(1, 1)

A discontinuous quantity can be introduced as well. by default, each reagion gets a new species number. This can be overwritten by the user. It is important that the speces numbers of neighboring regions differ.

const dspec = DiscontinuousQuantity(system6, 1:N; regionspec = [2 + i % 2 for i in 1:N])
DiscontinuousQuantity{Int32}(Int32[3, 2, 3, 2, 3, 2], 2)

For both quantities, we define simple diffusion fluxes:

function flux2(f, u, edge, data)
    f[dspec] = u[dspec, 1] - u[dspec, 2]
    return f[cspec] = u[cspec, 1] - u[cspec, 2]
end
flux2 (generic function with 1 method)

Define a thin layer interface condition for dspec and an interface source for cspec.

function breaction2(f, u, node, data)
    if node.region > 2
        react = (u[dspec, 1] - u[dspec, 2]) / d1
        f[dspec, 1] = react
        f[dspec, 2] = -react

        f[cspec] = -q1
    end
    return nothing
end
breaction2 (generic function with 1 method)

Add physics to the system, set dirichlet bc at both ends, and extract subgrids for plotting (until there will be a plotting API for this...)

begin
    physics!(system6, VoronoiFVM.Physics(; flux = flux2, breaction = breaction2))

    ## Set boundary conditions
    boundary_dirichlet!(system6, dspec, 1, g_1)
    boundary_dirichlet!(system6, dspec, 2, g_2)
    boundary_dirichlet!(system6, cspec, 1, 0)
    boundary_dirichlet!(system6, cspec, 2, 0)

    # ensure that `solve` is called only after this cell
    # as mutating circumvents the reactivity of the notebook
    physics_ok = true
end;
allsubgrids = subgrids(dspec, system6)
6-element Vector{ExtendableGrid{Float64, Int32}}:
 ExtendableGrids.ExtendableGrid{Float64, Int32}(dim=1, nnodes=11, ncells=10, nbfaces=2)
 ExtendableGrids.ExtendableGrid{Float64, Int32}(dim=1, nnodes=11, ncells=10, nbfaces=2)
 ExtendableGrids.ExtendableGrid{Float64, Int32}(dim=1, nnodes=11, ncells=10, nbfaces=2)
 ExtendableGrids.ExtendableGrid{Float64, Int32}(dim=1, nnodes=11, ncells=10, nbfaces=2)
 ExtendableGrids.ExtendableGrid{Float64, Int32}(dim=1, nnodes=11, ncells=10, nbfaces=2)
 ExtendableGrids.ExtendableGrid{Float64, Int32}(dim=1, nnodes=11, ncells=10, nbfaces=2)
if physics_ok
    sol6 = solve(system6; inival = 0.5)
end;
const d1 = 0.1
0.1
const q1 = 0.2
0.2
function plot2(U, subgrids, system6)
    dvws = VoronoiFVM.views(U, dspec, allsubgrids, system6)
    cvws = VoronoiFVM.views(U, cspec, allsubgrids, system6)
    vis = GridVisualizer(; resolution = (600, 300), legend = :rt)
    scalarplot!(
        vis,
        allsubgrids,
        grid2,
        dvws;
        flimits = (-0.5, 1.5),
        clear = false,
        color = :red,
        label = "discontinuous species"
    )
    scalarplot!(
        vis,
        allsubgrids,
        grid2,
        cvws;
        flimits = (-0.5, 1.5),
        clear = false,
        color = :green,
        label = "continuous species"
    )
    return reveal(vis)
end
plot2 (generic function with 1 method)
plot2(sol6, subgrids, system6)

Testing

if d1 == 0.1 && N == 6
    @test norm(system6, sol6, 2) ≈ 7.0215437706445245
end
Test Passed


Built with Julia 1.11.2 and

CairoMakie 0.12.18
ExtendableGrids 1.9.2
GridVisualize 1.7.0
HypertextLiteral 0.9.5
LinearAlgebra 1.11.0
LinearSolve 2.34.0
Pkg 1.11.0
PlutoUI 0.7.60
Revise 3.5.18
Test 1.11.0
VoronoiFVM 1.25.1