mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
milkImage.hpp
Go to the documentation of this file.
1/** \file milkImage.hpp
2 * \brief Interface to MILK::ImageStreamIO shared memory streams
3 * \ingroup image_processing_files
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 improc_milkImage_hpp
28#define improc_milkImage_hpp
29
30#ifdef MXLIB_MILK
31
32#ifdef MXLIB_CUDA
33#define HAVE_CUDA
34#endif
35
36#include <ImageStreamIO/ImageStreamIO.h>
37
38#include "../mxException.hpp"
39#include "eigenImage.hpp"
40
41namespace mx
42{
43namespace improc
44{
45
46template <typename typeT>
47struct ImageStructTypeCode;
48
49template <>
50struct ImageStructTypeCode<uint8_t>
51{
52 constexpr static int TypeCode = _DATATYPE_UINT8;
53};
54
55template <>
56struct ImageStructTypeCode<int8_t>
57{
58 constexpr static int TypeCode = _DATATYPE_INT8;
59};
60
61template <>
62struct ImageStructTypeCode<char>
63{
64 constexpr static int TypeCode = _DATATYPE_INT8;
65};
66
67template <>
68struct ImageStructTypeCode<uint16_t>
69{
70 constexpr static int TypeCode = _DATATYPE_UINT16;
71};
72
73template <>
74struct ImageStructTypeCode<int16_t>
75{
76 constexpr static int TypeCode = _DATATYPE_INT16;
77};
78
79template <>
80struct ImageStructTypeCode<uint32_t>
81{
82 constexpr static int TypeCode = _DATATYPE_UINT32;
83};
84
85template <>
86struct ImageStructTypeCode<int32_t>
87{
88 constexpr static int TypeCode = _DATATYPE_INT32;
89};
90
91template <>
92struct ImageStructTypeCode<uint64_t>
93{
94 constexpr static int TypeCode = _DATATYPE_UINT64;
95};
96template <>
97struct ImageStructTypeCode<int64_t>
98{
99 constexpr static int TypeCode = _DATATYPE_INT64;
100};
101template <>
102struct ImageStructTypeCode<float>
103{
104 constexpr static int TypeCode = _DATATYPE_FLOAT;
105};
106
107template <>
108struct ImageStructTypeCode<double>
109{
110 constexpr static int TypeCode = _DATATYPE_DOUBLE;
111};
112
113template <>
114struct ImageStructTypeCode<std::complex<float>>
115{
116 constexpr static int TypeCode = _DATATYPE_COMPLEX_FLOAT;
117};
118
119template <>
120struct ImageStructTypeCode<std::complex<double>>
121{
122 constexpr static int TypeCode = _DATATYPE_COMPLEX_DOUBLE;
123};
124
125/// Class to interface with an ImageStreamIO image in shared memory
126/**
127 *
128 * Use with Eigen::Map (aliased as mx::improc::eigenMap)
129 * \code
130 * using namespace mx::improc;
131 * milkImage<float> mim("image"); //connects to image.im.shm
132 * eigenMap<float> im(mim); //the conversion operator passes a reference to the internal map
133 * im.setRandom(); // im is now a map pointed at the image data. Eigen functions are now available.
134 * im(64,64) *= 2000;
135 * im /= 0.2;
136 * \endcode
137 * Once you have changed something via the Eigen::Map you want to notify others connected to the stream
138 * via
139 * \code
140 * mim.post();
141 * \endcode
142 *
143 * \ingroup eigen_image_processing
144 */
145template <typename _dataT>
147{
148 public:
149 typedef _dataT dataT; ///< The data type
150
151 protected:
152 std::string m_name; ///< The image name, from name.im.shm (the .im.shm should not be given).
153
154 IMAGE *m_image{ nullptr }; ///< Pointer to the ImageStreamIO IMAGE structure.
155
156 dataT *m_raw{ nullptr }; // The raw pointer address is stored here for checks
157
158 eigenMap<dataT> *m_map{ nullptr }; // An Eigen::Map of the array
159
160 uint64_t m_size_0{ 0 }; ///< The size[0] of the image when last opened.
161
162 uint64_t m_size_1{ 0 }; ///< The size[1] of the image when last opened.
163
164 public:
165 /// Default c'tor
166 milkImage();
167
168 /// Constructor which opens the specified image, which must already exist in shared memory.
169 milkImage(
170 const std::string &imname /**< [in] The image name, from name.im.shm (the .im.shm should not be given).*/ );
171
172 /// Constructor which (re-)creates the specified image with the given size
173 milkImage( const std::string &imname, ///< [in] The image name, from name.im.shm (the .im.shm should not be given).
174 uint32_t sz0, ///< [in] the x size of the image
175 uint32_t sz1 ///< [in] the y size of the image
176 );
177
178 /// Constructor which (re-)creates the specified image and copies the provided image to it
179 milkImage( const std::string &imname, ///< [in] The image name, from name.im.shm (the .im.shm should not be given).
180 const eigenImage<dataT> &im ///< [in] An existing eigenImage
181 );
182
183 /// D'tor
184 ~milkImage();
185
186 /// Open and connect to an image, allocating the eigenMap.
187 /**
188 * \throws std::invalid_argument if the image type_code does not match dataT.
189 */
190 void
191 open( const std::string &imname /**< [in] The image name, from name.im.shm (the .im.shm should not be given).*/ );
192
193 /// Create (or re-create) and connect to an image, allocating the eigenMap.
194 /**
195 * \throws std::invalid_argument if the image type_code does not match dataT.
196 */
197 void create( const std::string &imname, ///< [in] The image name, for name.im.shm (the .im.shm should not be given).
198 uint32_t sz0, ///< [in] the x size of the image
199 uint32_t sz1 ///< [in] the y size of the image
200 );
201
202 /// Create and connect to an image using an existing eigenImage, allocating the eigenMap and copying the image to
203 /// the shmim.
204 /**
205 * \throws std::invalid_argument if the image type_code does not match dataT.
206 */
207 void create( const std::string &imname, ///< [in] The image name, for name.im.shm (the .im.shm should not be given).
208 const eigenImage<dataT> &im ///< [in] An existing eigenImage
209 );
210
211 /// Get the width of the image
212 /**
213 * \returns the current value of m_size_0
214 */
215 uint32_t rows();
216
217 /// Get the height of the image
218 /**
219 * \returns the current value of m_size_1
220 */
221 uint32_t cols();
222
223 /// Get the size of a dimension of the image
224 /**
225 * \returns the current value of m_size_0 or m_size_1 depending on n
226 */
227 uint32_t size( unsigned n /**< [in] the dimension to get the size of*/ );
228
229 /// Checks if the image is connected and is still the same format as when connected.
230 /** Checks on pointer value, size[], and data_type.
231 *
232 * \todo check inode!
233 *
234 * \returns true if connected and no changes
235 * \returns false if not connected or something changed. All maps are now invalid.
236 */
237 bool valid();
238
239 /// Reopens the image.
240 /** Same as
241 * \code
242 * close();
243 * open(m_name);
244 * \endcode
245 */
246 void reopen();
247
248 // Close the image
249 void close();
250
251 /// Get an eigenMap
252 /** Use with caution:
253 * - there is no way to know if the image has changed
254 * - you should check \ref valid() before using
255 *
256 *
257 * \returns an Eigen::Map<Array,-1,-1> reference
258 *
259 * \throws mx::err::mxException if the image is not opened
260 *
261 */
263
264 /// Conversion operator returns an eigenMap
265 /** Use this like
266 * \code
267 * milkImage<float> mim("imname");
268 * eigenMap<float> im(mim); //you now have an Eigen interface to the image
269 * \endcode
270 * but with caution:
271 * - there is no way to know if the image has changed
272 * - you should check \ref valid() before using
273 *
274 *
275 * \returns an Eigen::Map<Array,-1,-1> object
276 *
277 * \throws mx::err::mxException if the image is not opened
278 *
279 */
280 operator eigenMap<dataT>();
281
282 /// Copy data from an Eigen Array type to the shared memory stream
283 /** Sets the write flag, copies using the Eigen assigment to map, unsets the write flag, then posts.
284 *
285 * \throws mxException on an error
286 *
287 */
288 template <typename eigenT>
289 milkImage &operator=( const eigenT &im /**< [in] the eigen array to copy to the stream*/ );
290
291 /// Set the write flag
292 /** The write flag is set to indicate whether or not the the data is being changed.
293 * The write flag will be set to false by \ref post().
294 *
295 * \throws mx::err::mxException if the image is not opened
296 */
297 void setWrite( bool wrflag = true /**< [in] [optional] the desired value of the write flag. Default is true.*/ );
298
299 /// Update the metadata and post all semaphores
300 /**
301 * \todo need to set wtime, have a version with atime
302 *
303 * \throws mx::err::mxException if the image is not opened
304 */
305 void post();
306};
307
308template <typename dataT>
312
313template <typename dataT>
314milkImage<dataT>::milkImage( const std::string &imname )
315{
316 open( imname );
317}
318
319template <typename dataT>
320milkImage<dataT>::milkImage( const std::string &imname, const eigenImage<dataT> &im )
321{
322 create( imname, im );
323}
324
325template <typename dataT>
326milkImage<dataT>::milkImage( const std::string &imname, uint32_t sz0, uint32_t sz1 )
327{
328 create( imname, sz0, sz1 );
329}
330
331template <typename dataT>
333{
334 close();
335}
336
337template <typename dataT>
338void milkImage<dataT>::open( const std::string &imname )
339{
340 if( m_image )
341 {
342 ImageStreamIO_closeIm( m_image );
343 delete m_image;
344 m_image = nullptr;
345 }
346
347 if( m_map )
348 {
349 delete m_map;
350 m_map = nullptr;
351 }
352
353 m_image = new IMAGE;
354
355 errno_t rv = ImageStreamIO_openIm( m_image, imname.c_str() );
356
357 if( rv != IMAGESTREAMIO_SUCCESS )
358 {
359 delete m_image;
360 m_image = nullptr;
361 throw err::mxException( "", 0, "", 0, "", 0, "ImageStreamIO_openIm returned an error" );
362 }
363
364 if( ImageStructTypeCode<dataT>::TypeCode != m_image->md->datatype )
365 {
366 delete m_image;
367 m_image = nullptr;
368 throw std::invalid_argument( "shmim datatype does not match template type" );
369 }
370
371 m_name = imname;
372 m_raw = (dataT *)m_image->array.raw;
373 m_size_0 = m_image->md->size[0];
374 m_size_1 = m_image->md->size[1];
375
376 m_map = new eigenMap<dataT>( m_raw, m_size_0, m_size_1 );
377}
378
379template <typename dataT>
380void milkImage<dataT>::create( const std::string &imname, uint32_t sz0, uint32_t sz1 )
381{
382 if( m_image )
383 {
384 ImageStreamIO_closeIm( m_image );
385 delete m_image;
386 m_image = nullptr;
387 }
388
389 if( m_map )
390 {
391 delete m_map;
392 m_map = nullptr;
393 }
394
395 m_image = new IMAGE;
396
397 uint32_t imsize[3];
398 imsize[0] = sz0;
399 imsize[1] = sz1;
400 imsize[2] = 1;
401
402 errno_t rv = ImageStreamIO_createIm_gpu( m_image,
403 imname.c_str(),
404 3,
405 imsize,
406 ImageStructTypeCode<dataT>::TypeCode,
407 -1,
408 1,
409 IMAGE_NB_SEMAPHORE,
410 0,
411 CIRCULAR_BUFFER | ZAXIS_TEMPORAL,
412 0 );
413
414 if( rv != IMAGESTREAMIO_SUCCESS )
415 {
416 delete m_image;
417 m_image = nullptr;
418 throw err::mxException( "", 0, "", 0, "", 0, "ImageStreamIO_createIm_gpu returned an error" );
419 }
420
421 if( ImageStructTypeCode<dataT>::TypeCode != m_image->md->datatype )
422 {
423 delete m_image;
424 m_image = nullptr;
425 throw std::invalid_argument( "shmim datatype does not match template type" );
426 }
427
428 m_name = imname;
429 m_raw = (dataT *)m_image->array.raw;
430 m_size_0 = m_image->md->size[0];
431 m_size_1 = m_image->md->size[1];
432
433 m_map = new eigenMap<dataT>( m_raw, m_size_0, m_size_1 );
434}
435
436template <typename dataT>
437void milkImage<dataT>::create( const std::string &imname, const eigenImage<dataT> &im )
438{
439 create( imname, im.rows(), im.cols() );
440 operator=( im );
441}
442
443template <typename dataT>
445{
446 if( m_map == nullptr )
447 {
448 throw err::mxException( "", 0, "", 0, "", 0, "Image is not open (map is null)" );
449 }
450
451 return *m_map;
452}
453
454template <typename dataT>
456{
457 return operator()();
458}
459
460template <typename dataT>
462{
463 close();
464 open( m_name );
465}
466
467template <typename dataT>
469{
470 if( m_map != nullptr )
471 {
472 delete m_map;
473 m_map = nullptr;
474 }
475
476 if( m_image == nullptr )
477 {
478 return;
479 }
480
481 errno_t rv = ImageStreamIO_closeIm( m_image );
482 delete m_image;
483 m_image = nullptr;
484
485 if( rv != IMAGESTREAMIO_SUCCESS )
486 throw err::mxException( "", 0, "", 0, "", 0, "ImageStreamIO_closeIm returned an error" );
487}
488
489template <typename dataT>
491{
492 return m_size_0;
493}
494
495template <typename dataT>
497{
498 return m_size_1;
499}
500
501template <typename dataT>
502uint32_t milkImage<dataT>::size( unsigned n )
503{
504 if( n == 0 )
505 {
506 return m_size_0;
507 }
508 else if( n == 1 )
509 {
510 return m_size_1;
511 }
512
513 return 0;
514}
515
516template <typename dataT>
518{
519 if( m_image == nullptr )
520 {
521 return false;
522 }
523
524 if( m_raw != (dataT *)m_image->array.raw || m_size_0 != m_image->md->size[0] || m_size_1 != m_image->md->size[1] ||
525 ImageStructTypeCode<dataT>::TypeCode != m_image->md->datatype )
526 {
527 return false;
528 }
529
530 return true;
531}
532
533template <typename dataT>
534template <typename eigenT>
536{
537 if( m_image == nullptr )
538 {
539 throw err::mxException( "", 0, "", 0, "", 0, "Image is not open (image is null)" );
540 }
541
542 if( m_map == nullptr )
543 {
544 throw err::mxException( "", 0, "", 0, "", 0, "Image is not open (map is null)" );
545 }
546
547 setWrite( true );
548
549 *m_map = im;
550
551 setWrite( false );
552
553 post();
554
555 return *this;
556}
557
558template <typename dataT>
559void milkImage<dataT>::setWrite( bool wrflag )
560{
561 if( m_image == nullptr )
562 {
563 throw err::mxException( "", 0, "", 0, "", 0, "Image is not open" );
564 }
565
566 m_image->md->write = wrflag;
567}
568
569template <typename dataT>
571{
572 if( m_image == nullptr )
573 {
574 throw err::mxException( "", 0, "", 0, "", 0, "Image is not open" );
575 }
576
577 errno_t rv = ImageStreamIO_UpdateIm( m_image );
578 m_image->md->atime = m_image->md->writetime;
579
580 if( rv != IMAGESTREAMIO_SUCCESS )
581 {
582 throw err::mxException( "", 0, "", 0, "", 0, "ImageStreamIO_UpdateIm returned an error" );
583 }
584}
585
586} // namespace improc
587} // namespace mx
588
589#endif // MXLIB_MILK
590#endif // improc_milkImage_hpp
The mxlib exception class.
Class to interface with an ImageStreamIO image in shared memory.
uint32_t size(unsigned n)
Get the size of a dimension of the image.
milkImage()
Default c'tor.
uint64_t m_size_1
The size[1] of the image when last opened.
void create(const std::string &imname, uint32_t sz0, uint32_t sz1)
Create (or re-create) and connect to an image, allocating the eigenMap.
milkImage & operator=(const eigenT &im)
Copy data from an Eigen Array type to the shared memory stream.
IMAGE * m_image
Pointer to the ImageStreamIO IMAGE structure.
uint32_t rows()
Get the width of the image.
_dataT dataT
The data type.
std::string m_name
The image name, from name.im.shm (the .im.shm should not be given).
bool valid()
Checks if the image is connected and is still the same format as when connected.
void setWrite(bool wrflag=true)
Set the write flag.
eigenMap< dataT > & operator()()
Get an eigenMap.
void reopen()
Reopens the image.
uint32_t cols()
Get the height of the image.
void post()
Update the metadata and post all semaphores.
void open(const std::string &imname)
Open and connect to an image, allocating the eigenMap.
uint64_t m_size_0
The size[0] of the image when last opened.
Tools for using the eigen library for image processing.
Eigen::Array< scalarT, -1, -1 > eigenImage
Definition of the eigenImage type, which is an alias for Eigen::Array.
Eigen::Map< Eigen::Array< scalarT, -1, -1 > > eigenMap
Definition of the eigenMap type, which is an alias for Eigen::Map<Array>.
The mxlib c++ namespace.
Definition mxError.hpp:106