Skip to content
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

Python interface for Gaussian beam source #1310

Merged
merged 7 commits into from
Aug 12, 2020
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
11 changes: 11 additions & 0 deletions python/meep.i
Original file line number Diff line number Diff line change
Expand Up @@ -1028,6 +1028,16 @@ void _get_gradient(PyObject *grad, PyObject *fields_a, PyObject *fields_f, PyObj
$1 = (std::complex<double> *)array_data($input);
}

// typemaps for gaussianbeam

%typecheck(SWIG_TYPECHECK_POINTER, fragment="NumPy_Fragments") std::complex<double> E0[3] {
$1 = is_array($input);
}

%typemap(in) std::complex<double> E0[3] {
oskooi marked this conversation as resolved.
Show resolved Hide resolved
std::complex<double> *pyE0 = (std::complex<double> *)array_data($input);
$1[0] = pyE0[0]; $1[1] = pyE0[1]; $1[2] = pyE0[2];
oskooi marked this conversation as resolved.
Show resolved Hide resolved
}

//--------------------------------------------------
// typemaps needed for get_eigenmode_coefficients
Expand Down Expand Up @@ -1648,6 +1658,7 @@ PyObject *_get_array_slice_dimensions(meep::fields *f, const meep::volume &where
Source,
SourceTime,
check_positive,
GaussianBeamSource,
)
from .visualization import (
plot2D,
Expand Down
31 changes: 29 additions & 2 deletions python/simulation.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@

import meep as mp
from meep.geom import Vector3, init_do_averaging
from meep.source import EigenModeSource, check_positive
from meep.source import EigenModeSource, GaussianBeamSource, check_positive
import meep.visualization as vis
from meep.verbosity_mgr import Verbosity

Expand Down Expand Up @@ -1997,14 +1997,27 @@ def get_epsilon_point(self, pt, frequency=0, omega=0):
Given a frequency `frequency` and a `Vector3` `pt`, returns the average eigenvalue
of the permittivity tensor at that location and frequency. If `frequency` is
non-zero, the result is complex valued; otherwise it is the real,
frequency-independent part of ε (the $\\omega\\to\\infty$ limit).
frequency-independent part of $\\varepsilon$ (the $\\omega\\to\\infty$ limit).
"""
v3 = py_v3_to_vec(self.dimensions, pt, self.is_cylindrical)
if omega != 0:
frequency = omega
warnings.warn("get_epsilon_point: omega has been deprecated; use frequency instead", RuntimeWarning)
return self.fields.get_eps(v3,frequency)

def get_mu_point(self, pt, frequency=0, omega=0):
"""
Given a frequency `frequency` and a `Vector3` `pt`, returns the average eigenvalue
of the permeability tensor at that location and frequency. If `frequency` is
non-zero, the result is complex valued; otherwise it is the real,
frequency-independent part of $\\mu$ (the $\\omega\\to\\infty$ limit).
"""
v3 = py_v3_to_vec(self.dimensions, pt, self.is_cylindrical)
if omega != 0:
frequency = omega
warnings.warn("get_mu_point: omega has been deprecated; use frequency instead", RuntimeWarning)
return self.fields.get_mu(v3,frequency)

def get_filename_prefix(self):
"""
Return the current prefix string that is prepended, by default, to all file names.
Expand Down Expand Up @@ -2234,6 +2247,20 @@ def add_source(self, src):
add_eig_src()
else:
add_eig_src(src.amp_func)
elif isinstance (src, GaussianBeamSource):
gaussianbeam_args = [
py_v3_to_vec(self.dimensions, src.beam_x0, is_cylindrical=self.is_cylindrical),
py_v3_to_vec(self.dimensions, src.beam_kdir, is_cylindrical=self.is_cylindrical),
src.beam_w0,
src.src.swigobj.frequency().real,
self.fields.get_eps(py_v3_to_vec(self.dimensions, src.center, self.is_cylindrical)).real,
self.fields.get_mu(py_v3_to_vec(self.dimensions, src.center, self.is_cylindrical)).real,
[src.beam_E0.x, src.beam_E0.y, src.beam_E0.z]
stevengj marked this conversation as resolved.
Show resolved Hide resolved
]
gaussianbeam = mp.gaussianbeam(*gaussianbeam_args)
add_vol_src_args = [src.src.swigobj, where, gaussianbeam]
add_vol_src = functools.partial(self.fields.add_volume_source, *add_vol_src_args)
add_vol_src()
else:
add_vol_src_args = [src.component, src.src.swigobj, where]
add_vol_src = functools.partial(self.fields.add_volume_source, *add_vol_src_args)
Expand Down
36 changes: 36 additions & 0 deletions python/source.py
Original file line number Diff line number Diff line change
Expand Up @@ -526,3 +526,39 @@ def eig_power(self,freq):
if callable(getattr(self.src, "fourier_transform", None)):
amp *= self.src.fourier_transform(freq)
return abs(amp)**2


class GaussianBeamSource(Source):

def __init__(self,
src,
center=None,
volume=None,
component=mp.ALL_COMPONENTS,
beam_x0=Vector3(),
beam_kdir=Vector3(),
beam_w0=None,
beam_E0=Vector3(),
**kwargs):

super(GaussianBeamSource, self).__init__(src, component, center, volume, **kwargs)
self._beam_x0 = beam_x0
self._beam_kdir = beam_kdir
self._beam_w0 = beam_w0
self._beam_E0 = beam_E0

@property
def beam_x0(self):
return self._beam_x0

@property
def beam_kdir(self):
return self._beam_kdir

@property
def beam_w0(self):
return self._beam_w0

@property
def beam_E0(self):
return self._beam_E0
2 changes: 1 addition & 1 deletion src/meep.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -1488,7 +1488,7 @@ class gaussianbeam {

public:
gaussianbeam(const vec &x0, const vec &kdir, double w0, double freq,
double eps, double mu, std::complex<double> EO[3]);
double eps, double mu, std::complex<double> E0[3]);
void get_fields(std::complex<double> *EH, const vec &x) const;
std::complex<double> get_E0(int n) const { return E0[n]; };

Expand Down