A Julia package for performing calculations with the ray transfer matrix (or ABCD) formalism, for both 1D ray tracing and Gaussian beam propagation in the paraxial approximation.
The following introduction to the package assumes familiarity with the ABCD formalism and its utility in optical analysis and design. In addition to the above links, the following are classic and useful introductory references:
- H. Kogelnik and T. Li, "Laser Beams and Resonators", Applied Optics 5, 1550-1567 (1966)
- A. E. Siegman, Lasers (University Science Books, Sausalito, 1986)
This package is not yet registered. It can be installed in Julia with the following:
Pkg.clone("git://github.com/ngedwin98/ABCDBeamTrace.jl.git")
The code currently requires Julia v0.6
.
This section discusses how we represent the ABCD model of optical elements and how to access their standard representation as numerical ray transfer matrices.
The basic type representing an optical element is Element
. As an abstract type, its implemented concrete types are (all physical distances taken to be in some consistent, arbitrary unit):
FreeSpace(L::Real)
: Represents a section of free space of distance represented byL
.ThinLens(f::Real)
: Represents a focusing element that behaves like a thin lens of focal length represented byf
. Optionally, a tilt angle represented byθ
in radians can be specified usingThinLens(f::Real,θ::Real)
; otherwise it defaults toθ=0
.Interface(η::Real)
: Represents an dielectric interface whereη
is the ratio of the initial to final refractive indices. Optionally, a tilt angle represented byθ
in radians and a radius of curvature (positive when concave side faces incoming beam) represented byR
can be specified usingInterface(η::Real,θ::Real,R::Real)
. Note that if either of these two are specified, they must both be specified, and otherwise they default toθ=0
andR=Inf
.
There are some additional optical elements which are physically distinct but which, in the paraxial approximation, can be represented in terms of the above elements. Currently, the following convenience methods allow such "elements" to be constructed:
Mirror(R::Real,θ::Real) = ThinLens(R/2,θ)
represents a curved mirror with radius of curvature (positive when concave side faces incoming beam) represented byR
and a tilt angle represented byθ
in radians.
An optical system in the ABCD formalism is a cascade of optical elements, represented in this package as Vector{<:Element}
. For example, to construct a Keplerian 2x beam expander with objective focal length represented by f::Real
, we can use
expander_2x = [ThinLens(f), FreeSpace(3f), ThinLens(2f)]
This means that we can also compose two optical systems using vector concatenations. For example, given any length represented by L::Real
, the following effectively creates a 1-to-1 beam expander with four lenses:
system = [expander_2x; FreeSpace(L); reverse(expander_2x)]
Note the use of vcat
in composing systems together.
The usual ABCD formalism for calculations involve the multiplication of 2x2 matrices, as discussed in the references above. To facilitate this formalism, each element has a well-defined matrix, which is accessed by calling RTM(e::Element)
. For example, RTM(ThinLens(100)) == [1 0 ; -1/100 1]
evaluates as true. Dot syntax for broadcasting can be used to create a corresponding vector of matrices, and the corresponding elements can be multiplied up, as in
# Total ABCD matrix for above 4-lens system
system_RTM = reduce(*, RTM.(system))
In the ABCD formalism, optical components that can have a non-zero tilt relative to the optical axis (i.e., with a nonzero field θ
) bifurcate into two distinct elements, acting differently on the sagittal and tangential components of the ray or Gaussian beam. To represent this dual behavior, composite Element
types called Tan
and Sag
wrap the fundamental Element
types and dispatch differently on RTM
in order to produce the respective ray transfer matrices for the tangential and sagittal components.
For example, suppose the above beam expander were misaligned with a tilt angle represented by θ::Real
. We would represent the optical system as
system = [ThinLens(f,θ), FreeSpace(3f), ThinLens(2f,θ)] # Component-agnostic description of system
system_tan = Tan.(system) # Elements as seen by tangential component
system_sag = Sag.(system) # Elements as seen by sagittal component
system_tan_RTM = reduce(*, RTM.(system_tan)) # Total matrix in the tangential component
system_sag_RTM = reduce(*, RTM.(system_sag)) # Total matrix in the sagittal component
Note: If no calls are made to either Tan
and Sag
(as in the first line above), the θ
field (if present) confers no effect; in this case, RTM
returns the ray transfer matrix as if we set θ=0
. Also, note that FreeSpace
does not wrap into these composite objects: Tan(e::FreeSpace) = Sag(e::FreeSpace) = e
. (This latter behavior could be changed, should there be a reason for it to be otherwise...)
The numerical ray transfer matrices in the ABCD formalism facilitate a transfer-function-based approach to optical analysis. However, it is also possible to take a state-based viewpoint, by representing the state of the ray or beam as it passes through the elements. Here, the parameters that describe the state of the beam are modified by each element, causing the beam to evolve as it propagates through the elements. This approach is most useful for visualizing the optical propagation of the beam, or for optimizing beam properties in the design.
Consistent with the ABCD formalism, we provide a description the state of a Gaussian beam (and the ray parameters) at a particular point along the optical axis. It is represented with a concrete type Beam
containing the following fields:
q
: The complex Gaussian beam parameterx
: The transverse component of the ray perpendicular to the optical axisk
: The slope of the ray relative to the optical axisz
: The location at which the state is being described, along the optical axisn
: The refractive index at this locationλ
: The physical wavelength of the (monochromatic) beam
All optical elements modify x
, k
, and q
, as prescribed by the ABCD formalism. The parameter z
is only modified by FreeSpace
(via addition by the field L
), while the parameter n
is only modified by Interface
(via division by the field η
). The wavelength λ
does not play a direct role in the (scale-free) beam propagation; it is only used for spot size calculations.
A useful constructor method is Beam(λ::Real,w0::Real,n0::Real=1)
, which represents a beam state with position represented by z=0
in refractive index represented by n0
and at a focus with 1/e² waist represented by w0
(so that the Gaussian beam parameter is imaginary), provided its wavelength λ
; furthermore, the ray position and slope are represented by x=0
and k=0
, respectively.
We think of the optical elements as effecting a transformation on the beam state. If we have an optical element represented by e::Element
and a beam state Γ::Beam
, there is a non-exported function transform(e::Element,Γ::Beam)
, which returns a new instance of Beam
with fields representing the beam state after propagation through that element. This function is unexported because it should rarely see utility given the beamtracing functionality described in the next subsection.
Note: It is very natural to use Julia's functor mechanism to represent the transformation, via some definition like (e::Element)(Γ::Beam)
which behaves identically to the currently-implemented transform(e,Γ)
. Unfortunately, this is not possible at the moment due to JuliaLang/julia#14919. While an obvious workaround is to define the transformation for each concrete subtype, this is inelegant enough to be not worth it.
Perhaps the most useful part of this whole package is the ability to trace the beam state by providing an initial beam state and propagating it forwards using the optical elements of a given system, recording the state after each element. For an optical system represented by elems::Vector{<:Element}
and an initial beam state represented by Γ0::Beam
, beamtrace(elems::Vector{<:Element},Γ0::Beam)
returns an instance of Vector{Beam}
(of length given by length(elems)+1
) where the first item is Γ0
, and each subsequent item is the result of applying transform
to the previous item using the corresponding item of elems
.
More detailed examples are upcoming. In particular, this package has been developed with the following tasks in mind:
- Cavity design
- Mode-matching
Jupyter notebooks implementing these tasks exist and are currently being cleaned so that they can be pushed to this repository to serve as examples.
Other interesting examples using this package are welcome!
- Add examples of beam and ray tracing function calls
- Add detailed Jupyter notebooks showing examples
- Systematic tests of correctness and consistency