mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
lyotCoronagraph.hpp
Go to the documentation of this file.
1/** \file lyotCoronagraph.hpp
2 * \brief Declares and defines a class to describe and optimize a Lyot Coronagraph.
3 * \ingroup imaging_files
4 * \author Jared R. Males (jaredmales@gmail.com)
5 *
6 */
7
8//***********************************************************************//
9// Copyright 2015, 2016, 2017, 2018 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 __lyotCoronagraph_hpp__
28#define __lyotCoronagraph_hpp__
29
30#include <iostream>
31
32#include "imagingArray.hpp"
33#include "imagingUtils.hpp"
35
36#include "../ioutils/fits/fitsFile.hpp"
37#include "../ioutils/fits/fitsUtils.hpp"
38
39#include "../improc/eigenImage.hpp"
40#include "../improc/eigenCube.hpp"
41
42namespace mx
43{
44
45namespace wfp
46{
47
48/// The Lyot Coronagraph
49/** A generalized Lyot coronagraph, which can include a pupil apodization, a focal plane mask with complex transmission,
50 * and a Lyot stop. Light is propagated through the coronagraph plane-by-plane, and the complex wavefront can be
51 * accessed at any plane. Also provides functions for optimizing the pupil apodizer and focal plane mask.
52 *
53 * \ingroup coronagraphs
54 */
55template <typename _realT, typename _fpmaskFloatT>
57{
58 public:
59 /// The real floating point type
60 typedef _realT realT;
61
62 /// The real floating point type for mask calculations.
63 typedef _fpmaskFloatT fpmaskFloatT;
64
65 /// The complex floating point type
66 typedef std::complex<realT> complexT;
67
68 /// The wavefront complex field type
69 typedef mx::wfp::imagingArray<std::complex<realT>, fftwAllocator<std::complex<realT>>, 0> complexFieldT;
70
71 /// The focal plane mask type
72 typedef Eigen::Array<std::complex<fpmaskFloatT>, Eigen::Dynamic, Eigen::Dynamic> fpMaskT;
73
74 /// The image type
75 typedef Eigen::Array<realT, Eigen::Dynamic, Eigen::Dynamic> imageT;
76
77 public:
78 /// The directory where coronagraph files are stored
79 std::string m_fileDir{ "coron" };
80
81 /// The linear size of the wavefront in pixels
82 int m_wfSz{ 0 };
83
84 /// Image containing the pupil apodization.
86
87 int m_maskSource; ///< 0= read from file, 1 = constructed by makeFocalMask, 2 = trans optimized.
88 std::string m_maskFile; ///< Name of file from which mask was loaded.
89
90 realT m_maskRad; ///< Radius of mask if it was constructed.
91 realT m_maskTrans; ///< Transmission of mask if it was constructed.
92
93 /// The focal plane mask.
95
96 /// Image containing the lyot stop.
98
99 complexFieldT m_focalPlane;
100
101 bool m_savePreMaskFocalPlane{ false };
102 complexFieldT m_preMaskFocalPlane;
103
104 bool m_savePreLyotPupilPlane{ false };
105 complexFieldT m_preLyotPupilPlane;
106
107 /// Fraunhofer propagator
109
110 public:
111 /// Default c'tor
113
114 /// Get the wavefront size in pixels
115 /**
116 * \returns the wavefront size in pixels
117 */
118 int wfSz();
119
120 /// Set the wavefront size in pixels.
121 /**
122 */
123 void wfSz( int sz /**< [in] is the new size */ );
124
125 /// Make the focal plane mask
126 void makeFocalMask( realT rad, fpmaskFloatT trans = 0.0, int sz = 0.0 );
127
128 /// Load the apodizer from a FITS file
129 /**
130 * \returns 0 on success.
131 * \returns -1 on error.
132 */
133 int loadApodizer( const std::string &apodName /**< [in] is the name of the FITS file containing the apodizer. */ );
134
135 /// Load the focal plane mask from a FITS file
136 /**
137 * \returns 0 on success.
138 * \returns -1 on error.
139 */
140 int loadFocalMask(
141 const std::string &fpmName /**< [in] is the name of the FITS file containing the focal plane mask. */ );
142
143 /// Load the Lyot stop from a FITS file
144 /**
145 * \returns 0 on success.
146 * \returns -1 on error.
147 */
148 int loadLyotStop( const std::string &lyotName /**< [in] is the name of the FITS file containing the lyot stop. */ );
149
150 /// Load the components of the coronagraph from FITS files
151 /**
152 * \returns 0 on success.
153 * \returns -1 on error.
154 */
155 int
156 loadCoronagraph( const std::string &apodName, ///< [in] is the name of the FITS file containing the apodizer.
157 const std::string &fpmName, ///< [in] is the name of the FITS file containing the focal plane mask.
158 const std::string &lyotName ///< [in] is the name of the FITS file containing the lyot stop.
159 );
160
161 /// Load the components of the coronagraph based in its base name
162 /** Looks in _filDir for the files.
163 *
164 * \returns 0 on success.
165 * \returns -1 on error.
166 */
167 int
168 loadCoronagraph( const std::string
169 &cName /**< [in] is the base name of the coronagraph without directory or file extension. */ );
170
171 void applyApodizer( complexFieldT &pupilPlane );
172 void applyFocalMask( complexFieldT &focalPlane );
173 void applyLyotStop( complexFieldT &lyotPlane );
174
175 /// Propagate the given pupil-plane wavefront through the coronagraph.
176 int propagate(
178 &pupilPlane /**< [in.out] The wavefront at the input pupil plane. It is modified by the coronagraph. */ );
179
180 int
181 propagate( imageT &fpIntensity, ///< [out] The intensity image in the focal plane. This should be pre-allocated.
183 &pupilPlane ///< [in.out] The wavefront at the input pupil plane. It is modified by the coronagraph.
184 );
185
186 /// Propagate the given pupil-plane wavefront without the coronagraph.
187 /** For a Lyot coronagraph, this applies the pupil apodization and Lyot stop, but not the FPM, to the given
188 * pupil-plane wavefront such that the result will produce the non-coronagraphic (off-axis) PSF.
189 */
190 int propagateNC(
192 &pupilPlane /**< [in.out] The wavefront at the input pupil plane. It is modified by the coronagraph. */ );
193
194 int propagateNC(
195 imageT &fpIntensity, ///< [out] The intensity image in the focal plane. This should be pre-allocated.
197 &pupilPlane ///< [in.out] The wavefront at the input pupil plane. It is modified by the coronagraph.
198 );
199
200 /// Optimize the pupil amplitude apodization and focal-plane mask complex transmission.
201 /** Uses the algorithm in Guyon (2014) \cite guyon_2014 for optimizing an Apodized-Pupil Lyot Complex Mask
202 * Coronagraph (APLCMC).
203 *
204 */
205 void
206 optimizeAPLCMC( imageT &geomPupil, ///< The geometric pupil mask, binary 1/0 transmission.
207 realT fpmRadPix, ///< The radius in pixels of the FPM.
208 realT relTol, ///< Relative tolerance for convergence.
209 realT absTol, ///< Absolute tolerance for convergence.
210 int maxIter, ///< Maximum number of iterations to allow.
211 const std::string &cname ///< Name of coronagraph, used as base-name for output files in m_fileDir.
212 );
213
214 void
215 optimizeAPLCMC( imageT &geomPupil, ///< The geometric pupil mask, binary 1/0 transmission.
216 realT fpmRadPix, ///< The radius in pixels of the FPM.
217 imageT &fpmPhase,
218 realT relTol, ///< Relative tolerance for convergence.
219 realT absTol, ///< Absolute tolerance for convergence.
220 int maxIter, ///< Maximum number of iterations to allow.
221 const std::string &cname ///< Name of coronagraph, used as base-name for output files in m_fileDir.
222 );
223};
224
225template <typename _realT, typename _fpmaskFloatT>
229
230template <typename _realT, typename _fpmaskFloatT>
232{
233 return m_wfSz;
234}
235
236template <typename _realT, typename _fpmaskFloatT>
238{
239 m_wfSz = sz;
240
241 m_fp.setWavefrontSizePixels( sz );
242 m_focalPlane.resize( sz, sz );
243}
244
245template <typename _realT, typename _fpmaskFloatT>
246void lyotCoronagraph<_realT, _fpmaskFloatT>::makeFocalMask( _realT rad, _fpmaskFloatT trans, int sz )
247{
248 if( sz == 0 )
249 {
250 sz = 2 * rad + 1;
251 if( sz % 2 == 1 )
252 ++sz;
253 }
254
255 // Make a real circular mask
257 mx::wfp::circularPupil( fpm, 0., rad );
258
259 // Convert to complex amplitude
260 m_focalMask.resize( sz, sz );
261 for( int i = 0; i < sz; ++i )
262 {
263 for( int j = 0; j < sz; ++j )
264 {
265 if( fpm( i, j ) == 1 )
266 m_focalMask( i, j ) = std::complex<fpmaskFloatT>( trans, 0 );
267 else
268 m_focalMask( i, j ) = std::complex<fpmaskFloatT>( 1, 0 );
269 }
270 }
271
272 m_maskSource = 1;
273 m_maskFile = "";
274 m_maskRad = rad;
275 m_maskTrans = 0.0;
276}
277
278template <typename _realT, typename _fpmaskFloatT>
280{
282
283 return ff.read( m_pupilApodizer, apodName );
284}
285
286template <typename _realT, typename _fpmaskFloatT>
288{
291
292 if( ff.read( fpm, fpmName ) < 0 )
293 return -1;
294
295 if( fpm.planes() == 1 )
296 {
297 m_focalMask = fpm.image( 0 );
298 }
299 else if( fpm.planes() == 2 )
300 {
301 m_focalMask.resize( fpm.rows(), fpm.cols() );
302
303 for( int r = 0; r < fpm.rows(); ++r )
304 {
305 for( int c = 0; c < fpm.cols(); ++c )
306 {
307 m_focalMask( r, c ) =
308 fpm.image( 0 )( r, c ) * exp( std::complex<fpmaskFloatT>( 0, fpm.image( 1 )( r, c ) ) );
309 }
310 }
311 }
312 else
313 {
314 std::cerr << "too many planes in focal mask file\n";
315 return -1;
316 }
317
318 m_maskSource = 0;
319 m_maskFile = fpmName;
320 m_maskRad = 0.0;
321 m_maskTrans = 0.0;
322
323 return 0;
324}
325
326template <typename _realT, typename _fpmaskFloatT>
328{
330
331 return ff.read( m_lyotStop, lyotName );
332}
333
334template <typename _realT, typename _fpmaskFloatT>
336 const std::string &fpmName,
337 const std::string &lyotName )
338{
339 if( loadApodizer( apodName ) < 0 )
340 return -1;
341 if( loadFocalMask( fpmName ) < 0 )
342 return -1;
343 if( loadLyotStop( lyotName ) < 0 )
344 return -1;
345
346 return 0;
347}
348
349template <typename _realT, typename _fpmaskFloatT>
351{
352 if( m_fileDir == "" )
353 {
354 mxError( "lyotCoronagraph", MXE_PARAMNOTSET, "file directory (fileDir) not set." );
355 return -1;
356 }
357
358 std::string apodName = m_fileDir + "/" + cName + "_apod.fits";
359 std::string fpmName = m_fileDir + "/" + cName + "_fpm.fits";
360 std::string lyotName = m_fileDir + "/" + cName + "_lyot.fits";
361
362 return loadCoronagraph( apodName, fpmName, lyotName );
363}
364
365template <typename _realT, typename _fpmaskFloatT>
366void lyotCoronagraph<_realT, _fpmaskFloatT>::applyApodizer( complexFieldT &pupilPlane )
367{
368 int sz = m_pupilApodizer.rows();
369
370 realT w = 0.5 * ( sz - 1.0 );
371
372 realT xc = 0.5 * ( pupilPlane.rows() - 1 );
373 realT yc = 0.5 * ( pupilPlane.cols() - 1 );
374
375 for( int i = 0; i < pupilPlane.rows(); ++i )
376 {
377 for( int j = 0; j < pupilPlane.cols(); ++j )
378 {
379 if( i >= xc - w && i <= xc + w && j >= yc - w && j <= yc + w )
380 pupilPlane( i, j ) *= m_pupilApodizer( (int)( i - ( xc - w ) ), (int)( j - ( yc - w ) ) );
381 else
382 pupilPlane( i, j ) *= 0;
383 }
384 }
385}
386
387template <typename _realT, typename _fpmaskFloatT>
388void lyotCoronagraph<_realT, _fpmaskFloatT>::applyFocalMask( complexFieldT &focalPlane )
389{
390 int sz = m_focalMask.rows();
391
392 realT w = 0.5 * ( sz - 1.0 );
393
394 realT xc = 0.5 * ( focalPlane.rows() - 1 );
395 realT yc = 0.5 * ( focalPlane.cols() - 1 );
396
397 for( int i = 0; i < sz; ++i )
398 {
399 for( int j = 0; j < sz; ++j )
400 {
401 focalPlane( xc - w + i, yc - w + j ) *= m_focalMask( i, j );
402 }
403 }
404}
405
406template <typename _realT, typename _fpmaskFloatT>
407void lyotCoronagraph<_realT, _fpmaskFloatT>::applyLyotStop( complexFieldT &lyotPlane )
408{
409 int sz = m_lyotStop.rows();
410
411 realT w = 0.5 * ( sz - 1.0 );
412
413 realT xc = 0.5 * ( lyotPlane.rows() - 1 );
414 realT yc = 0.5 * ( lyotPlane.cols() - 1 );
415
416 for( int i = 0; i < lyotPlane.rows(); ++i )
417 {
418 for( int j = 0; j < lyotPlane.cols(); ++j )
419 {
420 if( i >= xc - w && i <= xc + w && j >= yc - w && j <= yc + w )
421 lyotPlane( i, j ) *= m_lyotStop( (int)( i - ( xc - w ) ), (int)( j - ( yc - w ) ) );
422 else
423 lyotPlane( i, j ) = 0;
424 }
425 }
426}
427
428template <typename _realT, typename _fpmaskFloatT>
430{
431 applyApodizer( pupilPlane );
432
433 m_fp.propagatePupilToFocal( m_focalPlane, pupilPlane );
434
435 if( m_savePreMaskFocalPlane )
436 {
437 m_preMaskFocalPlane = m_focalPlane;
438 }
439
440 applyFocalMask( m_focalPlane );
441
442 m_fp.propagateFocalToPupil( pupilPlane, m_focalPlane );
443
444 if( m_savePreLyotPupilPlane )
445 {
446 m_preLyotPupilPlane = pupilPlane;
447 }
448
449 applyLyotStop( pupilPlane );
450
451 return 0;
452}
453
454template <typename _realT, typename _fpmaskFloatT>
456{
457 propagate( pupilPlane );
458
459 m_fp.propagatePupilToFocal( m_focalPlane, pupilPlane );
460
461 int x0 = 0.5 * ( m_wfSz - 1 ) - 0.5 * ( fpIntensity.rows() - 1 );
462 int y0 = 0.5 * ( m_wfSz - 1 ) - 0.5 * ( fpIntensity.cols() - 1 );
463
464 extractIntensityImage( fpIntensity, 0, fpIntensity.rows(), 0, fpIntensity.cols(), m_focalPlane, x0, y0 );
465
466 return 0;
467}
468template <typename _realT, typename _fpmaskFloatT>
470{
471 applyApodizer( pupilPlane );
472
473 m_fp.propagatePupilToFocal( m_focalPlane, pupilPlane );
474
475 m_fp.propagateFocalToPupil( pupilPlane, m_focalPlane );
476
477 applyLyotStop( pupilPlane );
478
479 return 0;
480}
481
482template <typename _realT, typename _fpmaskFloatT>
484{
485 propagateNC( pupilPlane );
486
487 m_fp.propagatePupilToFocal( m_focalPlane, pupilPlane );
488
489 int x0 = 0.5 * ( m_wfSz - 1 ) - 0.5 * ( fpIntensity.rows() - 1 );
490 int y0 = 0.5 * ( m_wfSz - 1 ) - 0.5 * ( fpIntensity.cols() - 1 );
491
492 extractIntensityImage( fpIntensity, 0, fpIntensity.rows(), 0, fpIntensity.cols(), m_focalPlane, x0, y0 );
493
494 return 0;
495}
496
497template <typename _realT, typename _fpmaskFloatT>
499 imageT &geomPupil, realT fpmRadPix, realT relTol, realT absTol, int maxIter, const std::string &cname )
500{
501 complexFieldT focalPlane, pupilPlane;
502
503 pupilPlane.resize( m_wfSz, m_wfSz );
504 focalPlane.resize( m_wfSz, m_wfSz );
505
506 mx::wfp::imagingArray<realT, fftwAllocator<realT>, 0> mask( m_wfSz, m_wfSz );
507 mx::wfp::circularPupil( mask, 0., fpmRadPix );
508 /*for(int c=0; c< mask.cols()*0.5; ++c)
509 {
510 for(int r=0; r < mask.rows(); ++r) mask(r,c) = 0;
511 }*/
512
513 // Initialize pupilImage
514 mx::wfp::imagingArray<realT, fftwAllocator<realT>, 0> pupilImage( m_wfSz, m_wfSz );
515 pupilImage.setZero();
516
517 int gpLLi = 0.5 * ( m_wfSz - 1 ) - 0.5 * ( geomPupil.rows() - 1 );
518 int gpLLj = 0.5 * ( m_wfSz - 1 ) - 0.5 * ( geomPupil.cols() - 1 );
519
520 int gpURi = gpLLi + geomPupil.rows();
521 int gpURj = gpLLj + geomPupil.cols();
522
523 for( int i = 0; i < geomPupil.rows(); ++i )
524 {
525 for( int j = 0; j < geomPupil.cols(); ++j )
526 {
527 pupilImage( gpLLi + i, gpLLj + j ) = geomPupil( i, j );
528 }
529 }
530
531 realT lastLambdaA, LambdaA;
532
533 lastLambdaA = 1;
534 int n;
535 std::string reason;
536 for( n = 0; n < maxIter; ++n )
537 {
538 mx::wfp::makeComplexPupil( pupilPlane, pupilImage, m_wfSz );
539
540 m_fp.propagatePupilToFocal( focalPlane, pupilPlane );
541
542 for( int i = 0; i < m_wfSz; ++i )
543 {
544 for( int j = 0; j < m_wfSz; ++j )
545 {
546 focalPlane( i, j ) *= mask( i, j );
547 }
548 }
549
550 m_fp.propagateFocalToPupil( pupilPlane, focalPlane );
551
552 for( int i = 0; i < m_wfSz; ++i )
553 for( int j = 0; j < m_wfSz; ++j )
554 pupilImage( i, j ) = abs( pupilPlane( i, j ) );
555
556 LambdaA = 0;
557 for( int i = 0; i < m_wfSz; ++i )
558 {
559 for( int j = 0; j < m_wfSz; ++j )
560 {
561 if( i >= gpLLi && i < gpURi && j >= gpLLj && j < gpURj )
562 {
563 pupilImage( i, j ) *= geomPupil( i - gpLLi, j - gpLLj );
564 if( pupilImage( i, j ) > LambdaA )
565 LambdaA = pupilImage( i, j );
566 }
567 else
568 {
569 pupilImage( i, j ) = 0;
570 }
571 }
572 }
573 // LambdaA = 1.0/LambdaA;
574 for( int i = 0; i < m_wfSz; ++i )
575 for( int j = 0; j < m_wfSz; ++j )
576 pupilImage( i, j ) /= LambdaA;
577
578 std::cout << n << " " << LambdaA << "\n";
579 if( fabs( LambdaA - lastLambdaA ) < absTol )
580 {
581 std::cout << "Converged on absTol.\n";
582 reason = "absTol";
583 break;
584 }
585 if( fabs( ( LambdaA - lastLambdaA ) / lastLambdaA ) < relTol )
586 {
587 std::cout << "Converged on relTol.\n";
588 reason = "relTol";
589 break;
590 }
591
592 if( n == maxIter - 1 )
593 {
594 std::cout << "maxIter reached.\n";
595 reason = "maxIter";
596 }
597
598 lastLambdaA = LambdaA;
599 }
600 if( LambdaA > 1 )
601 LambdaA = 1;
602
603 std::cout << "LambdaA: = " << LambdaA << "\n";
604
605 realT trans = 1.0 - 1.0 / LambdaA;
606
607 int pupSize = geomPupil.rows();
608
609 m_pupilApodizer.resize( pupSize, pupSize );
610
611 extractBlock( m_pupilApodizer,
612 0,
613 pupSize,
614 0,
615 pupSize,
616 pupilImage,
617 0.5 * ( ( pupilImage.rows() - 1 ) - ( pupSize - 1 ) ),
618 0.5 * ( ( pupilImage.rows() - 1 ) - ( pupSize - 1 ) ) );
619
620 makeFocalMask( fpmRadPix, trans, pupSize );
621
622 /*for(int c=0; c< m_focalMask.cols()*0.5; ++c)
623 {
624 for(int r=0; r < m_focalMask.rows(); ++r)
625 {
626 if(m_focalMask(r,c).real()==1) m_focalMask(r,c) = 0;
627 }
628 }*/
629
630 m_maskSource = 2;
631
632 m_lyotStop = geomPupil;
633
634 fits::fitsHeader head;
635
636 head.append( "", fits::fitsCommentType(), "----------------------------------------" );
637 head.append( "", fits::fitsCommentType(), "lyotCoronagraph optimization Parameters:" );
638 head.append( "", fits::fitsCommentType(), "----------------------------------------" );
639 head.append<int>( "WFSZ", m_wfSz, "Size of wavefront used for FFTs (pixels)" );
640 head.append<realT>( "FPMRADPX", fpmRadPix, "input radius of focal plane mask (pixels)" );
641 head.append<realT>( "ABSTOL", absTol, "input absolute tolerance" );
642 head.append<realT>( "RELTOL", relTol, "input relative tolerance" );
643 head.append<int>( "MAXITER", maxIter, "input maximum iterations" );
644 head.append<int>( "NITER", n, "actual number of iterations" );
645 head.append<std::string>( "XREASON", reason, "reason for convergence" );
646 head.append<realT>( "FPMTRANS", trans, "transmission of FPM" );
647
649
650 std::string fname = "coron/" + cname + "_apod.fits";
651
652 ff.write( fname, m_pupilApodizer, head );
653
654 fname = "coron/" + cname + "_fpm.fits";
655 improc::eigenImage<fpmaskFloatT> fpm( m_focalMask.rows(), m_focalMask.cols() );
656 for( int r = 0; r < m_focalMask.rows(); ++r )
657 {
658 for( int c = 0; c < m_focalMask.cols(); ++c )
659 {
660 fpm( r, c ) = m_focalMask( r, c ).real();
661 }
662 }
663
664 ff.write( fname, fpm, head );
665
666 fname = "coron/" + cname + "_lyot.fits";
667 ff.write( fname, m_lyotStop, head );
668}
669
670template <typename _realT, typename _fpmaskFloatT>
672 realT fpmRadPix,
673 imageT &fpmPhase,
674 realT relTol,
675 realT absTol,
676 int maxIter,
677 const std::string &cname )
678{
679 complexFieldT focalPlane, pupilPlane;
680
681 pupilPlane.resize( m_wfSz, m_wfSz );
682 focalPlane.resize( m_wfSz, m_wfSz );
683
684 mx::wfp::imagingArray<realT, fftwAllocator<realT>, 0> mask( m_wfSz, m_wfSz );
685 mx::wfp::circularPupil( mask, 0., fpmRadPix );
686
687 // Initialize pupilImage
688 mx::wfp::imagingArray<realT, fftwAllocator<realT>, 0> pupilImage( m_wfSz, m_wfSz );
689 pupilImage.setZero();
690
691 int gpLLi = 0.5 * ( m_wfSz - 1 ) - 0.5 * ( geomPupil.rows() - 1 );
692 int gpLLj = 0.5 * ( m_wfSz - 1 ) - 0.5 * ( geomPupil.cols() - 1 );
693
694 int gpURi = gpLLi + geomPupil.rows();
695 int gpURj = gpLLj + geomPupil.cols();
696
697 for( int i = 0; i < geomPupil.rows(); ++i )
698 {
699 for( int j = 0; j < geomPupil.cols(); ++j )
700 {
701 pupilImage( gpLLi + i, gpLLj + j ) = geomPupil( i, j );
702 }
703 }
704
705 realT lastLambdaA, LambdaA;
706
707 lastLambdaA = 1;
708 int n;
709 std::string reason;
710 for( n = 0; n < maxIter; ++n )
711 {
712 mx::wfp::makeComplexPupil( pupilPlane, pupilImage, m_wfSz );
713
714 m_fp.propagatePupilToFocal( focalPlane, pupilPlane );
715
716 for( int i = 0; i < m_wfSz; ++i )
717 {
718 for( int j = 0; j < m_wfSz; ++j )
719 {
720 focalPlane( i, j ) *= mask( i, j ) * exp( std::complex<realT>( 0, fpmPhase( i, j ) ) );
721 ;
722 }
723 }
724
725 m_fp.propagateFocalToPupil( pupilPlane, focalPlane );
726
727 for( int i = 0; i < m_wfSz; ++i )
728 for( int j = 0; j < m_wfSz; ++j )
729 pupilImage( i, j ) = abs( pupilPlane( i, j ) );
730
731 LambdaA = 0;
732 for( int i = 0; i < m_wfSz; ++i )
733 {
734 for( int j = 0; j < m_wfSz; ++j )
735 {
736 if( i >= gpLLi && i < gpURi && j >= gpLLj && j < gpURj )
737 {
738 pupilImage( i, j ) *= geomPupil( i - gpLLi, j - gpLLj );
739 if( pupilImage( i, j ) > LambdaA )
740 LambdaA = pupilImage( i, j );
741 }
742 else
743 {
744 pupilImage( i, j ) = 0;
745 }
746 }
747 }
748 // LambdaA = 1.0/LambdaA;
749 for( int i = 0; i < m_wfSz; ++i )
750 for( int j = 0; j < m_wfSz; ++j )
751 pupilImage( i, j ) /= LambdaA;
752
753 std::cout << n << " " << LambdaA << "\n";
754 if( fabs( LambdaA - lastLambdaA ) < absTol )
755 {
756 std::cout << "Converged on absTol.\n";
757 reason = "absTol";
758 break;
759 }
760 if( fabs( ( LambdaA - lastLambdaA ) / lastLambdaA ) < relTol )
761 {
762 std::cout << "Converged on relTol.\n";
763 reason = "relTol";
764 break;
765 }
766
767 if( n == maxIter - 1 )
768 {
769 std::cout << "maxIter reached.\n";
770 reason = "maxIter";
771 }
772
773 lastLambdaA = LambdaA;
774 }
775 if( LambdaA > 1 )
776 LambdaA = 1;
777
778 std::cout << "LambdaA: = " << LambdaA << "\n";
779
780 realT trans = 1.0 - 1.0 / LambdaA;
781
782 int pupSize = geomPupil.rows();
783
784 m_pupilApodizer.resize( pupSize, pupSize );
785
786 extractBlock( m_pupilApodizer,
787 0,
788 pupSize,
789 0,
790 pupSize,
791 pupilImage,
792 0.5 * ( ( pupilImage.rows() - 1 ) - ( pupSize - 1 ) ),
793 0.5 * ( ( pupilImage.rows() - 1 ) - ( pupSize - 1 ) ) );
794
795 makeFocalMask( fpmRadPix, trans, pupSize );
796 m_maskSource = 2;
797
798 m_lyotStop = geomPupil;
799
800 fits::fitsHeader head;
801
802 head.append( "", fits::fitsCommentType(), "----------------------------------------" );
803 head.append( "", fits::fitsCommentType(), "lyotCoronagraph optimization Parameters:" );
804 head.append( "", fits::fitsCommentType(), "----------------------------------------" );
805 head.append<int>( "WFSZ", m_wfSz, "Size of wavefront used for FFTs (pixels)" );
806 head.append<realT>( "FPMRADPX", fpmRadPix, "input radius of focal plane mask (pixels)" );
807 head.append<realT>( "ABSTOL", absTol, "input absolute tolerance" );
808 head.append<realT>( "RELTOL", relTol, "input relative tolerance" );
809 head.append<int>( "MAXITER", maxIter, "input maximum iterations" );
810 head.append<int>( "NITER", n, "actual number of iterations" );
811 head.append<std::string>( "XREASON", reason, "reason for convergence" );
812 head.append<realT>( "FPMTRANS", trans, "transmission of FPM" );
813
815
816 std::string fname = "coron/" + cname + "_apod.fits";
817
818 ff.write( fname, m_pupilApodizer, head );
819
820 fname = "coron/" + cname + "_fpm.fits";
821 improc::eigenImage<fpmaskFloatT> fpm( m_focalMask.rows(), m_focalMask.cols() );
822 for( int r = 0; r < m_focalMask.rows(); ++r )
823 {
824 for( int c = 0; c < m_focalMask.cols(); ++c )
825 {
826 fpm( r, c ) = m_focalMask( r, c ).real();
827 }
828 }
829
830 ff.write( fname, fpm, head );
831
832 fname = "coron/" + cname + "_lyot.fits";
833 ff.write( fname, m_lyotStop, head );
834}
835
836} // namespace wfp
837} // namespace mx
838
839#endif //__lyotCoronagraph_hpp__
Class to manage interactions with a FITS file.
Definition fitsFile.hpp:52
int write(const dataT *im, int d1, int d2, int d3, fitsHeader *head)
Write the contents of a raw array to the FITS file.
int read(dataT *data)
Read the contents of the FITS file into an array.
Definition fitsFile.hpp:829
Class to manage a complete fits header.
void append(const fitsHeaderCard &card)
Append a fitsHeaderCard to the end of the header.
An image cube with an Eigen-like API.
Definition eigenCube.hpp:30
Eigen::Map< Eigen::Array< dataT, Eigen::Dynamic, Eigen::Dynamic > > image(Index n)
Returns a 2D Eigen::Eigen::Map pointed at the specified image.
Class to perform Fraunhofer propagation between pupil and focal planes.
Declares and defines a class for Fraunhofer propagation of optical wavefronts.
Eigen::Array< scalarT, -1, -1 > eigenImage
Definition of the eigenImage type, which is an alias for Eigen::Array.
void makeComplexPupil(arrayOutT &complexPupil, const arrayInT &realPupil, int wavefrontSizePixels)
Create a complex pupil plane wavefront from a real amplitude mask.
int circularPupil(arrayT &m, typename arrayT::Scalar eps=0, typename arrayT::Scalar rad=0, typename arrayT::Scalar overscan=0)
Fill in an Eigen-like array with a circular pupil mask.
Declares and defines a class for managing images.
Utilities for modeling image formation.
void extractBlock(imageT1 &dest, int imX0, int imXsz, int imY0, int imYsz, imageT2 &src, int wfX0, int wfY0)
Extract a block from one image and insert it into a second.
The mxlib c++ namespace.
Definition mxError.hpp:106
The Lyot Coronagraph.
std::string m_maskFile
Name of file from which mask was loaded.
int loadCoronagraph(const std::string &apodName, const std::string &fpmName, const std::string &lyotName)
Load the components of the coronagraph from FITS files.
int m_maskSource
0= read from file, 1 = constructed by makeFocalMask, 2 = trans optimized.
Eigen::Array< realT, Eigen::Dynamic, Eigen::Dynamic > imageT
The image type.
int m_wfSz
The linear size of the wavefront in pixels.
int loadApodizer(const std::string &apodName)
Load the apodizer from a FITS file.
int propagate(complexFieldT &pupilPlane)
Propagate the given pupil-plane wavefront through the coronagraph.
mx::wfp::imagingArray< std::complex< realT >, fftwAllocator< std::complex< realT > >, 0 > complexFieldT
The wavefront complex field type.
imageT m_lyotStop
Image containing the lyot stop.
void makeFocalMask(realT rad, fpmaskFloatT trans=0.0, int sz=0.0)
Make the focal plane mask.
int propagateNC(complexFieldT &pupilPlane)
Propagate the given pupil-plane wavefront without the coronagraph.
fpMaskT m_focalMask
The focal plane mask.
fraunhoferPropagator< complexFieldT > m_fp
Fraunhofer propagator.
std::complex< realT > complexT
The complex floating point type.
void optimizeAPLCMC(imageT &geomPupil, realT fpmRadPix, realT relTol, realT absTol, int maxIter, const std::string &cname)
Optimize the pupil amplitude apodization and focal-plane mask complex transmission.
_realT realT
The real floating point type.
realT m_maskTrans
Transmission of mask if it was constructed.
imageT m_pupilApodizer
Image containing the pupil apodization.
int loadFocalMask(const std::string &fpmName)
Load the focal plane mask from a FITS file.
Eigen::Array< std::complex< fpmaskFloatT >, Eigen::Dynamic, Eigen::Dynamic > fpMaskT
The focal plane mask type.
int loadLyotStop(const std::string &lyotName)
Load the Lyot stop from a FITS file.
realT m_maskRad
Radius of mask if it was constructed.
_fpmaskFloatT fpmaskFloatT
The real floating point type for mask calculations.
int wfSz()
Get the wavefront size in pixels.
std::string m_fileDir
The directory where coronagraph files are stored.