8#ifndef simulatedAOSystem_hpp
9#define simulatedAOSystem_hpp
14#include <cuda_runtime.h>
17#include "../../improc/imagePads.hpp"
18#include "../../wfp/imagingUtils.hpp"
19#include "../../wfp/fraunhoferPropagator.hpp"
21#include "../../ioutils/fits/fitsFile.hpp"
22#include "../../ioutils/fits/fitsUtils.hpp"
24#include "../../improc/eigenImage.hpp"
25#include "../../improc/eigenCube.hpp"
27#include "../../sys/timeUtils.hpp"
28#include "../../sigproc/signalWindows.hpp"
30#include "wavefront.hpp"
31#include "../aoPaths.hpp"
34#define BREAD_CRUMB std::cout << "DEBUG: " << __FILE__ << " " << __LINE__ << "\n";
59template <
typename _realT,
73 typedef Eigen::Array<realT, Eigen::Dynamic, Eigen::Dynamic>
imageT;
75 typedef wfp::imagingArray<std::complex<realT>, wfp::fftwAllocator<std::complex<realT>>, 0>
79 typedef _reconT reconT;
80 typedef _filterT filterT;
82 typedef _turbSeqT turbSeqT;
83 typedef _coronT coronT;
86 cublasHandle_t m_cublasHandle;
92 bool m_sfImagePlane{
false };
100 void wfPS( realT wps )
110 int _loopClosedDelay;
113 std::string _rmsFile;
114 std::ofstream _rmsOut;
117 std::string _ampFile;
138 std::string m_wfFileBase{
"simAOWF" };
142 int m_nWFPerFile{ 500 };
144 int m_currWFFile{ 0 };
153 bool _writeIndFrames;
155 long _saveFrameStart;
157 std::string _psfFileBase;
177 coronT m_coronagraph;
181 std::vector<typename filterT::commandT> _delayedCommands;
183 std::vector<int> _goodCommands;
203 typename dmT::specT &dmSpec,
204 const std::string &wfsName,
205 const std::string &pupilName,
209 void initSim(
typename reconT::specT &reconSpec, realT
simStep,
int commandDelay,
const std::string &coronName );
236 void runTurbulence();
248 int simStep(
const double &ss );
256 int D(
const realT &nD );
260 int wfPS(
const realT &nwfPS );
266 int wfSz(
const int &nwfSz );
270 bool m_doCoron{
false };
275 double dt_turbulence{ 0 };
277 double dt_recon{ 0 };
278 double dt_dmcomb{ 0 };
279 double dt_total{ 0 };
282template <
typename realT,
293 _loopClosedDelay = 0;
298 _writeIndFrames =
false;
304 m_coronagraph.m_fileDir =
sys::getEnv(
"MX_AO_DATADIR" ) +
"/" +
"coron/";
308 cublasCreate( &m_cublasHandle );
309 cublasSetPointerMode( m_cublasHandle, CUBLAS_POINTER_MODE_DEVICE );
312template <
typename realT,
321 if( m_ampOut.rows() > 0 && _ampFile !=
"" )
324 ff.
write( _ampFile +
".fits", m_ampOut );
327 if( _rmsOut.is_open() )
331 if( _psfFileBase !=
"" && _currImage > 0 && _writeIndFrames )
339 ff.
write( fn, _psfs );
347 ff.
write( fn, _corons );
353template <
typename realT,
361 typename dmT::specT &dmSpec,
362 const std::string &wfsName,
363 const std::string &pupilName,
368 _pupilName = pupilName;
375 std::string pupilFile = mx::AO::path::pupil::pupilFile( _pupilName );
377 ff.
read( _pupil, head, pupilFile );
379 m_D = head[
"PUPILD"].Value<
realT>();
380 m_wfPS = head[
"SCALE"].Value<
realT>();
387 turbSeq._pupil = &_pupil;
388 turbSeq.wfPS( m_wfPS );
391 dm.initialize( *
this, dmSpec, _pupilName );
393 wfs.linkSystem( *
this );
400template <
typename realT,
410 const std::string &coronName )
417 wfs.simStep( m_simStep );
419 filter.initialize( dm.nModes() );
421 recon.initialize( *
this, reconSpec );
422 dm.calAmp( recon.calAmp() );
426 _commandDelay = commandDelay;
428 _delayedCommands.resize( ( _commandDelay + 1 ) * 5 );
429 _goodCommands.resize( ( _commandDelay + 1 ) * 5 );
431 if( coronName !=
"" )
433 m_coronagraph.wfSz( m_wfSz );
434 m_coronagraph.loadCoronagraph( coronName );
487template <
typename realT,
499 cpup.resize( m_wfSz, m_wfSz );
501 recon.initializeRMat( dm.nModes(), amp, wfs.detRows(), wfs.detCols() );
503 typename filterT::commandT measureVec;
506 currWF.setAmplitude( _pupil );
510 wfs.detector.noNoise(
true );
512 double tO, tF, t0, t1, t_applyMode = 0, t_senseWF = 0, t_calcMeas = 0, t_accum = 0;
514 std::cerr << dm.nModes() <<
"\n";
518 for(
int i = 0; i < dm.nModes(); ++i )
522 currWF.setPhase( _pupil * 0 );
525 if( nmodes > 0 && i >= nmodes )
528 wfs.detectorImage.image.setZero();
533 dm.applyMode( currWF, i, s_amp, 0.8e-6 );
535 t_applyMode += t1 - t0;
540 wfs.senseWavefrontCal( currWF );
543 t_senseWF += t1 - t0;
550 recon.calcMeasurement( measureVec, wfs.detectorImage );
554 t_calcMeas += t1 - t0;
559 recon.accumulateRMat( i, measureVec, wfs.detectorImage );
566 std::cout << ( (
realT)dm.nModes() ) / ( tF - tO ) <<
" Hz\n";
568 std::cout << t_applyMode / dm.nModes() <<
" " << t_senseWF / dm.nModes() <<
" " << t_calcMeas / dm.nModes() <<
" "
569 << t_accum / dm.nModes();
570 std::cout <<
" " << dm.t_mm / dm.nModes() <<
" " << dm.t_sum / dm.nModes() <<
" " <<
"\n";
573 fname = mx::AO::path::sys::cal::rMat( _sysName, dm.name(), _wfsName, _pupilName, dm.basisName(), rmatID,
true );
575 std::cout << fname <<
"\n";
576 recon.saveRMat( fname );
578 fname = mx::AO::path::sys::cal::rImages( _sysName, dm.name(), _wfsName, _pupilName, dm.basisName(), rmatID,
true );
579 recon.saveRImages( fname );
582template <
typename realT,
591 return turbSeq.frames();
619template <
typename realT,
626void simulatedAOSystem<realT, wfsT, reconT, filterT, dmT, turbSeqT, coronT>::nextWF( wavefrontT &wf )
629 realT rms_ol, rms_cl;
635 _npix = _pupil.sum();
641 turbSeq.nextWF( wf );
644 wf.iterNo = _frameCounter;
649 realT mn = ( wf.phase * _pupil ).sum() / _npix;
652 wf.phase = ( wf.phase - mn ) * _pupil;
654 if( _pupilMask.rows() == _pupil.rows() && _pupilMask.cols() == _pupil.cols() )
656 wf.phase *= _pupilMask;
661 if( _frameCounter % 10 == 0 )
662 rms_ol = sqrt( wf.phase.square().sum() / _npix );
666 typename filterT::commandT measuredAmps, commandAmps;
668 filter.initMeasurements( measuredAmps, commandAmps );
670 if( _frameCounter == 0 )
672 for(
size_t i = 0; i < _delayedCommands.size(); ++i )
674 _goodCommands[i] = 0;
681 dm.applyShape( wf, _wfsLambda );
688 newCV = wfs.senseWavefront( wf );
696 recon.reconstruct( measuredAmps, wfs.m_detectorImage );
705 if( m_ampOut.cols() != turbSeq.frames() || m_ampOut.rows() != measuredAmps.measurement.size() + 1 )
707 if( m_ampOut.cols() != 0 )
709 std::cerr <<
"m_ampOut already allocated but cols not frames\n";
713 if( m_ampOut.rows() != 0 )
715 std::cerr <<
"m_ampOut already allocated but rows not measurements\n";
719 m_ampOut.resize( measuredAmps.measurement.size() + 1, turbSeq.frames() );
723 int n = floor( measuredAmps.iterNo );
724 m_ampOut( 0, n ) = measuredAmps.iterNo;
725 for(
int i = 1; i < measuredAmps.measurement.size() + 1; ++i )
727 m_ampOut( i, n ) = measuredAmps.measurement[i - 1];
733 int nAmps = ( _frameCounter % _delayedCommands.size() );
735 _delayedCommands[nAmps].measurement = measuredAmps.measurement;
736 _delayedCommands[nAmps].iterNo = measuredAmps.iterNo;
738 _goodCommands[nAmps] = 1;
742 int nAmps = ( _frameCounter % _delayedCommands.size() );
743 _goodCommands[nAmps] = 0;
746 int _currCommand = ( _frameCounter % _delayedCommands.size() ) - _commandDelay;
747 if( _currCommand < 0 )
748 _currCommand += _delayedCommands.size();
750 if( _goodCommands[_currCommand] )
754 filter.filterCommands( commandAmps, _delayedCommands[_currCommand], _frameCounter );
759 dm.setShape( commandAmps );
765 int nAmps = ( _frameCounter % _delayedCommands.size() );
766 _goodCommands[nAmps] = 0;
770 if( _postMask.rows() != _pupil.rows() )
776 if( _frameCounter % 10 == 0 )
778 mn = ( wf.phase * _pupil ).sum() / _npix;
779 rms_cl = sqrt( ( wf.phase - mn ).square().sum() / _postMask.sum() );
780 std::cout << _frameCounter <<
" WFE: " << rms_ol <<
" " << rms_cl <<
" [rad rms phase]\n";
785 wfs.filter().filter( wf.phase );
792 if( !_rmsOut.is_open() )
794 _rmsOut.open( _rmsFile );
795 _rmsOut <<
"#open-loop-wfe closed-loop-wfe [rad rms phase]\n";
797 _rmsOut << rms_ol <<
" " << rms_cl << std::endl;
802 if( m_writeWavefronts )
804 if( m_wfPhase.rows() != wf.phase.rows() || m_wfPhase.cols() != wf.phase.cols() ||
805 m_wfPhase.planes() != m_nWFPerFile || m_wfAmp.rows() != wf.amplitude.rows() ||
806 m_wfAmp.cols() != wf.amplitude.cols() || m_wfAmp.planes() != m_nWFPerFile )
808 m_wfPhase.resize( wf.phase.rows(), wf.phase.cols(), m_nWFPerFile );
809 m_wfAmp.resize( wf.amplitude.rows(), wf.amplitude.cols(), m_nWFPerFile );
814 m_wfPhase.image( m_currWF ) = wf.phase;
815 m_wfAmp.image( m_currWF ) = wf.amplitude;
819 if( m_currWF >= m_nWFPerFile )
821 std::cerr <<
"Write to WF file here . . . \n";
825 std::string fn = m_wfFileBase +
"_phase_" + ioutils::convertToString<int, 5, '0'>( m_currWFFile ) +
".fits";
826 ff.
write( fn, m_wfPhase );
828 fn = m_wfFileBase +
"_amp_" + ioutils::convertToString<int, 5, '0'>( m_currWFFile ) +
".fits";
838 if( _psfFileBase !=
"" && _frameCounter > _saveFrameStart )
840 if( _psfOut.rows() == 0 || ( _psfs.planes() == 0 && _writeIndFrames ) )
845 if( _writeIndFrames )
846 _psfs.resize( m_saveSz, m_saveSz, _nPerFile );
847 _psfOut.resize( m_saveSz, m_saveSz );
850 _realFocal.resize( m_saveSz, m_saveSz );
854 if( _writeIndFrames )
855 _corons.resize( m_saveSz, m_saveSz, _nPerFile );
856 _coronOut.resize( m_saveSz, m_saveSz );
859 _realFocalCoron.resize( m_saveSz, m_saveSz );
868 wf.getWavefront( _complexPupilCoron, m_wfSz );
870 m_coronagraph.propagate( _realFocalCoron, _complexPupilCoron );
872 if( _writeIndFrames )
873 _corons.image( _currImage ) = _realFocalCoron;
875 _coronOut += _realFocalCoron;
879 wf.getWavefront( _complexPupil, m_wfSz );
881 m_coronagraph.propagateNC( _realFocal, _complexPupil );
883 if( _writeIndFrames )
884 _psfs.image( _currImage ) = _realFocal;
886 _psfOut += _realFocal;
892 if( _currImage >= _nPerFile && _writeIndFrames )
900 ff.
write( fn, _psfs );
908 ff.
write( fn, _corons );
921template <
typename realT,
928void simulatedAOSystem<realT, wfsT, reconT, filterT, dmT, turbSeqT, coronT>::runTurbulence()
934 if( m_coronagraph.wfSz() != m_wfSz )
936 std::cerr <<
"doCoron is true, but coronagraph not loaded or wavefront sizes don't match.\n";
937 std::cerr <<
"Continuing . . .\n";
941 _wfsLambda = wfs.lambda();
944 for(
size_t i = 0; i < turbSeq.frames(); ++i )
950 turbSeq._loopClosed =
true;
954 if( i == (
size_t)_loopClosedDelay )
966 if( _psfFileBase !=
"" )
968 fits::fitsFile<realT> ff;
969 std::string fn = _psfFileBase +
"_psf.fits";
971 ff.write( fn, _psfOut );
976 fn = _psfFileBase +
"_coron.fits";
977 ff.write( fn, _coronOut );
983 std::cout <<
"Timing: \n";
984 std::cout <<
" Total Time: " << dt_total <<
" sec / " << turbSeq.frames() / dt_total <<
" fps\n";
985 std::cout <<
" Turbulence: " << dt_turbulence <<
" sec / " << turbSeq.frames() / dt_turbulence <<
" fps / "
986 << dt_turbulence / dt_total <<
"%\n";
987 std::cout <<
" WFS: " << dt_wfs <<
" sec / " << turbSeq.frames() / dt_wfs <<
" fps / "
988 << dt_wfs / dt_total <<
"%\n";
989 std::cout <<
" Recon: " << dt_recon <<
" sec / " << turbSeq.frames() / dt_recon <<
" fps / "
990 << dt_recon / dt_total <<
"%\n";
991 std::cout <<
" dmcomb: " << dt_dmcomb <<
" sec / " << turbSeq.frames() / dt_dmcomb <<
" fps / "
992 << dt_dmcomb / dt_total <<
"%\n";
996template <
typename realT,
1010template <
typename realT,
1022template <
typename realT,
1036template <
typename realT,
1043realT simulatedAOSystem<realT, wfsT, reconT, filterT, dmT, turbSeqT, coronT>::D()
1048template <
typename realT,
1055int simulatedAOSystem<realT, wfsT, reconT, filterT, dmT, turbSeqT, coronT>::wfPS(
const realT &nwfPS )
1062template <
typename realT,
1069realT simulatedAOSystem<realT, wfsT, reconT, filterT, dmT, turbSeqT, coronT>::wfPS()
1074template <
typename realT,
1081int simulatedAOSystem<realT, wfsT, reconT, filterT, dmT, turbSeqT, coronT>::Dpix()
1083 return (
int)( m_D / m_wfPS + std::numeric_limits<realT>::epsilon() );
1086template <
typename realT,
1093int simulatedAOSystem<realT, wfsT, reconT, filterT, dmT, turbSeqT, coronT>::wfSz(
const int &nwfSz )
1100template <
typename realT,
1107int simulatedAOSystem<realT, wfsT, reconT, filterT, dmT, turbSeqT, coronT>::wfSz()
int initSystem(const std::string &sysName, typename dmT::specT &dmSpec, const std::string &wfsName, const std::string &pupilName, const int &wfSz)
Initialize the system.
std::string _wfsName
The WFS name for use in the mx::AO::path.
std::string _pupilName
The pupil name for use in the mx::AO::path.
~simulatedAOSystem()
Destructor.
_realT realT
The floating point type used for all calculations.
realT m_simStep
The simulation step size in seconds.
Eigen::Array< realT, Eigen::Dynamic, Eigen::Dynamic > imageT
The real image type.
std::string _sysName
The system name for use in mx::AO::path.
double simStep()
Get the simulation step size.
void takeResponseMatrix(realT amp, std::string rmatName, int nmodes=0)
Measure the system response matrix.
simulatedAOSystem()
Default c'tor.
wfp::imagingArray< std::complex< realT >, wfp::fftwAllocator< std::complex< realT > >, 0 > complexImageT
The complex image type.
mx::AO::sim::wavefront< realT > wavefrontT
The wavefront type.
Class to manage interactions with a FITS file.
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.
An image cube with an Eigen-like API.
Eigen::Array< scalarT, -1, -1 > eigenImage
Definition of the eigenImage type, which is an alias for Eigen::Array.
std::string convertToString(const typeT &value, int precision=0)
Convert a numerical value to a string.
std::string getEnv(const std::string &estr)
Return the value of an environment variable.
typeT get_curr_time()
Get the current system time in seconds.
Structure containing the phase and amplitude of a wavefront.