mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
fraunhoferPropagator.hpp
Go to the documentation of this file.
1/** \file fraunhoferPropagator.hpp
2 * \brief Declares and defines a class for Fraunhofer propagation of optical wavefronts
3 * \ingroup imaging
4 * \author Jared R. Males (jaredmales@gmail.com)
5 *
6 */
7
8//***********************************************************************//
9// Copyright 2015-2020 Jared R. Males (jaredmales@gmail.com)
10//
11// This file is part of mxlib.
12//
13// mxlib is free software: you can redistribute it and/or modify
14// it under the terms of the GNU General Public License as published by
15// the Free Software Foundation, either version 3 of the License, or
16// (at your option) any later version.
17//
18// mxlib is distributed in the hope that it will be useful,
19// but WITHOUT ANY WARRANTY; without even the implied warranty of
20// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21// GNU General Public License for more details.
22//
23// You should have received a copy of the GNU General Public License
24// along with mxlib. If not, see <http://www.gnu.org/licenses/>.
25//***********************************************************************//
26
27#ifndef wfp_fraunhoferPropagator_hpp
28#define wfp_fraunhoferPropagator_hpp
29
30#include "../math/constants.hpp"
31#include "imagingArray.hpp"
32#include "imagingUtils.hpp"
33
34#include "../math/fft/fft.hpp"
35
36namespace mx
37{
38
39namespace wfp
40{
41
42/// Class to perform Fraunhofer propagation between pupil and focal planes
43/** This class uses the FFT to propagate between planes, and normalizes so that flux
44 * is conserved. For propagation from pupil to focal plane, the pupil wavefront is tilted so that
45 * the focal-plane image is centered at the geometric center of the array. After propagation from
46 * focal plane to pupil plane, the pupil plane wavefront is un-tilted to restore the
47 * pupil to its original position.
48 *
49 * \tparam _wavefrontT is an Eigen::Array-like type, with std::complex values.
50 *
51 * \todo check if we should reverse the FFT orders to be proper
52 *
53 * \ingroup imaging
54 */
55template <typename _wavefrontT>
57{
58
59 public:
60 /// The wavefront data type
61 typedef _wavefrontT wavefrontT;
62
63 /// The complex data type
64 typedef typename wavefrontT::Scalar complexT;
65
66 /// The real data type
67 typedef typename wavefrontT::Scalar::value_type realT;
68
69 protected:
70 /// The size of the wavefront in pixels
72
73 realT m_xcen{ 0 }; ///< x-coordinate of focal plane center, in pixels
74 realT m_ycen{ 0 }; ///< x-coordinate of focal plane center, in pixels
75
76 /// Determines how the image is centered.
77 /** If 0 (default) it is at 0.5*(wfSz-1), if true it is shifted by 0.5*m_wholePixel in each axis.
78 */
80
81 /// Phase screen for tilting the pupil plane so that the focal plane image is centered.
83
84 /// Phase screen for un-tilting the pupil plane after propagating from a centered focal plane.
86
87 /// FFT object for forward FFTs
88 math::fft::fftT<complexT, complexT, 2, 0> m_fft_fwd;
89
90 /// FFT object for backward FFTs
91 math::fft::fftT<complexT, complexT, 2, 0> m_fft_back;
92
93 /// Initialize members
94 void initialize();
95
96 public:
97 /// Constructor
99
100 /// Destructor
102
103 /// Get the value of the wholePixel parameter
104 /** The wholePixel parameter determines how the image is centered. If 0 (default) it is at 0.5*(wfSz-1), if true it
105 * is shifted by 0.5*m_wholePixel in each axis.
106 *
107 * \returns the value of m_wholePixel
108 */
110
111 /// Set the value of the wholePixel parameter
112 /** The wholePixel parameter determines how the image is centered. If 0 (default) it is at 0.5*(wfSz-1), if true it
113 * is shifted by 0.5*m_wholePixel in each axis.
114 */
115 void wholePixel( realT wp /**< [in] the new wholePixel value */ );
116
117 /// Apply the shift to a pupil wavefront which will center the resultant focal plane image, and apply the
118 /// normalization.
119 /** You must have allocated the shift screens first, by calling propagatePupilToFocal, propagateFocalToPupil, or
120 * setWavefrontSizePixels.
121 */
122 void shiftPupil( wavefrontT &complexPupil /**< [in.out] the complex pupil plane wavefront to shift*/ );
123
124 /// Apply the shift to a pupil wavefront which will restore it to a centered pupil image, with correct flux.
125 /** You must have allocated the shift screens first, by calling propagatePupilToFocal, propagateFocalToPupil, or
126 * setWavefrontSizePixels.
127 */
128 void unshiftPupil( wavefrontT &complexPupil /**< [in.out] the complex pupil plane wavefront to shift*/ );
129
130 /// Propagate the wavefront from the pupil plane to the focal plane
131 /** The pupil plane wavefront (complexPupil) is multiplied by a tilt to place the
132 * image in the geometric center of the focal plane. This can be prevented by
133 * setting doCenter to false.
134 *
135 */
138 &complexFocal, ///< [out] the focal plane wavefront. Must be pre-allocated to same size as complexPupil.
139 wavefrontT &complexPupil, ///< [in] the pupil plane wavefront. Modified due to application of centering tilt.
140 bool doCenter = true ///< [in] [optional] set to false to not apply the centering shift
141 );
142
143 /// Propagate the wavefront from Focal plane to Pupil plane
144 /** After the fourier transform, the output pupil plane wavefront is de-tilted, restoring it
145 * to the state prior to calling \ref propagatePupilToFocal. This can be prevented by
146 * setting doCenter to false.
147 *
148 */
149 void propagateFocalToPupil( wavefrontT &complexPupil, ///< [out] the pupil plane wavefront. Must be pre-allocated to
150 ///< same size as complexFocal.
151 wavefrontT &complexFocal, ///< [in] the focal plane wavefront.
152 bool doCenter = true ///< [in] [optional] set to false to not apply the centering shift
153 );
154
155 /// Set the size of the wavefront, in pixels
156 /** Checks if the size changes, does nothing if no change. Otherwise, calls
157 * \ref makeShiftPhase to pre-calculate the tilt arrays and plans the FFTs.
158 *
159 */
160 void setWavefrontSizePixels( int wfsPix /**< [in] the desired new size of the wavefront */ );
161
162 protected:
163 /// Calculate the complex tilt arrays for centering and normalizing the wavefronts
164 /**
165 */
167
168}; // class fraunhoferPropagator
169
170template <typename wavefrontT>
172{
173 m_wavefrontSizePixels = 0;
174
175 m_xcen = 0;
176 m_ycen = 0;
177}
178
179template <typename wavefrontT>
183
184template <typename wavefrontT>
188
189template <typename wavefrontT>
191{
192 return m_wholePixel;
193}
194
195template <typename wavefrontT>
197{
198 m_wholePixel = wp;
199
200 // Re-make the shift phase if size already set.
201 if( m_wavefrontSizePixels > 0 )
202 makeShiftPhase();
203}
204
205template <typename wavefrontT>
207{
208 complexPupil *= m_centerFocal;
209}
210
211template <typename wavefrontT>
213{
214 complexPupil *= m_centerPupil;
215}
216
217template <typename wavefrontT>
219 wavefrontT &complexPupil,
220 bool doCenter /*default = true*/
221)
222{
223 // First setup the tilt screens (does nothing if there's no change in size)
224 setWavefrontSizePixels( complexPupil.rows() );
225
226 // Apply the centering shift -- this adjusts by 0.5 pixels and normalizes
227 if( doCenter )
228 shiftPupil( complexPupil );
229
230 // fft_fwd.fft(complexPupil.data(), complexFocal.data() );
231 m_fft_fwd( complexFocal.data(), complexPupil.data() );
232}
233
234template <typename wavefrontT>
236 wavefrontT &complexFocal,
237 bool doCenter /*default = true*/
238)
239{
240 // First setup the tilt screens (does nothing if there's no change in size)
241 setWavefrontSizePixels( complexPupil.rows() );
242
243 // fft_back.fft( complexFocal.data(), complexPupil.data());
244 m_fft_back( complexPupil.data(), complexFocal.data() );
245
246 // Unshift the wavefront and normalize
247 if( doCenter )
248 unshiftPupil( complexPupil );
249}
250
251template <typename wavefrontT>
253{
254 // If no change in size, do nothing
255 if( wfsPix == m_centerFocal.rows() )
256 return;
257
258 m_wavefrontSizePixels = wfsPix;
259
260 m_xcen = 0.5 * ( wfsPix - 1.0 );
261 m_ycen = 0.5 * ( wfsPix - 1.0 );
262
263 makeShiftPhase();
264
265 m_fft_fwd.plan( wfsPix, wfsPix );
266
267 m_fft_back.plan( wfsPix, wfsPix, MXFFT_BACKWARD );
268}
269
270template <typename wavefrontT>
272{
273 constexpr realT pi = math::pi<realT>();
274
275 // The normalization is included in the tilt.
276 realT norm = 1. / ( m_wavefrontSizePixels * sqrt( 2 ) );
277 complexT cnorm = complexT( norm, norm );
278
279 // Resize the center phases
280 m_centerFocal.resize( m_wavefrontSizePixels, m_wavefrontSizePixels );
281 m_centerPupil.resize( m_wavefrontSizePixels, m_wavefrontSizePixels );
282
283 // Shift by 0.5 pixels
284 realT arg = -2.0 * pi * 0.5 * ( m_wavefrontSizePixels - m_wholePixel ) / ( m_wavefrontSizePixels - 1 );
285
286 for( int ii = 0; ii < m_wavefrontSizePixels; ++ii )
287 {
288 for( int jj = 0; jj < m_wavefrontSizePixels; ++jj )
289 {
290 m_centerFocal( ii, jj ) = cnorm * exp( complexT( 0., arg * ( ( ii - m_xcen ) + ( jj - m_ycen ) ) ) );
291 m_centerPupil( ii, jj ) =
292 cnorm * exp( complexT( 0., 0.5 * pi - arg * ( ( ii - m_xcen ) + ( jj - m_ycen ) ) ) );
293 }
294 }
295}
296
297} // namespace wfp
298} // namespace mx
299
300#endif // wfp_fraunhoferPropagator_hpp
Class to perform Fraunhofer propagation between pupil and focal planes.
realT m_xcen
x-coordinate of focal plane center, in pixels
void unshiftPupil(wavefrontT &complexPupil)
Apply the shift to a pupil wavefront which will restore it to a centered pupil image,...
wavefrontT::Scalar::value_type realT
The real data type.
wavefrontT m_centerPupil
Phase screen for un-tilting the pupil plane after propagating from a centered focal plane.
void makeShiftPhase()
Calculate the complex tilt arrays for centering and normalizing the wavefronts.
int wholePixel()
Get the value of the wholePixel parameter.
void setWavefrontSizePixels(int wfsPix)
Set the size of the wavefront, in pixels.
realT m_ycen
x-coordinate of focal plane center, in pixels
void wholePixel(realT wp)
Set the value of the wholePixel parameter.
realT m_wholePixel
Determines how the image is centered.
int m_wavefrontSizePixels
The size of the wavefront in pixels.
wavefrontT m_centerFocal
Phase screen for tilting the pupil plane so that the focal plane image is centered.
void shiftPupil(wavefrontT &complexPupil)
wavefrontT::Scalar complexT
The complex data type.
void propagatePupilToFocal(wavefrontT &complexFocal, wavefrontT &complexPupil, bool doCenter=true)
Propagate the wavefront from the pupil plane to the focal plane.
void initialize()
Initialize members.
math::fft::fftT< complexT, complexT, 2, 0 > m_fft_fwd
FFT object for forward FFTs.
math::fft::fftT< complexT, complexT, 2, 0 > m_fft_back
FFT object for backward FFTs.
_wavefrontT wavefrontT
The wavefront data type.
void propagateFocalToPupil(wavefrontT &complexPupil, wavefrontT &complexFocal, bool doCenter=true)
Propagate the wavefront from Focal plane to Pupil plane.
constexpr floatT six_fifths()
Return 6/5 in the specified precision.
Declares and defines a class for managing images.
Utilities for modeling image formation.
The mxlib c++ namespace.
Definition mxError.hpp:106