1#ifndef directPhaseReconstructor_hpp
2#define directPhaseReconstructor_hpp
4#include "../../improc/eigenImage.hpp"
5#include "../../improc/eigenCube.hpp"
6#include "../../ioutils/fits/fitsFile.hpp"
7using namespace mx::improc;
8using namespace mx::fits;
10#include "../../math/cuda/cudaPtr.hpp"
11#include "../../math/cuda/templateCublas.hpp"
13#include "../../sigproc/signalWindows.hpp"
16#define BREAD_CRUMB std::cout << "DEBUG: " << __FILE__ << " " << __LINE__ << "\n";
28struct directPhaseReconstructorSpec
31 std::string basisName;
39template <
typename realT>
44 typedef Eigen::Array<realT, -1, -1>
imageT;
47 typedef Eigen::Array<realT, -1, -1>
rmatT;
50 typedef directPhaseReconstructorSpec
specT;
56 imageT *m_pupil{
nullptr };
76 cublasHandle_t *m_cublasHandle;
90 std::vector<size_t> *m_idx{
nullptr };
103 template <
typename AOSysT>
104 void initialize( AOSysT &AOSys,
specT &spec );
134 void loadRecon(
const std::string &fname );
144 template <
typename measurementT,
typename wfsImageT>
148 template <
typename measurementT,
typename wfsImageT>
149 void reconstruct( measurementT &commandVect, wfsImageT &wfsImage );
158 void initializeRMat(
int nmodes, realT calamp,
int detrows,
int detcols );
165 template <
typename measurementT>
168 template <
typename measurementT,
typename wfsImageT>
169 void accumulateRMat(
int i, measurementT &measureVec, wfsImageT &wfsImage );
177 void saveRImages( std::string fname );
180template <
typename realT>
185template <
typename realT>
186template <
typename AOSysT>
189 static_cast<void>( spec );
191 m_pupil = &AOSys._pupil;
193 m_nPix = m_pupil->sum();
195 m_modes = &AOSys.dm.m_infF;
197 m_nModes = m_modes->planes();
198 m_detRows = m_modes->rows();
199 m_detCols = m_modes->cols();
201 m_idx = &AOSys.dm.m_idx;
203 m_measurementSize = m_idx->size();
207 m_cublasHandle = &AOSys.m_cublasHandle;
214 m_one.upload( &one, 1 );
220 recon.resize( m_nModes, m_measurementSize );
221 for(
int pp = 0; pp < m_nModes; ++pp )
223 for(
int nn = 0; nn < m_idx->size(); ++nn )
225 recon( pp, nn ) = *( m_modes->image( pp ).data() + ( *m_idx )[nn] ) / m_nPix;
229 m_devRecon.upload( recon.data(), recon.rows() * recon.cols() );
231 m_devSlopes.resize( m_measurementSize );
233 m_devAmps.resize( m_nModes );
237 m_recon.resize( m_measurementSize, m_nModes );
239 for(
int pp = 0; pp < m_nModes; ++pp )
241 for(
int nn = 0; nn < m_idx->size(); ++nn )
243 m_recon( nn, pp ) = *( m_modes->image( pp ).data() + ( *m_idx )[nn] ) / m_nPix;
250template <
typename realT>
256template <
typename realT>
262template <
typename realT>
268template <
typename realT>
274template <
typename realT>
280template <
typename realT>
287 ff.
read(m_recon, head, fname);
291template <
typename realT>
294 return m_measurementSize;
297template <
typename realT>
298template <
typename measurementT,
typename wfsImageT>
301 slopes.measurement.resize( m_measurementSize );
303 realT *imp = wfsImage.image.data();
304 for(
int nn = 0; nn < m_idx->size(); ++nn )
306 slopes.measurement[nn] = *( imp + ( *m_idx )[nn] );
310template <
typename realT>
311template <
typename measurementT,
typename wfsImageT>
317 static measurementT slopes;
319 calcMeasurement( slopes, wfsImage );
321 m_devSlopes.upload( slopes.measurement.data(), slopes.measurement.size() );
326 cublasStatus_t stat = cuda::cublasTgemv<realT>( *m_cublasHandle,
335 if( stat != CUBLAS_STATUS_SUCCESS )
337 std::cerr <<
"cublas error\n";
340 commandVect.measurement.resize( m_nModes );
341 m_devAmps.download( commandVect.measurement.data() );
343 commandVect.iterNo = wfsImage.iterNo;
354#pragma omp parallel for
355 for(
int j = 0; j < m_nModes; ++j )
362 for(
size_t k = 0; k < m_idx->size(); ++k )
364 amp += *( wfsImage.image.data() + ( *m_idx )[k] ) * m_recon( k, j );
366 commandVect.measurement[j] = amp;
378 commandVect.iterNo = wfsImage.iterNo;
383template <
typename realT>
391 m_rMat.resize( measurementSize(), nModes );
394 m_rImages.resize( m_detRows, m_detCols, m_nModes );
397template <
typename realT>
398template <
typename measurementT>
402 for(
int j = 0; j < measureVec.measurement.rows(); ++j )
404 for(
int k = 0; k < measureVec.measurement.cols(); ++k )
406 m_rMat( l, i ) = measureVec.measurement( j, k );
414template <
typename realT>
415template <
typename measurementT,
typename wfsImageT>
418 accumulateRMat( i, measureVec );
419 m_rImages.image( i ) = wfsImage.image;
422template <
typename realT>
428 head.
append(
"DETROWS", m_detRows,
"WFS detector rows" );
429 head.
append(
"DETCOLS", m_detCols,
"WFS detector cols" );
430 head.
append(
"CALAMP", m_calAmp,
"DM Calibration amplitude" );
431 head.
append(
"NMODES", m_nModes,
"Number of modes included in the response matrix." );
433 ff.
write( fname, m_rMat, head );
436template <
typename realT>
442 head.
append(
"DETROWS", m_detRows,
"WFS detector rows" );
443 head.
append(
"DETCOLS", m_detCols,
"WFS detector cols" );
444 head.
append(
"CALAMP", m_calAmp,
"DM Calibration amplitude" );
445 head.
append(
"NMODES", m_nModes,
"Number of modes included in the response matrix." );
448 ff.
write( fname, m_rImages, head );
Direct Phase Reconstructor.
int m_detRows
The size of the WFS image, in rows.
void saveRMat(std::string fname)
Write the accumulated response matrix to disk.
int m_nModes
The number of modes to be reconstructed.
improc::eigenCube< realT > * m_modes
The mirror modes, managed by the DM.
int detRows()
Get the number of detector rows (m_detRows)
void loadRecon(const std::string &fname)
Load the reconstrutor from the specified FITS file.
void detCols(int dc)
Set the number of detector columns (m_detCols)
Eigen::Array< realT, -1, -1 > imageT
The type of the WFS image.
int measurementSize()
Return the size of the unbinned measurement.
void calcMeasurement(measurementT &slopes, wfsImageT &wfsImage)
Calculate the slope measurement.
Eigen::Array< realT, -1, -1 > m_recon
The reconstructor matrix.
void initializeRMat(int nmodes, realT calamp, int detrows, int detcols)
Initialize the response matrix for acquisition.
realT calAmp()
Get the calibration amplitude used in response matrix acquisition (m_calAmp)
int m_detCols
The size of the WFS image, in columns.
realT m_calAmp
The offset coordinates of non-zero pixels in the pupil. Set by the DM.
imageT m_rMat
The response matrix.
directPhaseReconstructorSpec specT
The specificaion type.
void accumulateRMat(int i, measurementT &measureVec)
Accumalte the next measurement in the response matrix.
int m_measurementSize
The number of values in the measurement.
void reconstruct(measurementT &commandVect, wfsImageT &wfsImage)
Reconstruct the wavefront from the input image, producing the modal amplitude vector.
directPhaseReconstructor()
Default c'tor.
Eigen::Array< realT, -1, -1 > rmatT
The type of the response matrix.
int detCols()
Get the number of detector columns (m_detCols)
int nModes()
Get the number of modes (m_nModes)
void detRows(int dr)
Set the number of detector rows (m_detRows)
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.
constexpr floatT six_fifths()
Return 6/5 in the specified precision.
A smart-pointer wrapper for cuda device pointers.