-
Notifications
You must be signed in to change notification settings - Fork 637
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
missing unit test for phase of mode coefficients #2409
Comments
Yes because FOMs must be purely real, right? Doesn't the power of a scattering coefficient implicitly test the imaginary part too? |
(Ah for some reason I thought this was wrt the gradients of the adjoint solver. But you're speaking generally, right?) |
I am referring to the mode-decomposition feature in general, not specific to its use as part of the objective functions of the adjoint solver. The tutorial Focusing Properties of a Metasurface Lens is the only example thus far where we actually make use of the complex mode coefficients (i.e., to design a lens using the quadratic formula for the local phase). I suppose we could adapt this example as the unit test but it seems overly complicated (i.e., it uses the near-to-far field transformation feature, etc.). |
If you just use a single interface, you can use the Fresnel formula. If it's just the flat interface, even for a 2d or 3d simulation you can make it 1 pixel wide. Basically, you would compute:
(Otherwise there is an arbitrary overall phase factor that depends on your conventions; it's not interesting or particularly meaningful to try to match the overallphase convention to that of the Fresnel formulas, particularly since we don't even document the phase choice imposed by MPB.) |
The reference analytic result which we will be validating the simulation results against — the complex Fresnel coefficients for a flat interface of two lossless materials with index Note: in this coordinate convention, TE corresponds to the |
From the Fresnel equations (shown above), a complex mode coefficient requires For the unit test, we can use the same parameters as the notes: (On slides 21 and 22, the notes review the operation of the Fresnel rhomb. This could be a useful tutorial demonstration.) |
Here is an initial attempt to validate the complex reflectance coefficient computed using the mode-decomposition feature in Meep with the Fresnel equations. The procedure for computing the reflectance coefficient in Meep is based on steps 1-3 from @stevengj's comment above. The setup is 2d (cell size is arbitrary) and consists of a flat interface of A comparison of the Meep and Fresnel results are shown in the tables below. Note that the magnitude of the reflectance coefficient (second column) is one in all cases. This is expected because of the TIR mode. However, there is an obvious discrepancy in the actual phase. Since the Meep results are converged with resolution, perhaps there is a bug in the calculation of the reflection coefficient itself?
'''
Computes the complex reflectance coefficient for a flat interface
of two lossless materials given a planewave at oblique incidence.
ref: https://www.patarnott.com/atms749/pdf/FresnelEquations.pdf
'''
import cmath
import math
import matplotlib
matplotlib.use('agg')
import matplotlib.pyplot as plt
import meep as mp
import numpy as np
def mode_coeff(dft_mon: mp.DftFlux) -> np.ndarray:
"""Computes the complex mode coefficients.
Args:
dft_mon: the DFT flux monitor object.
Returns:
The mode coefficients as a Numpy array of (bands, frequencies,
forward/backward).
"""
res = sim.get_eigenmode_coefficients(
dft_mon,
bands=[1],
eig_parity=mp.ODD_Z,
kpoint_func=lambda *not_used: k,
direction=mp.NO_DIRECTION,
)
return res.alpha
def refl_coeff_te(th: float, n1: float, n2: float) -> float:
"""Computes the reflectance coefficient for TE via the Fresnel equations.
Args:
th: angle of incident planewave (radians).
n1: refractive index of source medium.
n2: refractive index of medium adjacent to source medium.
Returns:
The reflectance coefficient (a complex number).
"""
return ((math.cos(th) - ((n2/n1)**2 - math.sin(th)**2)**0.5) /
(math.cos(th) + ((n2/n1)**2 - math.sin(th)**2)**0.5))
resolution = 100.
sx = 10.
sy = 5.
dpml = 1.
cell_size = mp.Vector3(sx + 2*dpml, sy, 0)
pml_layers = [mp.PML(dpml, direction=mp.X)]
n1 = 1.5
n2 = 1.0
# angle of incident planewave at center frequency
# 0° is +x; rotated CCW about +z axis
theta = np.radians(50.)
fcen = 1. # center frequency
df = 0.1 * fcen
cutoff = 10.0
# k (in source medium) with correct length
# plane of incidence is xy
k = mp.Vector3(n1*fcen, 0, 0).rotate(mp.Vector3(0, 0, 1), theta)
print("k:, ({:.6f}, {:.6f}, {:.6f})".format(k.x, k.y, k.z))
def pw_amp(k,x0):
def _pw_amp(x):
return cmath.exp(1j*2*math.pi*k.dot(x+x0))
return _pw_amp
src_pt = mp.Vector3(-0.5*sx, 0, 0)
sources = [
mp.Source(
mp.GaussianSource(fcen, fwidth=df, cutoff=cutoff),
component=mp.Ez, # S polarization
center=src_pt,
size=mp.Vector3(0, cell_size.y, 0),
amp_func=pw_amp(k, src_pt),
),
]
sim = mp.Simulation(
resolution=resolution,
cell_size=cell_size,
default_material=mp.Medium(index=n1),
boundary_layers=pml_layers,
k_point=k,
sources=sources,
)
# DFT monitor for reflected fields
refl_pt = mp.Vector3(-0.25*sx, 0, 0)
fluxreg = mp.FluxRegion(center=refl_pt, size=mp.Vector3(0, cell_size.y, 0))
refl_mon = sim.add_mode_monitor(fcen, 0, 1, fluxreg)
tol = 1e-8
sim.run(
until_after_sources=mp.stop_when_fields_decayed(
50,
mp.Ez,
src_pt,
tol,
),
)
input_mode_coeff = mode_coeff(refl_mon)[0, 0, 0]
input_flux_data = sim.get_flux_data(refl_mon)
sim.reset_meep()
geometry = [
mp.Block(
material=mp.Medium(index=n1),
center=mp.Vector3(-0.25*(sx+2*dpml), 0, 0),
size=mp.Vector3(0.5*(sx+2*dpml), mp.inf, mp.inf),
),
mp.Block(
material=mp.Medium(index=n2),
center=mp.Vector3(0.25*(sx+2*dpml), 0, 0),
size=mp.Vector3(0.5*(sx+2*dpml), mp.inf, mp.inf),
),
]
sim = mp.Simulation(
resolution=resolution,
cell_size=cell_size,
boundary_layers=pml_layers,
k_point=k,
sources=sources,
geometry=geometry,
)
refl_mon = sim.add_mode_monitor(fcen, 0, 1, fluxreg)
sim.load_minus_flux_data(refl_mon, input_flux_data)
fig, ax = plt.subplots()
sim.plot2D(ax=ax)
if mp.am_master():
fig.savefig('refl_coeff_flat_interface.png', dpi=150, bbox_inches='tight')
sim.run(
until_after_sources=mp.stop_when_fields_decayed(
50,
mp.Ez,
src_pt,
tol,
),
)
# mode coefficient of reflected planewave
refl_mode_coeff = mode_coeff(refl_mon)[0, 0, 1]
# reflection coefficient
refl_coeff = refl_mode_coeff / input_mode_coeff
# reflection coefficient (analytic)
Fresnel_refl_coeff = refl_coeff_te(theta, n1, n2)
print(f"refl-coeff:, {refl_coeff} (Meep), {Fresnel_refl_coeff} (Fresnel)")
print(f"phase:, {cmath.phase(refl_coeff)} (Meep), "
f"{cmath.phase(Fresnel_refl_coeff)} (Fresnel)") |
It seems that to demonstrate agreement in the phase of the reflected mode computed by Meep and the Fresnel equations requires placing the mode monitor exactly at the interface. The example above (incorrectly) placed the mode monitor halfway between the source and the interface. The position of the mode monitor is critical in calculations of the phase of the outgoing mode relative to the incident mode. With the mode monitor positioned at the interface, the phase values are in agreement (see table below). However, the magnitude of the reflectance coefficient computed by Meep is less than one which is incorrect. Could this simply be due to a discretization error?
|
Wouldn't a simpler test just be free-space propagation of a planewave in a uniform medium (or even a mode propagating down a waveguide)? Place two monitors an arbitrary distance apart and measure their phase difference (arg of the ratio of the complex coefficients). It should approach |
This could be an alternative test. The main reason for using the example of a flat interface is that its setup involving a line/area DFT monitor can be adapted to the diffraction orders of (metasurface) gratings which are of general interest. |
If you place it at a distance L from the interface, I forgot to mention that there is a frequency-dependent correction factor |
In one of the tests above, the reflectance coefficient measured by Meep is > r = -0.56759-0.82108j
> fcen = 1.0
> n1 = 1.5
> theta = math.radians(50.0)
> kx = math.cos(theta) * n1 * fcen
> L = 2.5
> corr_factor = cmath.exp(1j * kx * L) # phase correction factor
> cmath.phase(r / corr_factor)
1.6970909295581122 |
You forgot a 2π factor to convert |
Good catch. I updated the test in #2428 to include this phase-correction factor. |
Note that it's problematic to measure the reflection coefficient exactly at the interface, because the amplitude of the left-going wave drops discontinuously to zero across the interface. (Presumably, if you measure the left-going wave exactly at the interface in a discretized system like this, it gives some average of the correct results r and the 0 amplitude on the right of the interface. But that is equivalent to just multiplying r by a real number, which messes up the amplitude but not the phase, as you noted above.) |
The existing unit tests for$|S_{11}|^2$ , etc., which involves the magnitude of the complex mode coefficient. Currently, there are no tests for just the imaginary part of the mode coefficients (or alternatively the real and imaginary parts tested separately).
get_eigenmode_coefficients
includingtest_binary_grating.py
,test_mode_coeffs.py
,test_mode_decomposition.py
,test_special_kz.py
, and others all test purely real quantities such as the Poynting flux, e.g.A possible test could involve comparing the complex reflection and transmission coefficients of a 1d multilayer stack from Meep (2d and 3d simulations) to that computed analytically using a transfer-matrix approach. (We do not have to write our own TMM solver as part of this test. Instead, we can use hard-coded results obtained using a third-party package such as https://github.com/sbyrnes321/tmm.)
The text was updated successfully, but these errors were encountered: