mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
imagingArray.hpp
Go to the documentation of this file.
1/** \file imagingArray.hpp
2 * \brief Declares and defines a class for managing images
3 * \ingroup imaging
4 * \author Jared R. Males (jaredmales@gmail.com)
5 *
6 */
7
8#ifndef imagingArray_hpp
9#define imagingArray_hpp
10
11#pragma GCC system_header
12#include <Eigen/Dense>
13
14#include "../math/templateBLAS.hpp"
15#include "../math/fft/fft.hpp"
16
17namespace mx
18{
19namespace wfp
20{
21
22template <typename Scalar>
23struct fftwAllocator
24{
25 Scalar *alloc( size_t sz )
26 {
27 return (Scalar *)::fftw_malloc( sz * sizeof( Scalar ) );
28 }
29
30 void free( Scalar *ptr )
31 {
32 fftw_free( ptr );
33 }
34};
35
36/// Test whether a type is an imagingArray by testing whether it has a typedef of "is_imagingArray"
37/** Used for compile-time determination of type
38 * Example usage:
39 * \code
40 * bool is_eC = is_imagingArray<eigenCube<float> >; //Evaluates to true
41 * bool is_not_eC = is_imagingArray<float>; //Evaluates to false
42 * \endcode
43 */
44// This was taken directly from the example at http://en.wikipedia.org/wiki/Substitution_failure_is_not_an_error
45template <typename T>
47{
48 // Types "yes" and "no" are guaranteed to have different sizes,
49 // specifically sizeof(yes) == 1 and sizeof(no) == 2.
50 typedef char yes[1];
51 typedef char no[2];
52
53 template <typename imageT>
54 static yes &test( typename imageT::imagingArrayT * );
55
56 template <typename>
57 static no &test( ... );
58
59 // If the "sizeof" of the result of calling test<T>(0) would be equal to sizeof(yes),
60 // the first overload worked and T has a nested type named "is_mmatrix".
61 static const bool value = sizeof( test<T>( 0 ) ) == sizeof( yes );
62};
63
64template <typename imageT, typename typeT, bool isScalar = is_imagingArray<typeT>::value>
65struct imagingArrayInplaceProduct
66{
67 imagingArrayInplaceProduct()
68 {
69 static_assert( is_imagingArray<imageT>::value,
70 "imagingArrayInplaceProduct template parameter 1 must be an imagingArray type" );
71 }
72
73 void operator()( imageT &im, typeT &alpha )
74 {
75 math::scal<typename imageT::Scalar>( im.cols() * im.rows(), (typeT)alpha, im.data(), 1 );
76 }
77};
78
79template <typename imageT, typename typeT>
80struct imagingArrayInplaceProduct<imageT, typeT, true>
81{
82 imagingArrayInplaceProduct()
83 {
84 static_assert( is_imagingArray<imageT>::value,
85 "imagingArrayInplaceProduct template parameter 1 must be an imagingArray type" );
86 }
87
88 void operator()( imageT &im, typeT &alpha )
89 {
90 typedef typename imageT::Scalar scalarT;
91 // hadp<scalarT>(im.cols()*im.rows(), alpha.data(), im.data());
92
93 Eigen::Map<Eigen::Array<scalarT, -1, -1>> eigY( alpha.data(), alpha.cols(), alpha.rows() );
94 Eigen::Map<Eigen::Array<scalarT, -1, -1>> eigX( im.data(), im.cols(), im.rows() );
95
96 eigX *= eigY;
97 }
98};
99
100template <typename imageT, typename typeT, bool isScalar = is_imagingArray<typeT>::value>
101struct imagingArrayInplaceDivision
102{
103 imagingArrayInplaceDivision()
104 {
105 static_assert( is_imagingArray<imageT>::value,
106 "imagingArrayInplaceDivision template parameter 1 must be an imagingArray type" );
107 }
108
109 void operator()( imageT im, typeT alpha )
110 {
111 math::scal<typename imageT::Scalar>( im.cols() * im.rows(), ( (typeT)( 1 ) ) / alpha, im.data(), 1 );
112 }
113};
114
115template <typename imageT, typename typeT>
116struct imagingArrayInplaceDivision<imageT, typeT, true>
117{
118 imagingArrayInplaceDivision()
119 {
120 static_assert( is_imagingArray<imageT>::value,
121 "imagingArrayInplaceDivision template parameter 1 must be an imagingArray type" );
122 }
123
124 void operator()( imageT im, typeT alpha )
125 {
126 typedef typename imageT::Scalar scalarT;
127 math::hadd<scalarT>( im.cols() * im.rows(), (scalarT)1.0, alpha.data(), 1, im.data(), 1 );
128 }
129};
130
131template <typename imageT, typename typeT, bool isScalar = is_imagingArray<typeT>::value>
132struct imagingArrayInplaceAdd
133{
134 imagingArrayInplaceAdd()
135 {
136 static_assert( is_imagingArray<imageT>::value,
137 "imagingArrayInplaceAdd template parameter 1 must be an imagingArray type" );
138 }
139
140 void operator()( imageT &im, typeT &alpha )
141 {
142 typedef typename imageT::Scalar scalarT;
143 Eigen::Map<Eigen::Array<scalarT, -1, -1>> eigX( im.data(), im.cols(), im.rows() );
144 eigX += alpha;
145 }
146};
147
148template <typename imageT, typename typeT>
149struct imagingArrayInplaceAdd<imageT, typeT, true>
150{
151 imagingArrayInplaceAdd()
152 {
153 static_assert( is_imagingArray<imageT>::value,
154 "imagingArrayInplaceAdd template parameter 1 must be an imagingArray type" );
155 }
156
157 void operator()( imageT &im, typeT &alpha )
158 {
159 typedef typename imageT::Scalar scalarT;
160 // hadp<scalarT>(im.cols()*im.rows(), alpha.data(), im.data());
161
162 Eigen::Map<Eigen::Array<scalarT, -1, -1>> eigY( alpha.data(), alpha.cols(), alpha.rows() );
163 Eigen::Map<Eigen::Array<scalarT, -1, -1>> eigX( im.data(), im.cols(), im.rows() );
164
165 eigX += eigY;
166 }
167};
168
169template <typename _Scalar, class _allocatorT, int cudaGPU>
170class imagingArray;
171
172template <typename _Scalar, class _allocatorT>
173class imagingArray<_Scalar, _allocatorT, 0>
174{
175 public:
176 typedef bool imagingArrayT;
177 typedef _Scalar Scalar;
178 typedef _allocatorT allocatorT;
179
180 protected:
181 allocatorT allocator;
182
183 Scalar *_data;
184
185 bool _owner;
186
187 int _cols;
188 int _rows;
189
190 public:
191 imagingArray()
192 {
193 _data = 0;
194 _owner = false;
195 _cols = 0;
196 _rows = 0;
197 }
198
199 imagingArray( int x, int y )
200 {
201 _data = 0;
202 _owner = false;
203 _cols = 0;
204 _rows = 0;
205
206 resize( x, y );
207 }
208
209 template <typename eigenT>
210 explicit imagingArray( eigenT &im )
211 {
212 _data = im.data();
213 _owner = false;
214 _cols = im.cols();
215 _rows = im.rows();
216 }
217
218 ~imagingArray()
219 {
220 free();
221 }
222
223 void resize( int szx, int szy )
224 {
225 if( szx == _cols && szy == _rows && _owner )
226 return;
227
228 free();
229
230 _cols = szx;
231 _rows = szy;
232
233 _data = allocator.alloc( _cols * _rows );
234 _owner = true;
235 }
236
237 void free()
238 {
239 if( _data && _owner )
240 {
241 allocator.free( _data );
242 _data = 0;
243 _cols = 0;
244 _rows = 0;
245 }
246 }
247
248 int cols() const
249 {
250 return _cols;
251 }
252
253 int rows() const
254 {
255 return _rows;
256 }
257
258 Scalar *data()
259 {
260 return _data;
261 }
262
263 Scalar *data() const
264 {
265 return _data;
266 }
267
268 Scalar operator()( int x, int y ) const
269 {
270 return _data[x + y * _cols];
271 }
272
273 Scalar &operator()( int x, int y )
274 {
275 return _data[x + y * _cols];
276 }
277
278 template <typename argT>
279 void set( argT val )
280 {
281 int N = _cols * _rows;
282
283 Scalar dval = (Scalar)val;
284 for( int i = 0; i < N; ++i )
285 _data[i] = dval;
286 }
287
288 void setZero()
289 {
290 set( (Scalar)0 );
291 }
292
293 imagingArray &operator=( const imagingArray &arr )
294 {
295 resize( arr.rows(), arr.cols() );
296
297 for( int i = 0; i < _rows; ++i )
298 {
299 for( int j = 0; j < _cols; ++j )
300 {
301 operator()( i, j ) = arr( i, j );
302 }
303 }
304
305 return ( *this );
306 }
307
308 /// In-place product. Works for both scalars and imagingArray types.
309 template <typename typeT>
310 imagingArray &operator*=( typeT &alpha )
311 {
312 imagingArrayInplaceProduct<imagingArray, typeT> prod;
313 prod( *this, alpha );
314 return *this;
315 }
316
317 /// In-place division. Works for both scalars and imagingArray types.
318 template <typename typeT>
319 imagingArray &operator/=( typeT alpha )
320 {
321 imagingArrayInplaceDivision<imagingArray, typeT> div;
322 div( *this, alpha );
323 return *this;
324 }
325
326 /// In-place add. Works for both scalars and imagingArray types.
327 template <typename typeT>
328 imagingArray &operator+=( typeT &alpha )
329 {
330 imagingArrayInplaceAdd<imagingArray, typeT> prod;
331 prod( *this, alpha );
332 return *this;
333 }
334
335 Scalar sum()
336 {
337 int N = _cols * _rows;
338 Scalar _sum = 0;
339
340 for( int i = 0; i < N; ++i )
341 _sum += _data[i];
342
343 return _sum;
344 }
345
346 Eigen::Map<Eigen::Array<Scalar, -1, -1>> eigenMap()
347 {
348 return Eigen::Map<Eigen::Array<Scalar, -1, -1>>( _data, _rows, _cols );
349 }
350};
351
352template <typename scalarT>
353using imagingArrayT = imagingArray<scalarT, fftwAllocator<scalarT>, 0>;
354
355// typedef imagingArray<float,mx::fftwAllocator<float>,0> imagingArrayf;
356// typedef imagingArray<double,mx::fftwAllocator<double>,0> imagingArrayd;
357// typedef imagingArray<std::complex<float>,mx::fftwAllocator<std::complex<float> >,0> imagingArraycf;
358
359} // namespace wfp
360} // namespace mx
361
362#endif // imagingArray_hpp
Eigen::Map< Eigen::Array< scalarT, -1, -1 > > eigenMap
Definition of the eigenMap type, which is an alias for Eigen::Map<Array>.
void fftw_free(realT *p)
Call to fftw_free.
constexpr floatT six_fifths()
Return 6/5 in the specified precision.
The mxlib c++ namespace.
Definition mxError.hpp:106
Test whether a type is an imagingArray by testing whether it has a typedef of "is_imagingArray".