Loading [MathJax]/extensions/tex2jax.js
mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Modules Pages
directPhaseReconstructor.hpp
1#ifndef directPhaseReconstructor_hpp
2#define directPhaseReconstructor_hpp
3
4#include "../../improc/eigenImage.hpp"
5#include "../../improc/eigenCube.hpp"
6#include "../../ioutils/fits/fitsFile.hpp"
7using namespace mx::improc;
8using namespace mx::fits;
9
10#include "../../math/cuda/cudaPtr.hpp"
11#include "../../math/cuda/templateCublas.hpp"
12
13#include "../../sigproc/signalWindows.hpp"
14
15#ifdef DEBUG
16#define BREAD_CRUMB std::cout << "DEBUG: " << __FILE__ << " " << __LINE__ << "\n";
17#else
18#define BREAD_CRUMB
19#endif
20
21namespace mx
22{
23namespace AO
24{
25namespace sim
26{
27
28struct directPhaseReconstructorSpec
29{
30 std::string dmName;
31 std::string basisName;
32
33 std::string rMatId;
34};
35
36/// Direct Phase Reconstructor
37/** Calculates modal amplitudes by direct projection of modes onto the phase screen.
38 */
39template <typename realT>
41{
42 public:
43 /// The type of the WFS image
44 typedef Eigen::Array<realT, -1, -1> imageT;
45
46 /// The type of the response matrix
47 typedef Eigen::Array<realT, -1, -1> rmatT;
48
49 /// The specificaion type.
50 typedef directPhaseReconstructorSpec specT;
51
52 protected:
53 /** \name The Pupil
54 * @{
55 */
56 imageT *m_pupil{ nullptr };
57 int m_nPix{ 0 };
58 ///@}
59
60 /** \name The Basis
61 * @{
62 */
63 improc::eigenCube<realT> *m_modes{ nullptr }; ///< The mirror modes, managed by the DM
64
65 int m_nModes{ 0 }; ///< The number of modes to be reconstructed
66
67 int m_detRows{ 0 }; ///< The size of the WFS image, in rows
68 int m_detCols{ 0 }; ///< The size of the WFS image, in columns
69 ///@}
70
71 /** \name The Reconstructor
72 * @{
73 */
74
75#ifdef MXAO_USE_GPU
76 cublasHandle_t *m_cublasHandle;
77
80
81 cuda::cudaPtr<realT> m_devRecon;
82 cuda::cudaPtr<realT> m_devSlopes;
83 cuda::cudaPtr<realT> m_devAmps;
84
85#else
86 Eigen::Array<realT, -1, -1> m_recon; ///< The reconstructor matrix.
87#endif
88
89 int m_measurementSize{ 0 }; ///< The number of values in the measurement
90 std::vector<size_t> *m_idx{ nullptr }; /// The offset coordinates of non-zero pixels in the pupil. Set by the DM.
91 ///@}
92
93 realT m_calAmp{ 1e-6 }; ///< The calibration amplitude used for response matrix acquisition
94
95 imageT m_rMat; ///< The response matrix
96
97 eigenCube<realT> m_rImages;
98
99 public:
100 /// Default c'tor
102
103 template <typename AOSysT>
104 void initialize( AOSysT &AOSys, specT &spec );
105
106 /// Get the calibration amplitude used in response matrix acquisition (m_calAmp)
107 realT calAmp();
108
109 /// Set the calibration amplitude used in response matrix acquisition (m_calAmp)
110 /**
111 * \param ca [in] the new calibration amplitude
112 */
113 void calAmp( realT ca );
114
115 /// Get the number of modes (m_nModes)
116 int nModes();
117
118 /// Get the number of detector rows (m_detRows)
119 int detRows();
120
121 /// Set the number of detector rows (m_detRows)
122 void detRows( int dr );
123
124 /// Get the number of detector columns (m_detCols)
125 int detCols();
126
127 /// Set the number of detector columns (m_detCols)
128 void detCols( int dc );
129
130 /// Load the reconstrutor from the specified FITS file
131 /**
132 * \param fname is the name of the FITS file, including path
133 */
134 void loadRecon( const std::string &fname );
135
136 /// Return the size of the unbinned measurement
137 int measurementSize();
138
139 /// Calculate the slope measurement
140 /**
141 * \param slopes [out] a (m_measurementSize X 2) array of slopes
142 * \param wfsImage [in] the WFS image from which to measure the slopes
143 */
144 template <typename measurementT, typename wfsImageT>
145 void calcMeasurement( measurementT &slopes, wfsImageT &wfsImage );
146
147 /// Reconstruct the wavefront from the input image, producing the modal amplitude vector
148 template <typename measurementT, typename wfsImageT>
149 void reconstruct( measurementT &commandVect, wfsImageT &wfsImage );
150
151 /// Initialize the response matrix for acquisition
152 /**
153 * \param nmodes the number of modes
154 * \param calamp the calibration amplitude
155 * \param detrows the number of detector rows
156 * \param detcols the number of detector columns
157 */
158 void initializeRMat( int nmodes, realT calamp, int detrows, int detcols );
159
160 /// Accumalte the next measurement in the response matrix
161 /**
162 * \param i the measurement index
163 * \param measureVec is the i-th measurement vector
164 */
165 template <typename measurementT>
166 void accumulateRMat( int i, measurementT &measureVec );
167
168 template <typename measurementT, typename wfsImageT>
169 void accumulateRMat( int i, measurementT &measureVec, wfsImageT &wfsImage );
170
171 /// Write the accumulated response matrix to disk
172 /**
173 * \param fname the name, including path, of the response matrix
174 */
175 void saveRMat( std::string fname );
176
177 void saveRImages( std::string fname );
178};
179
180template <typename realT>
184
185template <typename realT>
186template <typename AOSysT>
187void directPhaseReconstructor<realT>::initialize( AOSysT &AOSys, specT &spec )
188{
189 static_cast<void>( spec );
190
191 m_pupil = &AOSys._pupil;
192
193 m_nPix = m_pupil->sum();
194
195 m_modes = &AOSys.dm.m_infF;
196
197 m_nModes = m_modes->planes();
198 m_detRows = m_modes->rows();
199 m_detCols = m_modes->cols();
200
201 m_idx = &AOSys.dm.m_idx;
202
203 m_measurementSize = m_idx->size();
204
205#ifdef MXAO_USE_GPU
206
207 m_cublasHandle = &AOSys.m_cublasHandle;
208 // m_cublasHandle = new cublasHandle_t; //&AOSys.m_cublasHandle;
209 // cublasCreate(m_cublasHandle);
210 // cublasSetPointerMode(*m_cublasHandle, CUBLAS_POINTER_MODE_DEVICE);
211
212 m_one.resize( 1 );
213 realT one = 1.0;
214 m_one.upload( &one, 1 );
215
216 m_zero.resize( 1 );
217 m_zero.initialize();
218
219 imageT recon;
220 recon.resize( m_nModes, m_measurementSize );
221 for( int pp = 0; pp < m_nModes; ++pp )
222 {
223 for( int nn = 0; nn < m_idx->size(); ++nn )
224 {
225 recon( pp, nn ) = *( m_modes->image( pp ).data() + ( *m_idx )[nn] ) / m_nPix;
226 }
227 }
228
229 m_devRecon.upload( recon.data(), recon.rows() * recon.cols() );
230
231 m_devSlopes.resize( m_measurementSize );
232
233 m_devAmps.resize( m_nModes );
234
235#else
236
237 m_recon.resize( m_measurementSize, m_nModes );
238
239 for( int pp = 0; pp < m_nModes; ++pp )
240 {
241 for( int nn = 0; nn < m_idx->size(); ++nn )
242 {
243 m_recon( nn, pp ) = *( m_modes->image( pp ).data() + ( *m_idx )[nn] ) / m_nPix;
244 }
245 }
246
247#endif
248}
249
250template <typename realT>
252{
253 return 0.5 * 800.0e-9 / math::two_pi<realT>();
254}
255
256template <typename realT>
258{
259 return;
260}
261
262template <typename realT>
264{
265 return m_nModes;
266}
267
268template <typename realT>
270{
271 return m_detRows;
272}
273
274template <typename realT>
276{
277 return m_detCols;
278}
279
280template <typename realT>
281void directPhaseReconstructor<realT>::loadRecon( const std::string &fname )
282{
283#if 0
285 fitsHeader head;
286
287 ff.read(m_recon, head, fname);
288#endif
289}
290
291template <typename realT>
293{
294 return m_measurementSize;
295}
296
297template <typename realT>
298template <typename measurementT, typename wfsImageT>
299void directPhaseReconstructor<realT>::calcMeasurement( measurementT &slopes, wfsImageT &wfsImage )
300{
301 slopes.measurement.resize( m_measurementSize );
302
303 realT *imp = wfsImage.image.data();
304 for( int nn = 0; nn < m_idx->size(); ++nn )
305 {
306 slopes.measurement[nn] = *( imp + ( *m_idx )[nn] );
307 }
308}
309
310template <typename realT>
311template <typename measurementT, typename wfsImageT>
312void directPhaseReconstructor<realT>::reconstruct( measurementT &commandVect, wfsImageT &wfsImage )
313{
314
315#ifdef MXAO_USE_GPU
316
317 static measurementT slopes; // static to prevent re-alloc
318
319 calcMeasurement( slopes, wfsImage );
320
321 m_devSlopes.upload( slopes.measurement.data(), slopes.measurement.size() );
322
323 // realT alpha = 1.;
324 // realT zero = 0;
325
326 cublasStatus_t stat = cuda::cublasTgemv<realT>( *m_cublasHandle,
327 CUBLAS_OP_N,
328 m_nModes,
329 m_measurementSize,
330 m_one(),
331 m_devRecon(),
332 m_devSlopes(),
333 m_zero(),
334 m_devAmps() );
335 if( stat != CUBLAS_STATUS_SUCCESS )
336 {
337 std::cerr << "cublas error\n";
338 }
339
340 commandVect.measurement.resize( m_nModes );
341 m_devAmps.download( commandVect.measurement.data() );
342
343 commandVect.iterNo = wfsImage.iterNo;
344
345#else
346
347 BREAD_CRUMB;
348
349 /* Only needed if using "slopes" below (keep for debugging):
350 static measurementT slopes;
351 calcMeasurement(slopes, wfsImage);
352 */
353
354#pragma omp parallel for
355 for( int j = 0; j < m_nModes; ++j )
356 {
357 // The brutest forcest way, slower:
358 // commandVect.measurement[j] = (wfsImage.image*m_modes->image(j)).sum()/ m_nPix;
359
360 // The fastest non-GPU way:
361 realT amp = 0;
362 for( size_t k = 0; k < m_idx->size(); ++k )
363 {
364 amp += *( wfsImage.image.data() + ( *m_idx )[k] ) * m_recon( k, j );
365 }
366 commandVect.measurement[j] = amp;
367
368 /* Slightly slower, using the slopes calc (keep for debugging slopes calc):
369 realT amp = 0;
370 for(size_t k=0; k < slopes.measurement.size(); ++k)
371 {
372 amp += slopes.measurement[k]*m_recon(k,j);
373 }
374 commandVect.measurement[j] = amp;
375 */
376 }
377
378 commandVect.iterNo = wfsImage.iterNo;
379
380#endif
381}
382
383template <typename realT>
384void directPhaseReconstructor<realT>::initializeRMat( int nModes, realT calamp, int detRows, int detCols )
385{
386 m_nModes = nModes;
387
388 m_detRows = detRows;
389 m_detCols = detCols;
390
391 m_rMat.resize( measurementSize(), nModes );
392 m_rMat.setZero();
393
394 m_rImages.resize( m_detRows, m_detCols, m_nModes );
395}
396
397template <typename realT>
398template <typename measurementT>
399void directPhaseReconstructor<realT>::accumulateRMat( int i, measurementT &measureVec )
400{
401 int l = 0;
402 for( int j = 0; j < measureVec.measurement.rows(); ++j )
403 {
404 for( int k = 0; k < measureVec.measurement.cols(); ++k )
405 {
406 m_rMat( l, i ) = measureVec.measurement( j, k );
407 ++l;
408 }
409 }
410
411 // m_rMat.col(i) = measureVec.measurement.row(0);
412}
413
414template <typename realT>
415template <typename measurementT, typename wfsImageT>
416void directPhaseReconstructor<realT>::accumulateRMat( int i, measurementT &measureVec, wfsImageT &wfsImage )
417{
418 accumulateRMat( i, measureVec );
419 m_rImages.image( i ) = wfsImage.image;
420}
421
422template <typename realT>
424{
426 fitsHeader head;
427
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." );
432
433 ff.write( fname, m_rMat, head );
434}
435
436template <typename realT>
437void directPhaseReconstructor<realT>::saveRImages( std::string fname )
438{
440 fitsHeader head;
441
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." );
446
447 // ff.write(fname, m_rImages.data(), m_rImages.rows(), m_rImages.cols(), m_rImages.planes(), &head);
448 ff.write( fname, m_rImages, head );
449}
450
451} // namespace sim
452} // namespace AO
453} // namespace mx
454
455#endif // directPhaseReconstructor_hpp
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.
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.
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.
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
constexpr floatT six_fifths()
Return 6/5 in the specified precision.
The mxlib c++ namespace.
Definition mxError.hpp:106
A smart-pointer wrapper for cuda device pointers.
Definition cudaPtr.hpp:47