In this example, we compute the reflection matrix of an open system with the input and output bases being Gaussian beams focused to different spatial locations. The magnitude of the diagonal elements of this matrix corresponds to what is measured in confocal microscopy. We first build the list of inputs B and list of outputs C, and then use mesti() to compute the reflection matrix.
# Call necessary packages
using MESTI, GeometryPrimitives, Plots
In this example, we consider a test system of a dielectric cylindrical scatterer located at (y0, z0) in air.
syst = Syst()
syst.length_unit = "µm"
syst.wavelength = 1.0 # wavelength (µm)
syst.dx = syst.wavelength/15 # discretization grid size (µm)
pml_npixels = 20 # number of PML pixels
W = 20 # width of simulation domain (including PML) (µm)
L = 10 # length of simulation domain (including PML) (µm)
r_0 = 0.75 # cylinder radius (µm)
n_bg = 1.0 # refractive index of the background
n_scat = 1.2 # refractive index of the cylinder
y_0 = W/2 # location of the cylinder
z_0 = L/2 # location of the cylinder
# Build the relative permittivity profile from subpixel smoothing
domain = Cuboid([W/2,L/2], [W,L]) # domain for subpixel smoothing cetering at (W/2, L/2) and with witdth W and thickness L
domain_epsilon = n_bg^2 # epsilon of the domain for subpixel smoothing
object = [Ball([y_0, z_0], r_0)] # object for subpixel smoothing: cylinder locating at (W/2, L/2) with radius r_0
object_epsilon = [n_scat^2] # epsilon of the object for subpixel smoothing
yBC = "PEC"; zBC = "PEC" # boundary conditions
epsilon_xx = mesti_subpixel_smoothing(syst.dx, domain, domain_epsilon, object, object_epsilon, yBC, zBC) # obtaining epsilon_xx from mesti_subpixel_smoothing()
ny_Ex, nz_Ex = size(epsilon_xx)
y = syst.dx:syst.dx:(W-syst.dx/2)
z = syst.dx:syst.dx:(L-syst.dx/2)
# Plot the relative permittivity profile
heatmap(z, y, epsilon_xx,
aspect_ratio=:equal,
xlabel = "z (µm)", ylabel = "y (µm)",
xlims=(0,10), title="εᵣ(y,z)",
c = cgrad(:grayC, rev=true))
We consider inputs being Gaussian beams focused at (yf, zf). In this example, we fix the focal depth at zf = z0 (i.e., the depth of the scatterer), and scan the transverse coordinate yf of the focus.
Perfect Gaussian beams can be generated with the total-field/scattered-field (TF/SF) method. But since the cross section of the beam decays exponentially in y, we can generate Gaussian beams to a high accuracy simply by placing line sources at a cross section on the low side, which is what we do here. We place the line sources at z = zs (location of the source plane), just in front of the PML.
To determine the required line sources, we (1) take the field profile of the desired incident Gaussian beam at the focal plane, Ein(y,zf) = Eyf(y,zf) = exp(-(y - yf)2/w2), (2) project it onto the propagating channels (i.e., ignoring evanescent contributions) of free space, (3) back propagate it to the source plane to determine Eyf(y,zs) in the propagating-channel basis, and (4) determine the line source necessary to generate such Eyf(y,zs).
# Parameters of the input Gaussian beams
NA = 0.5 # numerical aperture
z_f = z_0 # location of the focus in z (fixed)
y_f_start = 0.3*W # starting location of the focus in y
y_f_end = 0.7*W # ending location of the focus in y
dy_f = syst.wavelength/(10*NA) # spacing of focal spots in y
# Parameters of the line sources
n_source = pml_npixels + 1 # index of the source plane
z_s = z[n_source] # location of the source plane
z_d = z_s # location of the detection plane
w_0 = syst.wavelength/(pi*NA) # beam radius at z = z_f
y_f = y_f_start:dy_f:y_f_end # list of focal positions in y
M_in = length(y_f) # number of inputs
# Step 1: Generate the list of E_yf(z=z_f, y).
# Here, y is an ny_Ex-by-1 column vector, and y_f is a M_in-by-1 column vector.
# So, y .- transpose(y_f) is an ny_Ex-by-M_in matrix by implicit expansion.
# Then, E_yf is an ny_Ex-by-M_in matrix whose m-th column is the cross section
# of the m-th Gaussian beam at z = z_f.
E_yf = exp.(-(y .- transpose(y_f)).^2/(w_0^2)) # size(E_yf) = [ny_Ex, M_in]
# Get properties of propagating channels in the free space.
# We use PEC as the boundary condition for such channels since the default
# boundary condition in mesti() is PEC for TM waves, but the choice has
# little effect since E_yf should be exponentially small at the boundary of
# the simulation domain.
channels = mesti_build_channels(ny_Ex, "PEC", (2*pi/syst.wavelength)*syst.dx, n_bg^2)
# Transverse profiles of the propagating channels. Each column of u is
# one transverse profile. Different columns are orthonormal.
f_transverse = channels.f_x_m(channels.kydx_prop) # size(f_transverse) = [ny_Ex, N_prop]
# Step 2: Project E_yf(y, z_f) onto the propagating channels and obtain v_f.
# sqrt_nu_prop is the square root of nu for the propagating channel.
sqrt_nu_prop = reshape(channels.sqrt_nu_prop, :, 1)
# amplitude coefficients at z = z_f
v_f = (sqrt_nu_prop.* adjoint(f_transverse))*E_yf # size(v_f) = [N_prop, M_in]
# Step 3: Back propagate from z = z_f to z = z_s.
# This step assumes a PEC boundary in y, so it is not exact with PML in y,
# but it is sufficiently accurate since E_yf decays exponentially in y.
# Note we use implicit expansion here.
kz = reshape(channels.kzdx_prop/syst.dx, :, 1) # list of wave numbers
# amplitude coefficients at z = z_s
v_s = exp.(1im*kz*(z_s-z_f)).*v_f # size(v_s) = [N_prop, M_in]
# Step 4: Determine the line sources.
# In a closed geometry with no PML in y, a line source of
# -2i*sqrt(nu[a])*f_transverse[:,a] generates outgoing waves with transverse profile
# f_transverse[:,a]/sqrt(nu[a]). With PML in y, this is not strictly true but is sufficiently
# accurate since E_yf(y,z=z_s) decays exponentially in y.
# Note we use implicit expansion here.
B_low = (f_transverse.*transpose(channels.sqrt_nu_prop))*v_s # size(B_low) = [ny_Ex, M_in]
# We take the -2i prefactor out, to be multiplied at the end. The reason
# will be clear when we handle C below.
opts = Opts()
opts.prefactor = -2im
# In mesti(), Bx.pos = [m1, l1, w, h] specifies the position of a
# block source, where (m1, l1) is the index of the smaller-(y,z) corner,
# and (w, h) is the width and height of the block. Here, we put line
# sources (w=1) at l1 = l_source that spans the whole width of the
# simulation domain (m1=1, h=ny_Ex).
Bx = Source_struct()
Bx.pos = [[1, n_source, ny_Ex, 1]]
# Bx.data specifies the source profiles inside such block, with
# Bx.data[1][:, a] being the a-th source profile.
Bx.data = [B_low]
# We check that the input sources are sufficiently localized with little
# penetration into the PML; otherwise the Gaussian beams will not be
# accurately generated.
heatmap(1:M_in, collect(y), abs.(B_low),
aspect_ratio=:equal,
xlabel = "Input index", ylabel = "y (µm)",
title="|B_low|", c =cgrad(:grayC, rev=true))
We consider output projections onto the same set of Gaussian beams $\Psi$yf(y,zf) = exp(-(y - yf)2/w2) focused at (yf, zf),with the projection done at the detection plane (z = zd) same as the source plane (z = zs).
When the system has a closed boundary in y, as is the case in mesti2s(), the set of transverse modes form a complete and orthonormal basis, so it is clear what the output projection should be. But the Gaussian beams here are not orthogonal to each other, are not normalized, and do not form a complete basis. So, it is not obvious how our output projection should be defined in the detection plane.
What we do here is to project Gaussian beam at the focal plane onto the complete and orthonormal basis of transverse modes, do the propagating from focal to detection plane, and find the projection profiles. Specifically, we (1) project the Gaussian beam onto the propagating channels (i.e., ignoring evanescent contributions) of free space; (2) propagate such Gaussian beam in transverse mode basis to the detection plane at z = zd (3) reconstruct the Gaussian beams at the detection plane, and (4) construct the projection profile for the Gaussian beams at the detection plane.
Above, the incident field Ein(y,z) was not subtracted. Contribution from the incident field will be subtracted using matrix D in the next step.
# We perform the output projection on the same plane as the line source
Cx = Source_struct()
Cx.pos = Bx.pos
# size(Psi_yf) = [ny_Ex, num_of_Gaussian_input]
Psi_yf = exp.(-(y .- transpose(y_f)).^2/(w_0^2))
# Step 1: projecting Psi_yf onto the propagating channels and obtaining
# amplitude coefficients v_f_tilde
# size(v_f_tilde) = [num_of_prop_channel, num_of_Gaussian_output]
v_f_tilde = (sqrt_nu_prop.*adjoint(f_transverse))*Psi_yf
# Step 2: propagating the amplitude coefficient from plane z = z_f to plane z = z_d
# size(v_d) = [num_of_prop_channel, num_of_Gaussian_output]
v_d = exp.(1im*(-kz)*(z_d-z_f)).*v_f_tilde
# Step 3: Reconstruct Psi_yf at the detection plane z = z_d
# Psi_yd_zd = (f_transverse./transpose(sqrt_nu_prop))*v_d
# Step 4: We obtain the projection profile based on Psi_yd and v_d.
C_low = adjoint(v_d)*(sqrt_nu_prop.*adjoint(f_transverse))
# Normally, the next step would be
# C_struct.data = C_low.';
# However, we can see that C_low equals transpose(B_low)
println("max(|C_low-transpose(B_low)|) = $(maximum(abs.(C_low - transpose(B_low))))")
max(|C_low-transpose(B_low)|) = 1.1188630228279524e-16
# That means we will have C = transpose(B). So, we can save some computing
# time and memory usage by specifying C = transpose(B).
# This is expected by reciprocity -- when the set of inputs equals the set
# of outputs, we typically have C = transpose(B) or its permutation.
C_x = nothing
C = "transpose(B)";
The scattering matrix is given by S = C*inv(A)*B - D, with D = C*inv(A0)*B - S0 where A0 is a reference system for which its scattering matrix S0 is known. We consider A0 to be a homogeneous space with no scatterers, for which the reflection matrix S0 is zero.
pml = PML(pml_npixels)
pml.direction = "all"
syst.PML = [pml] # PML on all four sides
# For a homogeneous space, the length of the simulation domain doesn't
# matter, so we choose a minimal thickness of nz_Ex_temp = n_source + pml_npixels
# where n_source = pml_npixels + 1 is the index of the source plane.
syst.epsilon_xx = n_bg^2*ones(ny_Ex, n_source + pml_npixels)
D, _ = mesti(syst, [Bx], C, opts)
===System size===
ny_Ex = 299; nz_Ex = 41 for Ex(y,z)
UPML on -y +y -z +z sides; ; yBC = PEC; zBC = PEC
Building B,C... elapsed time: 0.575 secs
Building A ... elapsed time: 2.422 secs
< Method: APF using MUMPS in single precision with AMD ordering (symmetric K) >
Building K ... elapsed time: 0.485 secs
Analyzing ... elapsed time: 0.046 secs
Factorizing ... elapsed time: 0.048 secs
Total elapsed time: 5.444 secs
# Compute the reflection matrix
syst.epsilon_xx = epsilon
r, _ = mesti(syst, [Bx], C, D, opts)
===System size===
ny_Ex = 299; nz_Ex = 149 for Ex(y,z)
UPML on -y +y -z +z sides; ; yBC = PEC; zBC = PEC
Building B,C... elapsed time: 0.000 secs
Building A ... elapsed time: 0.159 secs
< Method: APF using MUMPS in single precision with AMD ordering (symmetric K) >
Building K ... elapsed time: 0.136 secs
Analyzing ... elapsed time: 0.017 secs
Factorizing ... elapsed time: 0.145 secs
Total elapsed time: 0.992 secs
For most applications, it is not necessary to compute the full field profile, since most experiments measure properties in the far field. Here, we compute the full field profile for the purpose of visualizing the system as the incident Gaussian beams are scanned across y.
# Exclude the PML pixels from the returned field profiles
opts.exclude_PML_in_field_profiles = true
y_interior = y[(pml_npixels+1):(ny_Ex-pml_npixels)]
z_interior = z[(pml_npixels+1):(nz_Ex-pml_npixels)]
field_profiles, _ = mesti(syst, [Bx], opts)
===System size===
ny_Ex = 299; nz_Ex = 149 for Ex(y,z)
UPML on -y +y -z +z sides; ; yBC = PEC; zBC = PEC
Building B,C... elapsed time: 0.000 secs
Building A ... elapsed time: 0.157 secs
< Method: factorize_and_solve using MUMPS in single precision with AMD ordering >
Analyzing ... elapsed time: 0.014 secs
Factorizing ... elapsed time: 0.170 secs
Solving ... elapsed time: 0.495 secs
Total elapsed time: 0.913 secs
theta = range(0, 2*pi, 100)
circ_x = cos.(theta)
circ_y = sin.(theta)
# Loop through Gaussian beams focused at different locations.
anim = @animate for ii ∈ 1:M_in
plt1 = heatmap(collect(z_interior), collect(y_interior), real(field_profiles[:, :, ii]),
xlims=(z_interior[1], z_interior[end]), aspect_ratio=:equal,
c = :balance, xticks = false, yticks = false, colorbar = false)
plot!(plt1, z_0.+r_0*circ_x, y_0.+r_0*circ_y, lw=1, color=:black,
xlims=(z_interior[1], z_interior[end]), ylims=(y_interior[1], y_interior[end]), legend=false)
plt2 = heatmap(collect(y_f), collect(y_f), abs.(r).^2,
xlims=(6,14), aspect_ratio=:equal,
xlabel = "Input position", ylabel = "Output position",
c =cgrad(:grayC, rev=true), xticks = false, yticks = false, colorbar = false)
plot!(plt2, [y_f[ii]], seriestype = :vline, linecolor=:blue, legend=false)
plot(plt1, plt2, layout = (1, 2))
end
gif(anim, "reflection_matrix_Gaussian_beams.gif", fps = 10)