mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
imagingUtils.hpp
Go to the documentation of this file.
1/** \file imagingUtils.hpp
2 * \brief Utilities for modeling image formation
3 * \ingroup imaging
4 * \author Jared R. Males (jaredmales@gmail.com)
5 *
6 */
7
8//***********************************************************************//
9// Copyright 2015, 2016, 2017 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 imagingUtils_hpp
28#define imagingUtils_hpp
29
30#include <cmath>
31
32#include "../math/constants.hpp"
33#include "../mxError.hpp"
34
35#include "imagingArray.hpp"
36
37namespace mx
38{
39
40namespace wfp
41{
42
43/// Calculate the angular plate scale (radians per pixel) of an image after propagation by FFT.
44/**
45 *
46 * \returns the platescale of the wavefront after propagation by FFT, in radians per pixel.
47 *
48 * \tparam realT a real floating point type
49 *
50 * \ingroup imaging
51 */
52template <typename realT>
53realT fftPlateScale( size_t pixels, ///< [in] the linear dimension of the FFT (including 0 pad, etc.)
54 realT metersPerPixel, ///< [in] the scale of the input wavefront [m/pix]
55 realT lambda ///< [in] the wavelength of the wavefront [m]
56)
57{
58 return ( lambda / metersPerPixel ) * ( 1. / pixels );
59}
60
61/// Fill in an Eigen-like array with a circular pupil mask.
62/** Sets any pixel which is at rad <= r < rad+(1.0/overscan) pixels to rho = 1,
63 *
64 *
65 * \retval 0 on success
66 * \retval -1 on error
67 *
68 * \ingroup imaging
69 */
70template <class arrayT>
71int circularPupil( arrayT &m, ///< [in.out] is the allocated Array. Dimensions are used to create the pupil.
72 typename arrayT::Scalar eps = 0, ///< [in] [optional] is the central obscuration. 0-1, default is 0.
73 typename arrayT::Scalar rad = 0, ///< [in] [optional] is the desired radius. If rad <= 0, then the
74 ///< maximum radius based on dimensions of m is used. Default is 0.
75 typename arrayT::Scalar overscan = 0 ///< [in] [optional] overscan in fractional pixels, to include
76 ///< partial pixels on the edge. Default is 0.
77)
78{
79
80 if( eps < 0 )
81 {
82 mxError( "circularPupil", MXE_INVALIDARG, "Central obscuration can not be < 0." );
83 return -1;
84 }
85
86 if( eps > 1 )
87 {
88 mxError( "circularPupil", MXE_INVALIDARG, "Central obscuration can not be > 1." );
89 return -1;
90 }
91
92 size_t l0 = m.rows();
93 size_t l1 = m.cols();
94
95 typename arrayT::Scalar r;
96 typename arrayT::Scalar xc = 0.5 * ( l0 - 1 );
97 typename arrayT::Scalar yc = 0.5 * ( l1 - 1 );
98
99 if( rad <= 0 )
100 rad = 0.5 * std::min( l0 - 1, l1 - 1 );
101
102 for( size_t i = 0; i < l0; i++ )
103 {
104 for( size_t j = 0; j < l1; j++ )
105 {
106 r = std::sqrt( std::pow( i - xc, 2 ) + std::pow( j - yc, 2 ) );
107
108 if( r <= rad + ( overscan ) && r >= eps * rad )
109 m( i, j ) = 1;
110 else
111 m( i, j ) = 0;
112 }
113 }
114
115 return 0;
116}
117
118/// Draw a line in an image
119/** \todo should handle width much more intelligently, this only works for ~45 degree lines.
120 *
121 * \tparam arrayT is an Eigen-like array with public typedef Scalar
122 */
123template <class arrayT>
124void drawLine( arrayT &im, ///< [in.out] The input image, modified.
125 typename arrayT::Scalar x0, ///< [in] the x value, relative to image center, of the starting point
126 typename arrayT::Scalar y0, ///< [in] the y value, relative to image center, of the starting point
127 typename arrayT::Scalar x1, ///< [in] the x value, relative to image center, of the end point
128 typename arrayT::Scalar y1, ///< [in] the y value, relative to image center, of the end point
129 typename arrayT::Scalar halfWidth ///< [in] the half-width of the line.
130)
131{
132
133 int d1 = im.rows();
134 int d2 = im.cols();
135
136 typename arrayT::Scalar xc, yc;
137
138 xc = 0.5 * ( im.rows() - 1 );
139 yc = 0.5 * ( im.cols() - 1 );
140
141 typename arrayT::Scalar m = ( y1 - y0 ) / ( x1 - x0 );
142 int y;
143
144 if( x1 > x0 )
145 {
146 for( int x = x0; x <= x1; ++x )
147 {
148 y = y0 + ( x - x0 ) * m;
149
150 for( int i = 0; i <= halfWidth; ++i )
151 {
152 if( x + xc >= 0 && x + xc < d1 && y + yc + i >= 0 && y + yc + i < d2 )
153 im( x + xc, y + yc + i ) = 0;
154 if( x + xc >= 0 && x + xc < d1 && y + yc - i >= 0 && y + yc - i < d2 )
155 im( x + xc, y + yc - i ) = 0;
156 }
157 }
158 }
159 else
160 {
161 for( int x = x1; x <= x0; ++x )
162 {
163 y = y0 + ( x - x0 ) * m;
164
165 for( int i = 0; i <= halfWidth; ++i )
166 {
167 if( x + xc >= 0 && x + xc < d1 && y + yc + i >= 0 && y + yc + i < d2 )
168 im( x + xc, y + yc + i ) = 0;
169 if( x + xc >= 0 && x + xc < d1 && y + yc - i >= 0 && y + yc - i < d2 )
170 im( x + xc, y + yc - i ) = 0;
171 }
172 }
173 }
174}
175
176/// Create a complex pupil plane wavefront from a real amplitude mask.
177/** The input real amplitude mask is placed in the center of a 0-padded complex array.
178 *
179 * \param [out] complexPupil the complex pupil plane wavefront
180 * \param [in] realPupil a real amplitude mask.
181 * \param [in] wavefrontSizePixels the desired size of the ouput wavefront, should be at least as big as realPupil
182 *
183 * \ingroup imaging
184 */
185template <typename arrayOutT, typename arrayInT>
186void makeComplexPupil( arrayOutT &complexPupil, const arrayInT &realPupil, int wavefrontSizePixels )
187{
188
189 complexPupil.resize( wavefrontSizePixels, wavefrontSizePixels );
190 complexPupil.set( typename arrayOutT::Scalar( 0, 0 ) );
191
192 // Lower-left corner of insertion region
193 int bl = 0.5 * ( complexPupil.rows() - 1 ) - 0.5 * ( realPupil.rows() - 1. );
194
195 for( int i = 0; i < realPupil.cols(); ++i )
196 {
197 for( int j = 0; j < realPupil.rows(); ++j )
198 {
199 complexPupil( bl + i, bl + j ) =
200 typename arrayOutT::Scalar( realPupil( i, j ), 0 ); //*exp( typename arrayOutT::Scalar(0,1));
201 }
202 }
203 // complexPupil.block(bl, bl, realPupil.rows(), realPupil.rows()) = realPupil*std::complex<realT>(1,0);
204}
205
206/// Create a complex wavefront from a real amplitude and a real phase.
207/** The wavefront is placed in the center of a 0-padded complex array.
208 *
209 * \param [out] complexWavefront the complex pupil plane wavefront
210 * \param [in] realAmplitude is the real-valued amplitude.
211 * \param [in] realPhase is the real-valued phase in radians, same size as realAmplitude
212 * \param [in] wavefrontSizePixels the desired size of the ouput wavefront, should be at least as big as the real arrays
213 *
214 * \ingroup imaging
215 */
216template <typename arrayOutT, typename arrayInT>
217void makeComplexPupil( arrayOutT &complexWavefront,
218 const arrayInT &realAmplitude,
219 const arrayInT &realPhase,
220 int wavefrontSizePixels )
221{
222
223 complexWavefront.resize( wavefrontSizePixels, wavefrontSizePixels );
224 complexWavefront.set( typename arrayOutT::Scalar( 0, 0 ) );
225
226 // Lower-left corner of insertion region
227 int bl = 0.5 * ( complexWavefront.rows() - 1 ) - 0.5 * ( realAmplitude.rows() - 1. );
228
229 for( int i = 0; i < realAmplitude.cols(); ++i )
230 {
231 for( int j = 0; j < realAmplitude.rows(); ++j )
232 {
233 complexWavefront( bl + j, bl + i ) =
234 realAmplitude( j, i ) * exp( ( typename arrayOutT::Scalar( 0, 1 ) ) * realPhase( j, i ) );
235 }
236 }
237}
238
239/// Apply a tilt to a wavefront
240/**
241 * \param complexWavefront [in.out] the complex wavefront to tilt, will be modified on output
242 * \param xTilt [input] the amount of tilt in the x direction, in pixels
243 * \param yTilt [input] the amount of tilt in the y direction, in pixels
244 *
245 * \ingroup imaging
246 */
247template <typename wavefrontT>
248void tiltWavefront( wavefrontT &complexWavefront,
249 typename wavefrontT::Scalar::value_type xTilt,
250 typename wavefrontT::Scalar::value_type yTilt )
251{
252 typedef typename wavefrontT::Scalar complexT;
253 typedef typename wavefrontT::Scalar::value_type realT;
254
255 realT pi = math::pi<realT>();
256
257 int wfsSizeX = complexWavefront.cols();
258 int wfsSizeY = complexWavefront.rows();
259
260 realT xCen = 0.5 * ( wfsSizeX - 1 );
261 realT yCen = 0.5 * ( wfsSizeY - 1 );
262
263 realT argX = 2.0 * pi / ( wfsSizeX - 1.0 );
264 realT argY = 2.0 * pi / ( wfsSizeY - 1.0 );
265
266 for( int ii = 0; ii < wfsSizeX; ++ii )
267 {
268 for( int jj = 0; jj < wfsSizeY; ++jj )
269 {
270 complexWavefront( ii, jj ) =
271 complexWavefront( ii, jj ) *
272 exp( complexT( (realT)0., argX * xTilt * ( ii - xCen ) + argY * yTilt * ( jj - yCen ) ) );
273 }
274 }
275}
276
277/// Extract a block from one image and insert it into a second
278template <typename imageT1,
279 typename imageT2>
280void extractBlock( imageT1 &dest, ///< [in/out] the image in which to place the extracted block. Must be pre-allocated.
281 int imX0, ///< [in] the x/row-coord of the lower left pixel of the destination region.
282 int imXsz, ///< [in] the x size (number of rows) of the block
283 int imY0, ///< [in] the y/col-coord of the lower left pixel of the desination region
284 int imYsz, ///< [in] the y=size (number of cols) ov the block
285 imageT2 &src, ///< [in] the source of the block. Must be large enough.
286 int wfX0, ///< [in] the x/row-coord of the lower left pixel of the source region
287 int wfY0 ) ///< [in] the y/col-coord of the lower left pixel of the source region
288{
289 int dest_cols = dest.cols();
290
291 int src_cols = src.cols();
292
293 typedef typename imageT1::Scalar dataT;
294
295 dataT *dest_data;
296 dataT *src_data;
297
298 for( int j = 0; j < imYsz; ++j )
299 {
300 dest_data = &dest.data()[imX0 + ( imY0 + j ) * dest_cols];
301 src_data = &src.data()[wfX0 + ( wfY0 + j ) * src_cols];
302
303 memcpy( dest_data, src_data, sizeof( dataT ) * imXsz );
304 }
305}
306
307/// Extract a pixels from one image and insert them into a second based on a mask
308/** Only pixels with a non-zero value in mask are changed in dest to have the value in src. Other pixels are not
309 * modified.
310 */
311template <typename imageT1,
312 typename imageT2,
313 typename imageT3>
314void extractMaskedPixels( imageT1 &dest, ///< [in/out] the image in which to place the extracted pixels. Must be the
315 ///< same size as src and mask.
316 const imageT2 &src, ///< [in] the source of the pixels. Must be the same size as mask.
317 const imageT3 &mask ///< [in] the mask image, where any value other than 0 indicates a pixel
318 ///< to extract. Must be the same size as src.
319)
320{
321 if( dest.rows() != src.rows() )
322 {
323 mxThrowException(
324 mx::err::sizeerr, "mx::imagingUtils::extractMaskedPixels", "dest and src do not have same size (rows)" );
325 }
326
327 if( dest.cols() != src.cols() )
328 {
329 mxThrowException(
330 mx::err::sizeerr, "mx::imagingUtils::extractMaskedPixels", "dest and src do not have same size (cols)" );
331 }
332
333 if( src.rows() != mask.rows() )
334 {
335 mxThrowException(
336 mx::err::sizeerr, "mx::imagingUtils::extractMaskedPixels", "src and mask do not have same size (rows)" );
337 }
338
339 if( src.cols() != mask.cols() )
340 {
341 mxThrowException(
342 mx::err::sizeerr, "mx::imagingUtils::extractMaskedPixels", "src and mask do not have same size (cols)" );
343 }
344
345 for( int cc = 0; cc < dest.cols(); ++cc )
346 {
347 for( int rr = 0; rr < dest.rows(); ++rr )
348 {
349 if( mask( rr, cc ) != 0 )
350 {
351 dest( rr, cc ) = src( rr, cc );
352 }
353 }
354 }
355}
356
357template <typename realImageT, typename complexImageT>
358void extractIntensityImage(
359 realImageT &im, int imX0, int imXsz, int imY0, int imYsz, complexImageT &wf, int wfX0, int wfY0 )
360{
361 int im_rows = im.cols();
362
363 int wf_rows = wf.cols();
364
365 typename realImageT::Scalar *im_data;
366 typename complexImageT::Scalar *wf_data;
367
368 for( int j = 0; j < imXsz; ++j )
369 {
370 im_data = &im.data()[imX0 + ( imY0 + j ) * im_rows];
371 wf_data = &wf.data()[wfX0 + ( wfY0 + j ) * wf_rows];
372 for( int i = 0; i < imYsz; ++i )
373 {
374 im_data[i] = norm( wf_data[i] );
375 }
376 }
377}
378
379template <typename realImageT, typename complexImageT>
380void extractIntensityImageAccum(
381 realImageT &im, int imX0, int imXsz, int imY0, int imYsz, complexImageT &wf, int wfX0, int wfY0 )
382{
383 int im_rows = im.cols();
384
385 int wf_rows = wf.cols();
386
387 typename realImageT::Scalar *im_data;
388 typename complexImageT::Scalar *wf_data;
389
390 for( int j = 0; j < imXsz; ++j )
391 {
392 im_data = &im.data()[imX0 + ( imY0 + j ) * im_rows];
393 wf_data = &wf.data()[wfX0 + ( wfY0 + j ) * wf_rows];
394 for( int i = 0; i < imYsz; ++i )
395 {
396 im_data[i] += norm( wf_data[i] );
397 }
398 }
399}
400
401} // namespace wfp
402} // namespace mx
403
404#endif //__imagingUtils_hpp__
mxException for a size error
constexpr floatT six_fifths()
Return 6/5 in the specified precision.
void tiltWavefront(wavefrontT &complexWavefront, typename wavefrontT::Scalar::value_type xTilt, typename wavefrontT::Scalar::value_type yTilt)
Apply a tilt to a wavefront.
void makeComplexPupil(arrayOutT &complexPupil, const arrayInT &realPupil, int wavefrontSizePixels)
Create a complex pupil plane wavefront from a real amplitude mask.
realT fftPlateScale(size_t pixels, realT metersPerPixel, realT lambda)
Calculate the angular plate scale (radians per pixel) of an image after propagation by FFT.
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.
void extractMaskedPixels(imageT1 &dest, const imageT2 &src, const imageT3 &mask)
Extract a pixels from one image and insert them into a second based on a mask.
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