In this example, we show how to use mesti2s() to compute the transmission matrix of a strongly scattering disordered medium, analyze the transmission matrix to determine an incident wavefront that can penetrate the disorder with almost 100% transmission (called an "open channel"), and then use mesti2s() again to compute the field profile of the open channel while comparing to that of a typical plane-wave input.
# load necessary packages
using MESTI, GeometryPrimitives, LinearAlgebra, Statistics, Printf
# include the function to build epsilon_xx for the disordered and plot the transmission eigvenvalue distribution
include("build_epsilon_disorder.jl")
include("plot_and_compare_distribution.jl")
# dimensions of the system, in units of the wavelength lambda_0
dx = 1/15 # discretization grid size
W = 360 # width of the scattering region
L = 90 # thickness of the scattering region
L_tot = 270 # full length of the system for plotting
r_min = 0.2 # minimal radius of the cylindrical scatterers
r_max = 0.4 # maximal radius of the cylindrical scatterers
min_sep = 0.05 # minimal separation between cylinders
number_density = 1.3 # number density, in units of 1/lambda_0^2
rng_seed = 0 # random number generator seed
# relative permittivity, unitless
epsilon_scat = 1.2^2 # cylindrical scatterers
epsilon_bg = 1.0^2 # background in the scattering region
epsilon_low = 1.0^2 # free space on the low side
epsilon_high = 1.0^2 # free space on the high side
yBC = "periodic" # boundary condition in y
# generate a random collection of non-overlapping cylinders
build_TM = true
(epsilon_xx, y0_list, z0_list, r0_list, y_Ex, z_Ex) =
build_epsilon_disorder(W, L, r_min, r_max, min_sep,
number_density, rng_seed, dx,
epsilon_scat, epsilon_bg, build_TM)
syst = Syst()
syst.epsilon_xx = epsilon_xx
syst.epsilon_low = epsilon_low
syst.epsilon_high = epsilon_high
syst.length_unit = "lambda_0"
syst.wavelength = 1
syst.dx = dx
syst.yBC = yBC
# specify the input and output
input = channel_type()
output = channel_type()
# input from the low side
input.side = "low"
# output to the high side
output.side = "high"
# put PML along z-direction
pml_npixels = 20
pml = PML(pml_npixels)
syst.zPML = [pml]
# transmission matrix: input from the low side, output to the high side
t, channels, _ = mesti2s(syst, input, output)
===System size===
ny_Ex = 5400; nz_Ex = 1349 => 1391 for Ex(y,z)
[N_prop_low, N_prop_high] = [725, 725] per polarization
yBC = periodic; zBC = [PML, PML]
Building B,C... elapsed time: 4.179 secs
... elapsed time: 2.163 secs
Building A ... elapsed time: 11.130 secs
< Method: APF using MUMPS in single precision with AMD ordering (symmetric K) >
Building K ... elapsed time: 6.310 secs
Analyzing ... elapsed time: 3.781 secs
Factorizing ... elapsed time: 122.678 secs
Total elapsed time: 160.125 secs
# perform the singular value decomposition (SVD) on the transmission matrix
_, sigma, v = svd(t)
# transmission eigenvalue (eigenvalue of t^(dag)*t) is the square of the signular value of t
tau = sigma.^2
# plot the transmission eigenvalue distribution and compare it with the DMPK theory
using Plots
plot_and_compare_distribution(tau)
# The most-open channels is the singular vector of the transmission matrix with
# the largest singular value.
v_open = v[:, 1]
sigma_open = sigma[1]
N_prop_low = channels.low.N_prop # number of propagating channels on the low side
ind_normal = Int(round((N_prop_low+1)/2)) # index of the normal-incident plane-wave
# compare the transmission
T_avg = sum(abs.(t).^2)/N_prop_low # average over all channels
T_PW = sum(abs.(t[:,ind_normal]).^2) # normal-incident plane-wave
T_open = sigma_open.^2 # open channel
println(" T_avg = ", @sprintf("%.2f", T_avg), "\n T_PW = ", @sprintf("%.2f", T_PW), "\n T_open = ", @sprintf("%.2f", T_open))
T_avg = 0.22
T_PW = 0.24
T_open = 1.00
# specify two input incident wavefronts:
# (1) normal-incident plane-wave
# (2) open channel
input = wavefront()
v_low = zeros(ComplexF64, N_prop_low, 2)
v_low[ind_normal, 1] = 1
v_low[:, 2] = v_open
input.v_low = v_low
# we will also get the field profile in the free spaces on the two sides, for
# plotting purpose.
opts = Opts()
nz_low = round(Int,(L_tot-L)/2/dx)
nz_high = nz_low
opts.nz_low = nz_low
opts.nz_high = nz_high
# opts.use_L0_threads = true would enhances the time performance,
# but marginally increases the memoery usage.
# the default of opts.use_L0_threads is true for 2D system.
# here we turn opts.use_L0_threads off, because this L0-threads layer
# in multithreads makes the numerical results slightly depends on the
# number of threads with acceptable error.
# later we want to show that the result from mesti() and mesti2s()
# should be exact the same later, so we turn it off to exclude the influence from
# the this L0-threads layer.
opts.use_L0_threads = false
# for field-profile computations
Ex, _, _ = mesti2s(syst, input, opts)
===System size===
ny_Ex = 5400; nz_Ex = 1349 => 1391 for Ex(y,z)
[N_prop_low, N_prop_high] = [725, 725] per polarization
yBC = periodic; zBC = [PML, PML]
Building B,C... elapsed time: 0.549 secs
... elapsed time: 0.343 secs
Building A ... elapsed time: 6.142 secs
< Method: factorize_and_solve using MUMPS in single precision with AMD ordering >
Analyzing ... elapsed time: 3.341 secs
Factorizing ... elapsed time: 103.652 secs
Solving ... elapsed time: 7.809 secs
... elapsed time: 22.006 secs
Total elapsed time: 147.811 secs
# do the same field-profile computations through defining matrix B by users and using mesti()
syst = Syst()
ny_Ex = size(epsilon_xx,1) # total number of pixel along y for Ex grids
# in mesti(), syst.epsilon_low and syst.epsilon_high are not needed
# in mesti(), we provide the whole epsilon_xx including the scattering region, source/detection region, and PML region
syst.epsilon_xx = cat(epsilon_low*ones(ny_Ex,pml_npixels+1), epsilon_xx, epsilon_high*ones(ny_Ex,pml_npixels+1), dims=2)
# in previous mesti2s() calculation,
# syst.epsilon_xx = epsilon_xx
# syst.epsilon_low = epsilon_low
# syst.epsilon_high = epsilon_high
syst.length_unit = "lambda_0"
syst.wavelength = 1
syst.dx = dx
syst.yBC = yBC
# put PML along z-direction
pml_npixels = 20
pml = PML(pml_npixels)
pml.direction = "z"
syst.PML = [pml]
# in previous mesti2s() calculation,
# syst.zPML = [pml] (and do not need to specify pml.direction = "z")
Bx = Source_struct()
f_prop_low_Ex = channels.f_x_m(channels.low.kydx_prop) # the transverse profiles, f_Ex, for the propagating ones
# If it is a single propagating channel, a line source of -2i*sqrt(nu)*f_Ex(m) at l=0 will generate an z-flux-normalized incident field of exp(i*kzdx*|l|)*f_Ex(l)/sqrt(nu), where nu = sin(kzdx)
# here, we want input wavefront fields on the low side, which is a superposition of propagating channels
# these input wavefront fields on the low side are specified by v_low, and we take superpositions of the propagating channels using the v coefficients, with the sqrt(nu)*exp(-i*kzdx*0.5) prefactors included.
# we will multiply the -2i prefactor at the end.
# the source B_Ex would generate the input wavefront fields
B_Ex_low = f_prop_low_Ex*(channels.low.sqrt_nu_prop.*exp.((-1im*0.5)*channels.low.kzdx_prop).*v_low) # 0.5 pixel backpropagation indicates that the source is half a pixel away from z = 0
Bx.data = [B_Ex_low] # the value of the source
# the position of the source specify by a rectangle.
# [m1, l1, w, h] specifies the location and the size of the
# rectangle. Here, (m1, l1) is the index of the (y,z) coordinate of
# the smaller-y, smaller-z corner of the rectangle, at the location
# of Ex(m1, l1); (w, h) is the width and height of the rectangle
# line source is put on the lower source region, where is one pixel outside PML region.
Bx.pos = [[1,pml_npixels+1,ny_Ex,1]]
opts = Opts()
# the -2i prefactor would be multiplied by.
opts.prefactor = -2im
opts.use_L0_threads = false
Ex_prime, _ = mesti(syst, [Bx], opts)
===System size===
ny_Ex = 5400; nz_Ex = 1381 for Ex(y,z)
UPML on -z +z sides; ; yBC = periodic; zBC = PEC
Building B,C... elapsed time: 0.001 secs
Building A ... elapsed time: 6.449 secs
< Method: factorize_and_solve using MUMPS in single precision with AMD ordering >
Analyzing ... elapsed time: 3.388 secs
Factorizing ... elapsed time: 102.293 secs
Solving ... elapsed time: 5.059 secs
Total elapsed time: 112.826 secs
# Excluding the extra padding and PML region, compare these field profiles
println("Maximum absolute value of field difference between constructing the source matrix B through mesti2s() and constructing by users = ",
maximum(abs.(Ex[:,nz_low+1:end-nz_high-1,:] - Ex_prime[:,pml_npixels+1+1:end-pml_npixels-1-1,:])))
Maximum absolute value of field difference between constructing the source matrix B through mesti2s() and constructing by users = 0.0
# normalize the field amplitude with respect to the plane-wave-input profile
Ex = Ex/maximum(abs.(Ex[:,:,1]))
nframes_per_period = 10
# extend the x coordinate to include free spaces on the two sides
z_Ex = vcat(z_Ex[1] .- (nz_low:-1:1)*dx, z_Ex, z_Ex[end] .+ (1:nz_high)*dx)
# animate the field profile with plane-wave input
anim_pw = @animate for ii ∈ 0:(nframes_per_period-1)
plt1 = (heatmap(z_Ex,collect(y_Ex),real.(Ex[:,:,1]*exp(-1im*2*π*ii/nframes_per_period)),
c = :balance, clims=(-1, 1), legend = :none,
aspect_ratio=:equal, dpi = 450,
ticks = false, framestyle = :none,
xlimits=(-90,180), ylimits=(0,360)))
scatter!(plt1, z0_list, y0_list,markersize=r0_list, alpha=0.3,
color=:black, legend=false, dpi = 450)
end
gif(anim_pw, "disorder_PW_input.gif", fps = 5)
# animate the field profile of the open channel
anim_open_ch = @animate for ii ∈ 0:(nframes_per_period-1)
plt2 = (heatmap(z_Ex,collect(y_Ex),real.(Ex[:,:,2]*exp(-1im*2*π*ii/nframes_per_period)),
c = :balance, clims=(-1, 1), legend = :none,
aspect_ratio=:equal, dpi = 450,
ticks = false, framestyle = :none,
xlimits=(-90,180), ylimits=(0,360)))
scatter!(plt2, z0_list, y0_list,markersize=r0_list, alpha=0.3,
color=:black, legend=false, dpi = 450)
end
gif(anim_open_ch, "disorder_open_channel.gif", fps = 5)