API
 
Loading...
Searching...
No Matches
shmimMonitor.hpp
Go to the documentation of this file.
1/** \file shmimMonitor.hpp
2 * \brief The MagAO-X generic shared memory monitor.
3 *
4 * \author Jared R. Males (jaredmales@gmail.com)
5 *
6 * \ingroup app_files
7 */
8
9#ifndef shmimMonitor_hpp
10#define shmimMonitor_hpp
11
12#include <ImageStreamIO/ImageStruct.h>
13#include <ImageStreamIO/ImageStreamIO.h>
14
15#include "../../libMagAOX/common/paths.hpp"
16
17namespace MagAOX
18{
19namespace app
20{
21namespace dev
22{
23
24struct shmimT
25{
26 static std::string configSection()
27 {
28 return "shmimMonitor";
29 };
30
31 static std::string indiPrefix()
32 {
33 return "sm";
34 };
35};
36
37/** MagAO-X generic shared memory monitor
38 *
39 *
40 * The derived class `derivedT` has the following requirements (see below for discussion of specificT):
41 *
42 * - Must be derived from MagAOXApp<true>
43 *
44 * - Must contain the following friend declaration:
45 * \code
46 * friend class dev::shmimMonitor<derivedT, specificT>; //specificT may not need to be included
47 * \endcode
48 *
49 * * - Must contain the following typedef:
50 * \code
51 * typedef dev::shmimMonitor<derivedT, specificT> shmimMonitorT; //specificT may not need to be included
52 * \endcode
53 *
54 * - Must provide the following interfaces:
55 * \code
56 *
57 * //The allocate function is called after connecting to the shared memory buffer
58 * //It should check that the buffer has the expected size, and perform any internal allocations
59 * //to prepare for processing.
60 * int derivedT::allocate( const specificT & ///< [in] tag to differentiate shmimMonitor parents. Normally this is dev::shmimT for a single parent.
61 * );
62 *
63 * int derivedT::processImage( void * curr_src, ///< [in] pointer to the start of the current frame
64 * const specificT & ///< [in] tag to differentiate shmimMonitor parents. Normally this is dev::shmimT for a single parent.
65 * )
66 * \endcode
67 * Each of the above functions should return 0 on success, and -1 on an error.
68 *
69 * - Calls to this class's `setupConfig`, `loadConfig`, `appStartup`, `appLogic`, `updateINDI` and `appShutdown`
70 * functions must be placed in the derived class's functions of the same name. For convenience the
71 * following macros are defined to provide error checking:
72 * \code
73 * SHMIMMONITOR_SETUP_CONFIG( cfig )
74 * SHMIMMONITOR_LOAD_CONFIG( cfig )
75 * SHMIMMONITOR_APP_STARTUP
76 * SHMIMMONITOR_APP_LOGIC
77 * SHMIMMONITOR_UPDATE_INDI
78 * SHMIMMONITOR_APP_SHUTDOWN
79 * \endcode
80 * See below for the macros to use if `specificT` is used to specialize.
81 *
82 * The template specifier `specificT` allows inheritance of multiple shmimMonitor classes. This type must have at least
83 * the static member function:
84 * \code
85 * static std::string indiPrefix()
86 * \endcode
87 * which returns the string which is prefixed to INDI properties. The default `shmimT` uses "sm".
88 *
89 * Additionally, if `specificT` is used, the following convenience macros can be used:
90 * \code
91 * SHMIMMONITORT_SETUP_CONFIG( SHMIMMONITORT, cfig )
92 * SHMIMMONITORT_LOAD_CONFIG( SHMIMMONITORT, cfig )
93 * SHMIMMONITORT_APP_STARTUP( SHMIMMONITORT )
94 * SHMIMMONITORT_APP_LOGIC( SHMIMMONITORT )
95 * SHMIMMONITORT_APP_SHUTDOWN( SHMIMMONITORT )
96 * \endcode
97 *
98 *
99 * \ingroup appdev
100 */
101template <class derivedT, class specificT = shmimT>
103{
104
105protected:
106 /** \name Configurable Parameters
107 * @{
108 */
109 std::string m_shmimName{""}; ///< The name of the shared memory image, is used in `/tmp/<shmimName>.im.shm`. Derived classes should set a default.
110
111 int m_smThreadPrio{2}; ///< Priority of the shmimMonitor thread, should normally be > 00.
112
113 std::string m_smCpuset; ///< The cpuset to assign the shmimMonitor thread to. Ignored if empty (the default).
114
115 ///@}
116
117 bool m_getExistingFirst{false}; ///< If set to true by derivedT, any existing image will be grabbed and sent to processImage before waiting on the semaphore.
118
119 int m_semaphoreNumber{5}; ///< The image structure semaphore index.
120
121 uint32_t m_width{0}; ///< The width of the images in the stream
122 uint32_t m_height{0}; ///< The height of the images in the stream
123 uint32_t m_depth{0}; ///< The depth of the circular buffer in the stream
124
125 uint8_t m_dataType{0}; ///< The ImageStreamIO type code.
126 size_t m_typeSize{0}; ///< The size of the type, in bytes. Result of sizeof.
127
128 IMAGE m_imageStream; ///< The ImageStreamIO shared memory buffer.
129
130 ino_t m_inode{0}; ///< The inode of the image stream file
131
132public:
133
134 const std::string & shmimName() const;
135
136 const uint32_t &width() const;
137
138 const uint32_t &height() const;
139
140 const uint32_t &depth() const;
141
142 const uint8_t &dataType() const;
143
144 const size_t &typeSize() const;
145
146 /// Setup the configuration system
147 /**
148 * This should be called in `derivedT::setupConfig` as
149 * \code
150 shmimMonitor<derivedT, specificT>::setupConfig(config);
151 \endcode
152 * with appropriate error checking.
153 */
154 int setupConfig(mx::app::appConfigurator &config /**< [out] the derived classes configurator*/);
155
156 /// load the configuration system results
157 /**
158 * This should be called in `derivedT::loadConfig` as
159 * \code
160 shmimMonitor<derivedT, specificT>::loadConfig(config);
161 \endcode
162 * with appropriate error checking.
163 */
164 int loadConfig(mx::app::appConfigurator &config /**< [in] the derived classes configurator*/);
165
166 /// Startup function
167 /** Starts the shmimMonitor thread
168 * This should be called in `derivedT::appStartup` as
169 * \code
170 shmimMonitor<derivedT, specificT>::appStartup();
171 \endcode
172 * with appropriate error checking.
173 *
174 * \returns 0 on success
175 * \returns -1 on error, which is logged.
176 */
178
179 /// Checks the shmimMonitor thread
180 /** This should be called in `derivedT::appLogic` as
181 * \code
182 shmimMonitor<derivedT, specificT>::appLogic();
183 \endcode
184 * with appropriate error checking.
185 *
186 * \returns 0 on success
187 * \returns -1 on error, which is logged.
188 */
189 int appLogic();
190
191 /// Shuts down the shmimMonitor thread
192 /** This should be called in `derivedT::appShutdown` as
193 * \code
194 shmimMonitor<derivedT, specificT>::appShutdown();
195 \endcode
196 * with appropriate error checking.
197 *
198 * \returns 0 on success
199 * \returns -1 on error, which is logged.
200 */
202
203protected:
204 /** \name SIGSEGV & SIGBUS signal handling
205 * These signals occur as a result of a ImageStreamIO source server resetting (e.g. changing frame sizes).
206 * When they occur a restart of the shmim monitor thread main loops is triggered.
207 *
208 * @{
209 */
210 bool m_restart{false}; ///< Flag indicating tha the shared memory should be reinitialized.
211
212 static shmimMonitor *m_selfMonitor; ///< Static pointer to this (set in constructor). Used for getting out of the static SIGSEGV handler.
213
214 /** \name shmimmonitor Thread
215 * This thread actually monitors the shared memory buffer
216 * @{
217 */
218
219 bool m_smThreadInit{true}; ///< Synchronizer for thread startup, to allow priority setting to finish.
220
221 pid_t m_smThreadID{0}; ///< The s.m. thread PID.
222
223 pcf::IndiProperty m_smThreadProp; ///< The property to hold the s.m. thread details.
224
225 std::thread m_smThread; ///< A separate thread for the actual monitoring
226
227 /// Thread starter, called by MagAOXApp::threadStart on thread construction. Calls smThreadExec.
228 static void smThreadStart(shmimMonitor *s /**< [in] a pointer to a shmimMonitor instance (normally this) */);
229
230 /// Execute the monitoring thread
232
233 ///@}
234
235 /** \name INDI
236 *
237 *@{
238 */
239protected:
240 // declare our properties
241
242 pcf::IndiProperty m_indiP_shmimName; ///< Property used to report the shmim buffer name
243
244 pcf::IndiProperty m_indiP_frameSize; ///< Property used to report the current frame size
245
246public:
247 /// Update the INDI properties for this device controller
248 /** You should call this once per main loop.
249 * It is not called automatically.
250 *
251 * \returns 0 on success.
252 * \returns -1 on error.
253 */
255
256 ///@}
257
258private:
259 derivedT &derived()
260 {
261 return *static_cast<derivedT *>(this);
262 }
263};
264
265// Set self pointer to null so app starts up uninitialized.
266template <class derivedT, class specificT>
267shmimMonitor<derivedT, specificT> *shmimMonitor<derivedT, specificT>::m_selfMonitor = nullptr;
268
269template <class derivedT, class specificT>
271{
272 return m_shmimName;
273}
274
275template <class derivedT, class specificT>
277{
278 return m_width;
279}
280
281template <class derivedT, class specificT>
283{
284 return m_height;
285}
286
287template <class derivedT, class specificT>
289{
290 return m_depth;
291}
292
293template <class derivedT, class specificT>
295{
296 return m_dataType;
297}
298
299template <class derivedT, class specificT>
301{
302 return m_typeSize;
303}
304
305template <class derivedT, class specificT>
306int shmimMonitor<derivedT, specificT>::setupConfig(mx::app::appConfigurator &config)
307{
308 config.add(specificT::configSection() + ".threadPrio", "", specificT::configSection() + ".threadPrio", argType::Required, specificT::configSection(), "threadPrio", false, "int", "The real-time priority of the shmimMonitor thread.");
309
310 config.add(specificT::configSection() + ".cpuset", "", specificT::configSection() + ".cpuset", argType::Required, specificT::configSection(), "cpuset", false, "string", "The cpuset for the shmimMonitor thread.");
311
312 config.add(specificT::configSection() + ".shmimName", "", specificT::configSection() + ".shmimName", argType::Required, specificT::configSection(), "shmimName", false, "string", "The name of the ImageStreamIO shared memory image. Will be used as /tmp/<shmimName>.im.shm.");
313
314 config.add(specificT::configSection() + ".getExistingFirst", "", specificT::configSection() + ".getExistingFirst", argType::Required, specificT::configSection(), "getExistingFirst", false, "bool", "If true an existing image is loaded. If false we wait for a new image.");
315
316 // Set this here to allow derived classes to set their own default before calling loadConfig
317 m_shmimName = derived().configName();
318
319 return 0;
320}
321
322template <class derivedT, class specificT>
323int shmimMonitor<derivedT, specificT>::loadConfig(mx::app::appConfigurator &config)
324{
325 config(m_smThreadPrio, specificT::configSection() + ".threadPrio");
326 config(m_smCpuset, specificT::configSection() + ".cpuset");
327 config(m_shmimName, specificT::configSection() + ".shmimName");
328 config(m_getExistingFirst, specificT::configSection() + ".getExistingFirst");
329
330 return 0;
331}
332
333template <class derivedT, class specificT>
335{
336 // Register the shmimName INDI property
337 m_indiP_shmimName = pcf::IndiProperty(pcf::IndiProperty::Text);
338 m_indiP_shmimName.setDevice(derived().configName());
339 m_indiP_shmimName.setName(specificT::indiPrefix() + "_shmimName");
340 m_indiP_shmimName.setPerm(pcf::IndiProperty::ReadOnly);
341 m_indiP_shmimName.setState(pcf::IndiProperty::Idle);
342 m_indiP_shmimName.add(pcf::IndiElement("name"));
343 m_indiP_shmimName["name"] = m_shmimName;
344
345 if (derived().registerIndiPropertyNew(m_indiP_shmimName, nullptr) < 0)
346 {
347 #ifndef SHMIMMONITOR_TEST_NOLOG
348 derivedT::template log<software_error>({__FILE__, __LINE__});
349 #endif
350 return -1;
351 }
352
353 // Register the frameSize INDI property
354 m_indiP_frameSize = pcf::IndiProperty(pcf::IndiProperty::Number);
355 m_indiP_frameSize.setDevice(derived().configName());
356 m_indiP_frameSize.setName(specificT::indiPrefix() + "_frameSize");
357 m_indiP_frameSize.setPerm(pcf::IndiProperty::ReadOnly);
358 m_indiP_frameSize.setState(pcf::IndiProperty::Idle);
359 m_indiP_frameSize.add(pcf::IndiElement("width"));
360 m_indiP_frameSize["width"] = 0;
361 m_indiP_frameSize.add(pcf::IndiElement("height"));
362 m_indiP_frameSize["height"] = 0;
363
364 if (derived().registerIndiPropertyNew(m_indiP_frameSize, nullptr) < 0)
365 {
366 #ifndef SHMIMMONITOR_TEST_NOLOG
367 derivedT::template log<software_error>({__FILE__, __LINE__});
368 #endif
369 return -1;
370 }
371
372 // Install empty signal handler for USR1, which is used to interrupt sleeps in the monitor threads.
373 struct sigaction act;
374 sigset_t set;
375
376 act.sa_sigaction = &sigUsr1Handler;
377 act.sa_flags = SA_SIGINFO;
378 sigemptyset(&set);
379 act.sa_mask = set;
380
381 errno = 0;
382 if (sigaction(SIGUSR1, &act, 0) < 0)
383 {
384 std::string logss = "Setting handler for SIGUSR1 failed. Errno says: ";
385 logss += strerror(errno);
386
387 derivedT::template log<software_error>({__FILE__, __LINE__, errno, 0, logss});
388
389 return -1;
390 }
391
392 if (derived().threadStart(m_smThread, m_smThreadInit, m_smThreadID, m_smThreadProp, m_smThreadPrio, m_smCpuset, specificT::configSection(), this, smThreadStart) < 0)
393 {
394 derivedT::template log<software_error>({__FILE__, __LINE__});
395 return -1;
396 }
397
398 return 0;
399}
400
401template <class derivedT, class specificT>
403{
404 // do a join check to see if other threads have exited.
405 if (pthread_tryjoin_np(m_smThread.native_handle(), 0) == 0)
406 {
407 derivedT::template log<software_error>({__FILE__, __LINE__, "shmimMonitor thread " + std::to_string(m_smThreadID) + " has exited"});
408
409 return -1;
410 }
411
412 return 0;
413}
414
415template <class derivedT, class specificT>
417{
418 if (m_smThread.joinable())
419 {
420 pthread_kill(m_smThread.native_handle(), SIGUSR1);
421 try
422 {
423 m_smThread.join(); // this will throw if it was already joined
424 }
425 catch (...)
426 {
427 }
428 }
429
430 return 0;
431}
432
433template <class derivedT, class specificT>
438
439template <class derivedT, class specificT>
441{
442 m_smThreadID = syscall(SYS_gettid);
443
444 // Wait for the thread starter to finish initializing this thread.
445 while (m_smThreadInit == true && derived().shutdown() == 0)
446 {
447 sleep(1);
448 }
449
450 bool opened = false;
451
452 // bool semgot = false;
453
454 while (derived().shutdown() == 0)
455 {
456 while ((derived().state() != stateCodes::OPERATING || m_shmimName == "") && !derived().shutdown() && !m_restart)
457 {
458 sleep(1);
459 }
460
461 if (derived().shutdown())
462 return;
463
464 /* Initialize ImageStreamIO
465 */
466 opened = false;
467 m_restart = false; // Set this up front, since we're about to restart.
468
469 int logged = 0;
470 while (!opened && !derived().m_shutdown && !m_restart && derived().state() == stateCodes::OPERATING)
471 {
472 // b/c ImageStreamIO prints every single time, and latest version don't support stopping it yet, and that isn't thread-safe-able anyway
473 // we do our own checks. This is the same code in ImageStreamIO_openIm...
474 int SM_fd;
475 char SM_fname[200];
476 ImageStreamIO_filename(SM_fname, sizeof(SM_fname), m_shmimName.c_str());
477 SM_fd = open(SM_fname, O_RDWR);
478 if (SM_fd == -1)
479 {
480 if (!logged)
481 derivedT::template log<text_log>("ImageStream " + m_shmimName + " not found (yet). Retrying . . .", logPrio::LOG_NOTICE);
482 logged = 1;
483 sleep(1); // be patient
484 continue;
485 }
486
487 // Found and opened, close it and then use ImageStreamIO
488 logged = 0;
489 close(SM_fd);
490
491 if (ImageStreamIO_openIm(&m_imageStream, m_shmimName.c_str()) == 0)
492 {
493 if (m_imageStream.md[0].sem <= m_semaphoreNumber) ///<\todo this isn't right--> isn't there a define in cacao to use?
494 {
495 ImageStreamIO_closeIm(&m_imageStream);
496 mx::sys::sleep(1); // We just need to wait for the server process to finish startup.
497 }
498 else
499 {
500 opened = true;
501 char SM_fname[200];
502 ImageStreamIO_filename(SM_fname, sizeof(SM_fname), m_shmimName.c_str());
503
504 struct stat buffer;
505 int rv = stat(SM_fname, &buffer);
506
507 if (rv != 0)
508 {
509 derivedT::template log<software_critical>({__FILE__, __LINE__, errno, "Could not get inode for " + m_shmimName + ". Source process will need to be restarted."});
510 ImageStreamIO_closeIm(&m_imageStream);
511 return;
512 }
513 m_inode = buffer.st_ino;
514 }
515 }
516 else
517 {
518 mx::sys::sleep(1); // be patient
519 }
520 }
521
522 if (m_restart)
523 continue; // this is kinda dumb. we just go around on restart, so why test in the while loop at all?
524
525 if (derived().state() != stateCodes::OPERATING)
526 continue;
527
528 if (derived().m_shutdown)
529 {
530 if (!opened)
531 return;
532
533 ImageStreamIO_closeIm(&m_imageStream);
534 return;
535 }
536
537 m_semaphoreNumber = ImageStreamIO_getsemwaitindex(&m_imageStream, m_semaphoreNumber); // ask for semaphore we had before
538
539 if (m_semaphoreNumber < 0)
540 {
541 derivedT::template log<software_critical>({__FILE__, __LINE__, "No valid semaphore found for " + m_shmimName + ". Source process will need to be restarted."});
542 return;
543 }
544
545 derivedT::template log<software_info>({__FILE__, __LINE__, "got semaphore index " + std::to_string(m_semaphoreNumber) + " for " + m_shmimName});
546
547 ImageStreamIO_semflush(&m_imageStream, m_semaphoreNumber);
548
549 sem_t *sem = m_imageStream.semptr[m_semaphoreNumber]; ///< The semaphore to monitor for new image data
550
551 m_dataType = m_imageStream.md[0].datatype;
552 m_typeSize = ImageStreamIO_typesize(m_dataType);
553 m_width = m_imageStream.md[0].size[0];
554 int dim = 1;
555 if (m_imageStream.md[0].naxis > 1)
556 {
557 m_height = m_imageStream.md[0].size[1];
558
559 if (m_imageStream.md[0].naxis > 2)
560 {
561 dim = 3;
562 m_depth = m_imageStream.md[0].size[2];
563 }
564 else
565 {
566 dim = 2;
567 m_depth = 1;
568 }
569 }
570 else
571 {
572 m_height = 1;
573 m_depth = 1;
574 }
575
576 if (derived().allocate(specificT()) < 0)
577 {
578 derivedT::template log<software_error>({__FILE__, __LINE__, "allocation failed"});
579 break;
580 }
581
582 uint8_t atype;
583 size_t snx, sny, snz;
584 uint64_t curr_image; // The current cnt1 index
585
586 if (m_getExistingFirst && !m_restart && derived().shutdown() == 0) // If true, we always get the existing image without waiting on the semaphore.
587 {
588 if (m_imageStream.md[0].size[2] > 0) ///\todo change to naxis?
589 {
590 curr_image = m_imageStream.md[0].cnt1;
591 }
592 else
593 {
594 curr_image = 0;
595 }
596
597 atype = m_imageStream.md[0].datatype;
598 snx = m_imageStream.md[0].size[0];
599
600 if (dim == 2)
601 {
602 sny = m_imageStream.md[0].size[1];
603 snz = 1;
604 }
605 else if (dim == 3)
606 {
607 sny = m_imageStream.md[0].size[1];
608 snz = m_imageStream.md[0].size[2];
609 }
610 else
611 {
612 sny = 1;
613 snz = 1;
614 }
615
616 if (atype != m_dataType || snx != m_width || sny != m_height || snz != m_depth)
617 {
618 continue; // exit the nearest while loop and get the new image setup.
619 }
620
621 char *curr_src = (char *)m_imageStream.array.raw + curr_image * m_width * m_height * m_typeSize;
622
623 if (derived().processImage(curr_src, specificT()) < 0)
624 {
625 derivedT::template log<software_error>({__FILE__, __LINE__});
626 }
627 }
628
629 // This is the main image grabbing loop.
630 while (derived().shutdown() == 0 && !m_restart && derived().state() == stateCodes::OPERATING)
631 {
632
633 timespec ts;
634
635 if (clock_gettime(CLOCK_REALTIME, &ts) < 0)
636 {
637 derivedT::template log<software_critical>({__FILE__, __LINE__, errno, 0, "clock_gettime"});
638 return;
639 }
640
641 ts.tv_sec += 1;
642
643 if (sem_timedwait(sem, &ts) == 0)
644 {
645 if (m_imageStream.md[0].size[2] > 0) ///\todo change to naxis?
646 {
647 curr_image = m_imageStream.md[0].cnt1;
648 }
649 else
650 curr_image = 0;
651
652 atype = m_imageStream.md[0].datatype;
653 snx = m_imageStream.md[0].size[0];
654
655 if (dim == 2)
656 {
657 sny = m_imageStream.md[0].size[1];
658 snz = 1;
659 }
660 else if (dim == 3)
661 {
662 sny = m_imageStream.md[0].size[1];
663 snz = m_imageStream.md[0].size[2];
664 }
665 else
666 {
667 sny = 1;
668 snz = 1;
669 }
670
671 if (atype != m_dataType || snx != m_width || sny != m_height || snz != m_depth)
672 {
673 break; // exit the nearest while loop and get the new image setup.
674 }
675
676 if (derived().shutdown() != 0 || m_restart || derived().state() != stateCodes::OPERATING)
677 break; // Check for exit signals
678
679 char *curr_src = (char *)m_imageStream.array.raw + curr_image * m_width * m_height * m_typeSize;
680
681 if (derived().processImage(curr_src, specificT()) < 0)
682 {
683 derivedT::template log<software_error>({__FILE__, __LINE__});
684 }
685 }
686 else
687 {
688 if (m_imageStream.md[0].sem <= 0)
689 break; // Indicates that the server has cleaned up.
690
691 // Check for why we timed out
692 if (errno == EINTR)
693 break; // This indicates signal interrupted us, time to restart or shutdown, loop will exit normally if flags set.
694
695 // ETIMEDOUT means we should check for deletion, and then wait more.
696 // Otherwise, report an error.
697 if (errno != ETIMEDOUT)
698 {
699 derivedT::template log<software_error>({__FILE__, __LINE__, errno, "sem_timedwait"});
700 break;
701 }
702
703 // Check if the file has disappeared.
704 int SM_fd;
705 char SM_fname[200];
706 ImageStreamIO_filename(SM_fname, sizeof(SM_fname), m_shmimName.c_str());
707 SM_fd = open(SM_fname, O_RDWR);
708 if (SM_fd == -1)
709 {
710 m_restart = true;
711 }
712 close(SM_fd);
713
714 // Check if the inode changed
715 struct stat buffer;
716 int rv = stat(SM_fname, &buffer);
717 if (rv != 0)
718 {
719 m_restart = true;
720 }
721
722 if (buffer.st_ino != m_inode)
723 {
724 m_restart = true;
725 }
726 }
727 }
728
729 //*******
730 // call derived().cleanup()
731 //*******
732
733 // opened == true if we can get to this
734 if (m_semaphoreNumber >= 0)
735 m_imageStream.semReadPID[m_semaphoreNumber] = 0; // release semaphore
736 ImageStreamIO_closeIm(&m_imageStream);
737 opened = false;
738
739 } // outer loop, will exit if m_shutdown==true
740
741 // derivedT::template log<software_error>({__FILE__,__LINE__, std::to_string(m_smThreadID) + " " + std::to_string(derived().shutdown() == 0)});
742
743 //*******
744 // call derived().cleanup()
745 //*******
746
747 if (opened)
748 {
749 ImageStreamIO_closeIm(&m_imageStream);
750 opened = false;
751 }
752}
753
754template <class derivedT, class specificT>
756{
757 if (!derived().m_indiDriver)
758 return 0;
759
760 indi::updateIfChanged(m_indiP_shmimName, "name", m_shmimName, derived().m_indiDriver);
761 indi::updateIfChanged(m_indiP_frameSize, "width", m_width, derived().m_indiDriver);
762 indi::updateIfChanged(m_indiP_frameSize, "height", m_height, derived().m_indiDriver);
763
764 return 0;
765}
766
767/// Call shmimMonitorT::setupConfig with error checking for shmimMonitor
768/**
769 * \param cfig the application configurator
770 */
771#define SHMIMMONITOR_SETUP_CONFIG( cfig ) \
772 if(shmimMonitorT::setupConfig(cfig) < 0) \
773 { \
774 log<software_error>({__FILE__, __LINE__, "Error from shmimMonitorT::setupConfig"}); \
775 m_shutdown = true; \
776 return; \
777 }
778
779/// Call shmimMonitorT::setupConfig with error checking for a typedef-ed shmimMonitor
780/**
781 * \param SHMIMMONITORT is the typedef-ed name of the shmimMonitor class, e.g. darkShmimMonitorT.
782 * \param cfig the application configurator
783 */
784#define SHMIMMONITORT_SETUP_CONFIG( SHMIMMONITORT, cfig ) \
785 if(SHMIMMONITORT::setupConfig(cfig) < 0) \
786 { \
787 log<software_error>({__FILE__, __LINE__, "Error from " #SHMIMMONITORT "::setupConfig"}); \
788 m_shutdown = true; \
789 return; \
790 }
791
792/// Call shmimMonitorT::loadConfig with error checking for shmimMonitor
793/** This must be inside a function that returns int, e.g. the standard loadConfigImpl.
794 * \param cfig the application configurator
795 */
796#define SHMIMMONITOR_LOAD_CONFIG( cfig ) \
797 if(shmimMonitorT::loadConfig(cfig) < 0) \
798 { \
799 return log<software_error,-1>({__FILE__, __LINE__, "Error from shmimMonitorT::loadConfig"}); \
800 }
801
802/// Call shmimMonitorT::loadConfig with error checking for a typedef-ed shmimMonitor
803/** This must be inside a function that returns int, e.g. the standard loadConfigImpl.
804 * \param SHMIMMONITORT is the typedef-ed name of the shmimMonitor class, e.g. darkShmimMonitorT.
805 * \param cfig the application configurator
806 */
807#define SHMIMMONITORT_LOAD_CONFIG( SHMIMMONITORT, cfig ) \
808 if(SHMIMMONITORT::loadConfig(cfig) < 0) \
809 { \
810 return log<software_error,-1>({__FILE__, __LINE__, "Error from " #SHMIMMONITORT "::loadConfig"}); \
811 }
812
813/// Call shmimMonitorT::appStartup with error checking for shmimMonitor
814#define SHMIMMONITOR_APP_STARTUP \
815 if(shmimMonitorT::appStartup() < 0) \
816 { \
817 return log<software_error,-1>({__FILE__, __LINE__, "Error from shmimMonitorT::appStartup"}); \
818 }
819
820/// Call shmimMonitorT::appStartup with error checking for a typedef-ed shmimMonitor
821/**
822 * \param SHMIMMONITORT is the typedef-ed name of the shmimMonitor class, e.g. darkShmimMonitorT.
823 */
824#define SHMIMMONITORT_APP_STARTUP( SHMIMMONITORT ) \
825 if(SHMIMMONITORT::appStartup() < 0) \
826 { \
827 return log<software_error,-1>({__FILE__, __LINE__, "Error from " #SHMIMMONITORT "::appStartup"}); \
828 }
829
830/// Call shmimMonitorT::appLogic with error checking for shmimMonitor
831#define SHMIMMONITOR_APP_LOGIC \
832 if(shmimMonitorT::appLogic() < 0) \
833 { \
834 return log<software_error,-1>({__FILE__, __LINE__, "Error from shmimMonitorT::appLogic"}); \
835 }
836
837/// Call shmimMonitorT::appLogic with error checking for a typedef-ed shmimMonitor
838/**
839 * \param SHMIMMONITORT is the typedef-ed name of the shmimMonitor class, e.g. darkShmimMonitorT.
840 */
841#define SHMIMMONITORT_APP_LOGIC( SHMIMMONITORT ) \
842 if(SHMIMMONITORT::appLogic() < 0) \
843 { \
844 return log<software_error,-1>({__FILE__, __LINE__, "Error from " #SHMIMMONITORT "::appLogic"}); \
845 }
846
847/// Call shmimMonitorT::updateINDI with error checking for shmimMonitor
848#define SHMIMMONITOR_UPDATE_INDI \
849 if(shmimMonitorT::updateINDI() < 0) \
850 { \
851 return log<software_error,-1>({__FILE__, __LINE__, "Error from shmimMonitorT::updateINDI"}); \
852 }
853
854/// Call shmimMonitorT::updateINDI with error checking for a typedef-ed shmimMonitor
855/**
856 * \param SHMIMMONITORT is the typedef-ed name of the shmimMonitor class, e.g. darkShmimMonitorT.
857 */
858#define SHMIMMONITORT_UPDATE_INDI( SHMIMMONITORT ) \
859 if(SHMIMMONITORT::updateINDI() < 0) \
860 { \
861 return log<software_error,-1>({__FILE__, __LINE__, "Error from " #SHMIMMONITORT "::updateINDI"}); \
862 }
863
864/// Call shmimMonitorT::appShutdown with error checking for shmimMonitor
865#define SHMIMMONITOR_APP_SHUTDOWN \
866 if(shmimMonitorT::appShutdown() < 0) \
867 { \
868 return log<software_error,-1>({__FILE__, __LINE__, "Error from shmimMonitorT::appShutdown"}); \
869 }
870
871/// Call shmimMonitorT::appShutodwn with error checking for a typedef-ed shmimMonitor
872/**
873 * \param SHMIMMONITORT is the typedef-ed name of the shmimMonitor class, e.g. darkShmimMonitorT.
874 */
875#define SHMIMMONITORT_APP_SHUTDOWN( SHMIMMONITORT ) \
876 if(SHMIMMONITORT::appShutdown() < 0) \
877 { \
878 return log<software_error,-1>({__FILE__, __LINE__, "Error from " #SHMIMMONITORT "::appShutdown"}); \
879 }
880
881} // namespace dev
882} // namespace app
883} // namespace MagAOX
884#endif
bool m_smThreadInit
Synchronizer for thread startup, to allow priority setting to finish.
uint32_t m_depth
The depth of the circular buffer in the stream.
int appStartup()
Startup function.
pid_t m_smThreadID
The s.m. thread PID.
void smThreadExec()
Execute the monitoring thread.
static shmimMonitor * m_selfMonitor
Static pointer to this (set in constructor). Used for getting out of the static SIGSEGV handler.
uint32_t m_width
The width of the images in the stream.
const uint8_t & dataType() const
const size_t & typeSize() const
int setupConfig(mx::app::appConfigurator &config)
Setup the configuration system.
int updateINDI()
Update the INDI properties for this device controller.
int appLogic()
Checks the shmimMonitor thread.
uint32_t m_height
The height of the images in the stream.
std::thread m_smThread
A separate thread for the actual monitoring.
int m_smThreadPrio
Priority of the shmimMonitor thread, should normally be > 00.
IMAGE m_imageStream
The ImageStreamIO shared memory buffer.
pcf::IndiProperty m_indiP_shmimName
Property used to report the shmim buffer name.
const uint32_t & depth() const
pcf::IndiProperty m_indiP_frameSize
Property used to report the current frame size.
static void smThreadStart(shmimMonitor *s)
Thread starter, called by MagAOXApp::threadStart on thread construction. Calls smThreadExec.
int m_semaphoreNumber
The image structure semaphore index.
std::string m_shmimName
The name of the shared memory image, is used in /tmp/<shmimName>.im.shm. Derived classes should set a...
const uint32_t & width() const
int appShutdown()
Shuts down the shmimMonitor thread.
const uint32_t & height() const
uint8_t m_dataType
The ImageStreamIO type code.
bool m_restart
Flag indicating tha the shared memory should be reinitialized.
int loadConfig(mx::app::appConfigurator &config)
load the configuration system results
pcf::IndiProperty m_smThreadProp
The property to hold the s.m. thread details.
size_t m_typeSize
The size of the type, in bytes. Result of sizeof.
ino_t m_inode
The inode of the image stream file.
const std::string & shmimName() const
std::string m_smCpuset
The cpuset to assign the shmimMonitor thread to. Ignored if empty (the default).
bool m_getExistingFirst
If set to true by derivedT, any existing image will be grabbed and sent to processImage before waitin...
@ OPERATING
The device is operating, other than homing.
void updateIfChanged(pcf::IndiProperty &p, const std::string &el, const T &newVal, indiDriverT *indiDriver, pcf::IndiProperty::PropertyStateType newState=pcf::IndiProperty::Ok)
Update the value of the INDI element, but only if it has changed.
Definition indiUtils.hpp:92
void sigUsr1Handler(int signum, siginfo_t *siginf, void *ucont)
Empty signal handler. SIGUSR1 is used to interrupt sleep in various threads.
Definition MagAOXApp.cpp:42
Definition dm.hpp:24
static std::string configSection()
static std::string indiPrefix()