API
 
Loading...
Searching...
No Matches
frameGrabber.hpp
Go to the documentation of this file.
1/** \file frameGrabber.hpp
2 * \brief The MagAO-X generic frame grabber.
3 *
4 * \author Jared R. Males (jaredmales@gmail.com)
5 *
6 * \ingroup app_files
7 */
8
9#ifndef frameGrabber_hpp
10#define frameGrabber_hpp
11
12#include <sys/syscall.h>
13
14#include <mx/sigproc/circularBuffer.hpp>
15#include <mx/math/vectorUtils.hpp>
16#include <mx/improc/imageUtils.hpp>
17
18#include <ImageStreamIO/ImageStruct.h>
19#include <ImageStreamIO/ImageStreamIO.h>
20
21#include "../../common/paths.hpp"
22
23
24namespace MagAOX
25{
26namespace app
27{
28namespace dev
29{
30
31
32
33
34/** MagAO-X generic frame grabber
35 *
36 *
37 * The derived class `derivedT` has the following requirements:
38 *
39 * - Must be derived from MagAOXApp<true>
40 *
41 * - Must contain the following friend declaration:
42 * \code
43 * friend class dev::frameGrabber<derivedT>; //replace derivedT
44 * \endcode
45 *
46 * - Must declare the following typedef:
47 * \code
48 * typedef dev::frameGrabber<derivedT> frameGrabberT; //replace derivedT
49 * \endcode
50 *
51 * - must expose the following interface
52 * \code
53 //Configures the camera for acquistion, must also set m_width, m_height, and m_dataType
54 //so that the shared memory can be allocated
55 int derivedT::configureAcquisition();
56
57 //Gets the frames-per-second readout rate
58 //used for the latency statistics
59 float derivedT::fps();
60
61 //Start acquisition.
62 int derivedT::startAcquisition();
63
64 //Acquires the data, and checks if it is valid.
65 //This should set m_currImageTimestamp to the image timestamp.
66 // returns 0 if valid, < 0 on error, > 0 on no data.
67 int derivedT::acquireAndCheckValid()
68
69 //Loads the acquired image into the stream, copying it to the appropriate member of m_imageStream->array.
70 //This could simply be a memcpy.
71 int derivedT::loadImageIntoStream(void * dest);
72
73 //Take any actions needed to reconfigure the system. Called if m_reconfig is set to true.
74 int derivedT::reconfig()
75 * \endcode
76 * Each of the above functions should return 0 on success, and -1 on an error (except fps).
77 * For `acquireAndCheckValid` >0 will indicate no data but not an error. In most cases,
78 * an appropriate state code, such as NOTCONNECTED, should be set as well.
79 *
80 * A static configuration variable must be defined in derivedT as
81 * \code
82 * static constexpr bool c_frameGrabber_flippable =true; //or: false
83 * \endcode
84 * which determines whether or not the images can be flipped programatically.
85 *
86 * Calls to this class's `setupConfig`, `loadConfig`, `appStartup`, `appLogic`, `updateINDI`, and `appShutdown`
87 * functions must be placed in the derived class's functions of the same name. For convenience the
88 * following macros are defined to provide error checking:
89 * \code
90 * FRAMEGRABBER_SETUP_CONFIG( cfig )
91 * FRAMEGRABBER_LOAD_CONFIG( cfig )
92 * FRAMEGRABBER_APP_STARTUP
93 * FRAMEGRABBER_APP_LOGIC
94 * FRAMEGRABBER_UPDATE_INDI
95 * FRAMEGRABBER_APP_SHUTDOWN
96 * \endcode
97 *
98 * \ingroup appdev
99 */
100template<class derivedT>
102{
103public:
105
106 typedef int32_t cbIndexT;
107
108protected:
109
110 /** \name Configurable Parameters
111 * @{
112 */
113 std::string m_shmimName {""}; ///< The name of the shared memory image, is used in `/tmp/<shmimName>.im.shm`. Derived classes should set a default.
114
115 int m_fgThreadPrio {2}; ///< Priority of the framegrabber thread, should normally be > 00.
116 std::string m_fgCpuset; ///< The cpuset to assign the framegrabber thread to. Not used if empty, the default.
117
118 uint32_t m_circBuffLength {1}; ///< Length of the circular buffer, in frames
119
120 cbIndexT m_latencyCircBuffMaxLength {100000}; ///< Maximum length of the latency measurement circular buffers
121
122 float m_latencyCircBuffMaxTime {5}; ///< Maximum time of the latency meaurement circular buffers
123
125
126 ///@}
127
129
130 uint32_t m_width {0}; ///< The width of the image, once deinterlaced etc.
131 uint32_t m_height {0}; ///< The height of the image, once deinterlaced etc.
132
133 uint8_t m_dataType{0}; ///< The ImageStreamIO type code.
134 size_t m_typeSize {0}; ///< The size of the type, in bytes. Result of sizeof.
135
136 int m_xbinning {0}; ///< The x-binning according to the framegrabber
137 int m_ybinning {0}; ///< The y-binning according to the framegrabber
138
139
140 timespec m_currImageTimestamp {0,0}; ///< The timestamp of the current image.
141
142 bool m_reconfig {false}; ///< Flag to set if a camera reconfiguration requires a framegrabber reset.
143
144 IMAGE * m_imageStream {nullptr}; ///< The ImageStreamIO shared memory buffer.
145
146 mx::sigproc::circularBufferIndex<timespec, cbIndexT> m_atimes;
147 mx::sigproc::circularBufferIndex<timespec, cbIndexT> m_wtimes;
148
149 std::vector<double> m_atimesD;
150 std::vector<double> m_wtimesD;
151 std::vector<double> m_watimesD;
152
153 timespec m_dummy_ts {0,0};
154 uint64_t m_dummy_cnt {0};
155 char m_dummy_c {0};
156
157 double m_mna;
158 double m_vara;
159 double m_mina;
160 double m_maxa;
161
162 double m_mnw;
163 double m_varw;
164 double m_minw;
165 double m_maxw;
166
167 double m_mnwa;
168 double m_varwa;
169
170
171
172
173public:
174
175 /// Setup the configuration system
176 /**
177 * This should be called in `derivedT::setupConfig` as
178 * \code
179 framegrabber<derivedT>::setupConfig(config);
180 \endcode
181 * with appropriate error checking.
182 */
183 int setupConfig(mx::app::appConfigurator & config /**< [out] the derived classes configurator*/);
184
185 /// load the configuration system results
186 /**
187 * This should be called in `derivedT::loadConfig` as
188 * \code
189 framegrabber<derivedT>::loadConfig(config);
190 \endcode
191 * with appropriate error checking.
192 */
193 int loadConfig(mx::app::appConfigurator & config /**< [in] the derived classes configurator*/);
194
195 /// Startup function
196 /** Starts the framegrabber thread
197 * This should be called in `derivedT::appStartup` as
198 * \code
199 framegrabber<derivedT>::appStartup();
200 \endcode
201 * with appropriate error checking.
202 *
203 * \returns 0 on success
204 * \returns -1 on error, which is logged.
205 */
207
208 /// Checks the framegrabber thread
209 /** This should be called in `derivedT::appLogic` as
210 * \code
211 framegrabber<derivedT>::appLogic();
212 \endcode
213 * with appropriate error checking.
214 *
215 * \returns 0 on success
216 * \returns -1 on error, which is logged.
217 */
218 int appLogic();
219
220 /// On power off, sets m_reconfig to true.
221 /** This should be called in `derivedT::onPowerOff` as
222 * \code
223 framegrabber<derivedT>::onPowerOff();
224 \endcode
225 * with appropriate error checking.
226 *
227 * \returns 0 on success
228 * \returns -1 on error, which is logged.
229 */
231
232 /// Shuts down the framegrabber thread
233 /** This should be called in `derivedT::appShutdown` as
234 * \code
235 framegrabber<derivedT>::appShutdown();
236 \endcode
237 * with appropriate error checking.
238 *
239 * \returns 0 on success
240 * \returns -1 on error, which is logged.
241 */
243
244protected:
245
246
247 /** \name Framegrabber Thread
248 * This thread actually manages the framegrabbing hardware
249 * @{
250 */
251
252 bool m_fgThreadInit {true}; ///< Synchronizer for thread startup, to allow priority setting to finish.
253
254 pid_t m_fgThreadID {0}; ///< The ID of the framegrabber thread.
255
256 pcf::IndiProperty m_fgThreadProp; ///< The property to hold the f.g. thread details.
257
258 std::thread m_fgThread; ///< A separate thread for the actual framegrabbings
259
260 ///Thread starter, called by MagAOXApp::threadStart on thread construction. Calls fgThreadExec.
261 static void fgThreadStart( frameGrabber * o /**< [in] a pointer to a frameGrabber instance (normally this) */);
262
263 /// Execute framegrabbing.
265
266
267 ///@}
268
269 void * loadImageIntoStreamCopy( void * dest,
270 void * src,
271 size_t width,
272 size_t height,
273 size_t szof
274 );
275
276
277 /** \name INDI
278 *
279 *@{
280 */
281protected:
282 //declare our properties
283
284 pcf::IndiProperty m_indiP_shmimName; ///< Property used to report the shmim buffer name
285
286 pcf::IndiProperty m_indiP_frameSize; ///< Property used to report the current frame size
287
288 pcf::IndiProperty m_indiP_timing;
289public:
290
291 /// Update the INDI properties for this device controller
292 /** You should call this once per main loop.
293 * It is not called automatically.
294 *
295 * \returns 0 on success.
296 * \returns -1 on error.
297 */
299
300 ///@}
301
302 /** \name Telemeter Interface
303 * @{
304 */
305
306 int recordFGTimings( bool force = false );
307
308 /// @}
309
310private:
311 derivedT & derived()
312 {
313 return *static_cast<derivedT *>(this);
314 }
315};
316
317template<class derivedT>
318int frameGrabber<derivedT>::setupConfig(mx::app::appConfigurator & config)
319{
320 config.add("framegrabber.threadPrio", "", "framegrabber.threadPrio", argType::Required, "framegrabber", "threadPrio", false, "int", "The real-time priority of the framegrabber thread.");
321
322 config.add("framegrabber.cpuset", "", "framegrabber.cpuset", argType::Required, "framegrabber", "cpuset", false, "string", "The cpuset to assign the framegrabber thread to.");
323
324 config.add("framegrabber.shmimName", "", "framegrabber.shmimName", argType::Required, "framegrabber", "shmimName", false, "string", "The name of the ImageStreamIO shared memory image. Will be used as /milk/shm/<shmimName>.im.shm.");
325
326 config.add("framegrabber.circBuffLength", "", "framegrabber.circBuffLength", argType::Required, "framegrabber", "circBuffLength", false, "size_t", "The length of the circular buffer. Sets m_circBuffLength, default is 1.");
327
328 config.add("framegrabber.latencyTime", "", "framegrabber.latencyTime", argType::Required, "framegrabber", "latencyTime", false, "float", "The maximum length of time to measure latency timings. Sets m_latencyCircBuffMaxTime, default is 5.");
329
330 config.add("framegrabber.latencySize", "", "framegrabber.latencySize", argType::Required, "framegrabber", "latencySize", false, "float", "The maximum length of the buffer used to measure latency timings. Sets m_latencyCircBuffMaxLength, default is 3600.");
331
332
333 if(derivedT::c_frameGrabber_flippable)
334 {
335 config.add("framegrabber.defaultFlip", "", "framegrabber.defaultFlip", argType::Required, "framegrabber", "defaultFlip", false, "string", "The default flip of the image. Options are flipNone, flipUD, flipLR, flipUDLR. The default is flipNone.");
336 }
337
338 return 0;
339}
340
341template<class derivedT>
342int frameGrabber<derivedT>::loadConfig(mx::app::appConfigurator & config)
343{
344 config(m_fgThreadPrio, "framegrabber.threadPrio");
345 config(m_fgCpuset, "framegrabber.cpuset");
346 if(m_shmimName == "") m_shmimName = derived().configName();
347 config(m_shmimName, "framegrabber.shmimName");
348
349 config(m_circBuffLength, "framegrabber.circBuffLength");
350
351 if(m_circBuffLength < 1)
352 {
353 m_circBuffLength = 1;
354 derivedT::template log<text_log>("circBuffLength set to 1");
355 }
356
357 config(m_latencyCircBuffMaxTime, "framegrabber.latencyTime");
358 if(m_latencyCircBuffMaxTime < 0)
359 {
360 m_latencyCircBuffMaxTime = 0;
361 derivedT::template log<text_log>("latencyTime set to 0 (off)");
362 }
363
364 config(m_latencyCircBuffMaxLength, "framegrabber.latencySize");
365
366
367 if(derivedT::c_frameGrabber_flippable)
368 {
369 std::string flip = "flipNone";
370 config(flip, "framegrabber.defaultFlip");
371 if(flip == "flipNone")
372 {
373 m_defaultFlip = fgFlipNone;
374 }
375 else if(flip == "flipUD")
376 {
377 m_defaultFlip = fgFlipUD;
378 }
379 else if(flip == "flipLR")
380 {
381 m_defaultFlip = fgFlipLR;
382 }
383 else if(flip == "flipUDLR")
384 {
385 m_defaultFlip = fgFlipUDLR;
386 }
387 else
388 {
389 derivedT::template log<text_log>({std::string("invalid framegrabber flip specification (") + flip + "), setting flipNone"}, logPrio::LOG_ERROR);
390 m_defaultFlip = fgFlipNone;
391 }
392 }
393
394 return 0;
395}
396
397
398template<class derivedT>
400{
401 //Register the shmimName INDI property
402 m_indiP_shmimName = pcf::IndiProperty(pcf::IndiProperty::Text);
403 m_indiP_shmimName.setDevice(derived().configName());
404 m_indiP_shmimName.setName("fg_shmimName");
405 m_indiP_shmimName.setPerm(pcf::IndiProperty::ReadOnly);
406 m_indiP_shmimName.setState(pcf::IndiProperty::Idle);
407 m_indiP_shmimName.add(pcf::IndiElement("name"));
408 m_indiP_shmimName["name"] = m_shmimName;
409
410 if( derived().registerIndiPropertyNew( m_indiP_shmimName, nullptr) < 0)
411 {
412 #ifndef FRAMEGRABBER_TEST_NOLOG
413 derivedT::template log<software_error>({__FILE__,__LINE__});
414 #endif
415 return -1;
416 }
417
418 //Register the frameSize INDI property
419 m_indiP_frameSize = pcf::IndiProperty(pcf::IndiProperty::Number);
420 m_indiP_frameSize.setDevice(derived().configName());
421 m_indiP_frameSize.setName("fg_frameSize");
422 m_indiP_frameSize.setPerm(pcf::IndiProperty::ReadOnly);
423 m_indiP_frameSize.setState(pcf::IndiProperty::Idle);
424 m_indiP_frameSize.add(pcf::IndiElement("width"));
425 m_indiP_frameSize["width"] = 0;
426 m_indiP_frameSize.add(pcf::IndiElement("height"));
427 m_indiP_frameSize["height"] = 0;
428
429 if( derived().registerIndiPropertyNew( m_indiP_frameSize, nullptr) < 0)
430 {
431 #ifndef FRAMEGRABBER_TEST_NOLOG
432 derivedT::template log<software_error>({__FILE__,__LINE__});
433 #endif
434 return -1;
435 }
436
437 //Register the timing INDI property
438 derived().createROIndiNumber( m_indiP_timing, "fg_timing");
439 m_indiP_timing.add(pcf::IndiElement("acq_fps"));
440 m_indiP_timing.add(pcf::IndiElement("acq_min"));
441 m_indiP_timing.add(pcf::IndiElement("acq_max"));
442 m_indiP_timing.add(pcf::IndiElement("acq_jitter"));
443 m_indiP_timing.add(pcf::IndiElement("write_fps"));
444 m_indiP_timing.add(pcf::IndiElement("write_min"));
445 m_indiP_timing.add(pcf::IndiElement("write_max"));
446 m_indiP_timing.add(pcf::IndiElement("write_jitter"));
447 m_indiP_timing.add(pcf::IndiElement("delta_aw"));
448 m_indiP_timing.add(pcf::IndiElement("delta_aw_jitter"));
449
450 if( derived().registerIndiPropertyReadOnly( m_indiP_timing ) < 0)
451 {
452 #ifndef STDCAMERA_TEST_NOLOG
453 derivedT::template log<software_error>({__FILE__,__LINE__});
454 #endif
455 return -1;
456 }
457
458 //Start the f.g. thread
459 if(derived().threadStart( m_fgThread, m_fgThreadInit, m_fgThreadID, m_fgThreadProp, m_fgThreadPrio, m_fgCpuset, "framegrabber", this, fgThreadStart) < 0)
460 {
461 derivedT::template log<software_error, -1>({__FILE__, __LINE__});
462 return -1;
463 }
464
465 return 0;
466
467}
468
469template<class derivedT>
471{
472 //do a join check to see if other threads have exited.
473 if(pthread_tryjoin_np(m_fgThread.native_handle(),0) == 0)
474 {
475 derivedT::template log<software_error>({__FILE__, __LINE__, "framegrabber thread has exited"});
476
477 return -1;
478 }
479
480 if( derived().state() == stateCodes::OPERATING && m_atimes.size() > 0 && derived().fps() > 0)
481 {
482 if(m_atimes.size() >= m_atimes.maxEntries() )
483 {
484
485 cbIndexT latTime = m_latencyCircBuffMaxTime*derived().fps();
486 if(latTime > m_atimes.maxEntries())
487 {
488 latTime = m_atimes.maxEntries();
489 }
490
491 m_atimesD.resize(latTime-1);
492 m_wtimesD.resize(latTime-1);
493 m_watimesD.resize(latTime-1);
494
495 cbIndexT refEntry = m_atimes.latest();
496
497 if(refEntry >= latTime)
498 {
499 refEntry -= latTime;
500 }
501 else
502 {
503 refEntry = m_atimes.maxEntries() + refEntry - latTime;
504 }
505
506 timespec ts = m_atimes.at(refEntry, 0);
507 double a0 = ts.tv_sec + ((double) ts.tv_nsec)/1e9;
508
509 ts = m_wtimes.at(refEntry, 0);
510 double w0 = ts.tv_sec + ((double) ts.tv_nsec)/1e9;
511
512 double mina = 1e9;
513 double maxa = -1e9;
514 double minw = 1e9;
515 double maxw = -1e9;
516
517 for(size_t n=1; n <= m_atimesD.size(); ++n)
518 {
519 ts = m_atimes.at(refEntry, n);
520 double a = ts.tv_sec + ((double) ts.tv_nsec)/1e9;
521
522 ts = m_wtimes.at(refEntry, n);
523 double w = ts.tv_sec + ((double) ts.tv_nsec)/1e9;
524
525 m_atimesD[n-1] = a - a0;
526 m_wtimesD[n-1] = w - w0;
527 m_watimesD[n-1] = w - a;
528 a0 = a;
529 w0 = w;
530
531 if(m_atimesD[n-1] < mina)
532 {
533 mina = m_atimesD[n-1];
534 }
535
536 if(m_atimesD[n-1] > maxa)
537 {
538 maxa = m_atimesD[n-1];
539 }
540
541 if(m_wtimesD[n-1] < minw)
542 {
543 minw = m_wtimesD[n-1];
544 }
545
546 if(m_wtimesD[n-1] > maxw)
547 {
548 maxw = m_wtimesD[n-1];
549 }
550
551 if(m_wtimesD[n-1] < 0)
552 {
553 std::cerr << "negative wtime: " << m_wtimesD[n-1] << ' ' << n << ' ' << m_atimesD.size() << ' ' << refEntry << '\n';
554 return derivedT::template log<software_error,-1>({__FILE__, __LINE__, "negative write time. latency circ buff is not long enought"});
555 }
556 }
557
558 m_mna = mx::math::vectorMean(m_atimesD);
559 m_vara = mx::math::vectorVariance(m_atimesD, m_mna);
560 m_mina = mina;
561 m_maxa = maxa;
562
563 m_mnw = mx::math::vectorMean(m_wtimesD);
564 m_varw = mx::math::vectorVariance(m_wtimesD, m_mnw);
565 m_minw = minw;
566 m_maxw = maxw;
567
568 m_mnwa = mx::math::vectorMean(m_watimesD);
569 m_varwa = mx::math::vectorVariance(m_watimesD, m_mnwa);
570
571 recordFGTimings();
572 }
573 else
574 {
575 m_mna = 0;
576 m_vara = 0;
577 m_mina = 0;
578 m_maxa = 0;
579 m_mnw = 0;
580 m_varw = 0;
581 m_minw = 0;
582 m_maxw = 0;
583 m_mnwa = 0;
584 m_varwa = 0;
585 }
586 }
587 else
588 {
589 m_mna = 0;
590 m_vara = 0;
591 m_mina = 0;
592 m_maxa = 0;
593 m_mnw = 0;
594 m_varw = 0;
595 m_minw = 0;
596 m_maxw = 0;
597 m_mnwa = 0;
598 m_varwa = 0;
599 }
600
601 return 0;
602
603}
604
605template<class derivedT>
607{
608 m_mna = 0;
609 m_vara = 0;
610 m_mina = 0;
611 m_maxa = 0;
612 m_mnw = 0;
613 m_varw = 0;
614 m_minw = 0;
615 m_maxw = 0;
616 m_mnwa = 0;
617 m_varwa = 0;
618
619 m_width = 0;
620 m_height = 0;
621
622 updateINDI();
623
624 m_reconfig = true;
625
626
627 return 0;
628}
629
630
631template<class derivedT>
633{
634 if(m_fgThread.joinable())
635 {
636 try
637 {
638 m_fgThread.join(); //this will throw if it was already joined
639 }
640 catch(...)
641 {
642 }
643 }
644
645
646
647 return 0;
648}
649
650
651
652template<class derivedT>
657
658
659template<class derivedT>
661{
662 //Get the thread PID immediately so the caller can return.
663 m_fgThreadID = syscall(SYS_gettid);
664
665 //timespec writestart;
666
667 //Wait fpr the thread starter to finish initializing this thread.
668 while(m_fgThreadInit == true && derived().shutdown() == 0)
669 {
670 sleep(1);
671 }
672
673 uint32_t imsize[3] = {0,0,0};
674 std::string shmimName;
675
676 while(derived().shutdown() == 0)
677 {
678 ///\todo this ought to wait until OPERATING, using READY as a sign of "not integrating"
679 while(!derived().shutdown() && (!( derived().state() == stateCodes::READY || derived().state() == stateCodes::OPERATING) || derived().powerState() <= 0 ) )
680 {
681 sleep(1);
682 }
683
684 if(derived().shutdown())
685 {
686 continue;
687 }
688 else
689 {
690 //At the end of this, must have m_width, m_height, m_dataType set, and derived()->fps must be valid.
691 if(derived().configureAcquisition() < 0) continue;
692
693 if(m_latencyCircBuffMaxLength == 0 || m_latencyCircBuffMaxTime == 0 || derived().fps() <= 0)
694 {
695 m_atimes.maxEntries(0);
696 m_wtimes.maxEntries(0);
697 }
698 else
699 {
700 //Set up the latency circ. buffs
701 cbIndexT cbSz = 2*m_latencyCircBuffMaxTime * derived().fps();
702 if(cbSz > m_latencyCircBuffMaxLength) cbSz = m_latencyCircBuffMaxLength;
703 if(cbSz < 3) cbSz = 3; //Make variance meaningful
704
705 m_atimes.maxEntries(cbSz);
706 m_wtimes.maxEntries(cbSz);
707 }
708
709 m_typeSize = ImageStreamIO_typesize(m_dataType);
710
711
712 //Here we resolve currentFlip somehow.
713 m_currentFlip = m_defaultFlip;
714 }
715
716 /* Initialize ImageStreamIO
717 */
718 if(m_shmimName == "") m_shmimName = derived().configName();
719
720 if(m_width != imsize[0] || m_height != imsize[1] || static_cast<uint32_t>(m_circBuffLength) != imsize[2] || m_shmimName != shmimName || m_imageStream == nullptr)
721 {
722 if(m_imageStream != nullptr)
723 {
724 ImageStreamIO_destroyIm(m_imageStream);
725 free(m_imageStream);
726 }
727
728 m_imageStream = reinterpret_cast<IMAGE *>(malloc(sizeof(IMAGE)));
729
730 imsize[0] = m_width;
731 imsize[1] = m_height;
732 imsize[2] = m_circBuffLength;
733 shmimName = m_shmimName;
734
735 std::cerr << "Creating: " << m_shmimName << " " << m_width << " " << m_height << " " << m_circBuffLength << "\n";
736
737 ImageStreamIO_createIm_gpu(m_imageStream, m_shmimName.c_str(), 3, imsize, m_dataType, -1, 1, IMAGE_NB_SEMAPHORE, 0, CIRCULAR_BUFFER | ZAXIS_TEMPORAL, 0);
738
739 m_imageStream->md->cnt1 = m_circBuffLength - 1;
740 }
741
742 //This completes the reconfiguration.
743 m_reconfig = false;
744
745 if(derived().startAcquisition() < 0) continue;
746
747 uint64_t next_cnt1 = 0;
748 char * next_dest = reinterpret_cast<char *>(m_imageStream->array.raw);
749 timespec * next_wtimearr = &m_imageStream->writetimearray[0];
750 timespec * next_atimearr = &m_imageStream->atimearray[0];
751 uint64_t * next_cntarr = &m_imageStream->cntarray[0];
752
753 //This is the main image grabbing loop.
754 while(!derived().shutdown() && !m_reconfig && derived().powerState() > 0)
755 {
756 //==================
757 //Get next image, process validity.
758 //====================
759 int isValid = derived().acquireAndCheckValid();
760 if(isValid != 0)
761 {
762 if( isValid < 0)
763 {
764 break;
765 }
766 else
767 {
768 continue;
769 }
770 }
771
772 //Ok, no timeout, so we process the image and publish it.
773 m_imageStream->md->write=1;
774
775 //Set the time of last write
776 //clock_gettime(CLOCK_REALTIME, &writestart);
777
778 if(derived().loadImageIntoStream(next_dest) < 0)
779 {
780 break;
781 }
782
783 //Set the time of last write
784 //clock_gettime(CLOCK_REALTIME, &m_imageStream->md->writetime);
785 if(clock_gettime(CLOCK_REALTIME, &m_imageStream->md->writetime) < 0)
786 {
787 derivedT::template log<software_critical>({__FILE__, __LINE__, errno, 0, "clock_gettime"});
788 }
789
790 //Set the image acquisition timestamp
791 m_imageStream->md->atime = m_currImageTimestamp;
792
793 //Update cnt1
794 m_imageStream->md->cnt1 = next_cnt1;
795
796 //Update cnt0
797 m_imageStream->md->cnt0++;
798
799 *next_wtimearr = m_imageStream->md->writetime;
800 *next_atimearr = m_currImageTimestamp;
801 *next_cntarr = m_imageStream->md->cnt0;
802
803 //And post
804 m_imageStream->md->write=0;
805 ImageStreamIO_sempost(m_imageStream,-1);
806
807 //Update the latency circ. buffs
808 if(m_atimes.maxEntries() > 0)
809 {
810 m_atimes.nextEntry(m_imageStream->md->atime);
811 m_wtimes.nextEntry(m_imageStream->md->writetime);
812 }
813
814 //Now we increment pointers outside the time-critical part of the loop.
815 next_cnt1 = m_imageStream->md->cnt1+1;
816 if(next_cnt1 >= m_circBuffLength) next_cnt1 = 0;
817
818 next_dest = reinterpret_cast<char *>(m_imageStream->array.raw) + next_cnt1*m_width*m_height*m_typeSize;
819 next_wtimearr = &m_imageStream->writetimearray[next_cnt1];
820 next_atimearr = &m_imageStream->atimearray[next_cnt1];
821 next_cntarr = &m_imageStream->cntarray[next_cnt1];
822
823 //Touch them to make sure we move
824 m_dummy_c = next_dest[0];
825 m_dummy_ts.tv_sec = next_wtimearr[0].tv_sec + next_atimearr[0].tv_sec;
826 m_dummy_cnt = next_cntarr[0];
827 }
828
829 if(m_reconfig && !derived().shutdown())
830 {
831 derived().reconfig();
832 }
833
834 } //outer loop, will exit if m_shutdown==true
835
836 if(m_imageStream != nullptr)
837 {
838 ImageStreamIO_destroyIm( m_imageStream );
839 free(m_imageStream);
840 m_imageStream = nullptr;
841 }
842}
843
844template<class derivedT>
846 void * src,
847 size_t width,
848 size_t height,
849 size_t szof
850 )
851{
852 if(!derivedT::c_frameGrabber_flippable)
853 {
854 return memcpy(dest, src, width*height*szof);
855 }
856 else
857 {
858 switch(m_currentFlip)
859 {
860 case fgFlipNone:
861 return mx::improc::imcpy(dest, src, width, height, szof);
862 case fgFlipUD:
863 return mx::improc::imcpy_flipUD(dest, src, width, height, szof);
864 case fgFlipLR:
865 return mx::improc::imcpy_flipLR(dest, src, width, height, szof);
866 case fgFlipUDLR:
867 return mx::improc::imcpy_flipUDLR(dest, src, width, height, szof);
868 default:
869 return nullptr;
870 }
871 }
872}
873
874
875
876template<class derivedT>
878{
879 if( !derived().m_indiDriver ) return 0;
880
881 indi::updateIfChanged(m_indiP_shmimName, "name", m_shmimName, derived().m_indiDriver);
882 indi::updateIfChanged(m_indiP_frameSize, "width", m_width, derived().m_indiDriver);
883 indi::updateIfChanged(m_indiP_frameSize, "height", m_height, derived().m_indiDriver);
884
885 double fpsa = 0;
886 double fpsw = 0;
887 if(m_mna != 0 ) fpsa = 1.0/m_mna;
888 if(m_mnw != 0 ) fpsw = 1.0/m_mnw;
889
890 indi::updateIfChanged<double>(m_indiP_timing, {"acq_fps","acq_min", "acq_max", "acq_jitter","write_fps","write_min", "write_max","write_jitter","delta_aw","delta_aw_jitter"},
891 {fpsa, m_mina, m_maxa, sqrt(m_vara), fpsw, m_minw, m_maxw, sqrt(m_varw), m_mnwa, sqrt(m_varwa)},derived().m_indiDriver);
892
893 return 0;
894}
895
896template<class derivedT>
898{
899 static double last_mna = 0;
900 static double last_vara = 0;
901
902 static double last_mnw = 0;
903 static double last_varw = 0;
904
905 static double last_mnwa = 0;
906 static double last_varwa = 0;
907
908 if(force || m_mna != last_mna || m_vara != last_vara ||
909 m_mnw != last_mnw || m_varw != last_varw ||
910 m_mnwa != last_mnwa || m_varwa != last_varwa )
911 {
912 derived().template telem<telem_fgtimings>({m_mna, sqrt(m_vara), m_mnw, sqrt(m_varw), m_mnwa, sqrt(m_varwa)});
913
914 last_mna = m_mna;
915 last_vara = m_vara;
916 last_mnw = m_mnw;
917 last_varw = m_varw;
918 last_mnwa = m_mnwa;
919 last_varwa = m_varwa;
920 }
921
922 return 0;
923
924}
925
926/// Call frameGrabberT::setupConfig with error checking for frameGrabber
927/**
928 * \param cfig the application configurator
929 */
930#define FRAMEGRABBER_SETUP_CONFIG( cfig ) \
931 if(frameGrabberT::setupConfig(cfig) < 0) \
932 { \
933 log<software_error>({__FILE__, __LINE__, "Error from frameGrabberT::setupConfig"}); \
934 m_shutdown = true; \
935 return; \
936 }
937
938/// Call frameGrabberT::loadConfig with error checking for frameGrabber
939/** This must be inside a function that returns int, e.g. the standard loadConfigImpl.
940 * \param cfig the application configurator
941 */
942#define FRAMEGRABBER_LOAD_CONFIG( cfig ) \
943 if(frameGrabberT::loadConfig(cfig) < 0) \
944 { \
945 return log<software_error,-1>({__FILE__, __LINE__, "Error from frameGrabberT::loadConfig"}); \
946 }
947
948/// Call frameGrabberT::appStartup with error checking for frameGrabber
949#define FRAMEGRABBER_APP_STARTUP \
950 if(frameGrabberT::appStartup() < 0) \
951 { \
952 return log<software_error,-1>({__FILE__, __LINE__, "Error from frameGrabberT::appStartup"}); \
953 }
954
955/// Call frameGrabberT::appLogic with error checking for frameGrabber
956#define FRAMEGRABBER_APP_LOGIC \
957 if(frameGrabberT::appLogic() < 0) \
958 { \
959 return log<software_error,-1>({__FILE__, __LINE__, "Error from frameGrabberT::appLogic"}); \
960 }
961
962/// Call frameGrabberT::updateINDI with error checking for frameGrabber
963#define FRAMEGRABBER_UPDATE_INDI \
964 if(frameGrabberT::updateINDI() < 0) \
965 { \
966 return log<software_error,-1>({__FILE__, __LINE__, "Error from frameGrabberT::updateINDI"}); \
967 }
968
969/// Call frameGrabberT::appShutdown with error checking for frameGrabber
970#define FRAMEGRABBER_APP_SHUTDOWN \
971 if(frameGrabberT::appShutdown() < 0) \
972 { \
973 return log<software_error,-1>({__FILE__, __LINE__, "Error from frameGrabberT::appShutdown"}); \
974 }
975
976} //namespace dev
977} //namespace app
978} //namespace MagAOX
979#endif
int m_xbinning
The x-binning according to the framegrabber.
timespec m_currImageTimestamp
The timestamp of the current image.
uint32_t m_width
The width of the image, once deinterlaced etc.
int appShutdown()
Shuts down the framegrabber thread.
mx::sigproc::circularBufferIndex< timespec, cbIndexT > m_wtimes
void * loadImageIntoStreamCopy(void *dest, void *src, size_t width, size_t height, size_t szof)
int recordFGTimings(bool force=false)
pcf::IndiProperty m_fgThreadProp
The property to hold the f.g. thread details.
void fgThreadExec()
Execute framegrabbing.
int loadConfig(mx::app::appConfigurator &config)
load the configuration system results
bool m_fgThreadInit
Synchronizer for thread startup, to allow priority setting to finish.
std::string m_fgCpuset
The cpuset to assign the framegrabber thread to. Not used if empty, the default.
static void fgThreadStart(frameGrabber *o)
Thread starter, called by MagAOXApp::threadStart on thread construction. Calls fgThreadExec.
int m_ybinning
The y-binning according to the framegrabber.
std::vector< double > m_watimesD
size_t m_typeSize
The size of the type, in bytes. Result of sizeof.
int updateINDI()
Update the INDI properties for this device controller.
std::vector< double > m_wtimesD
int onPowerOff()
On power off, sets m_reconfig to true.
pcf::IndiProperty m_indiP_shmimName
Property used to report the shmim buffer name.
std::string m_shmimName
The name of the shared memory image, is used in /tmp/<shmimName>.im.shm. Derived classes should set a...
uint8_t m_dataType
The ImageStreamIO type code.
bool m_reconfig
Flag to set if a camera reconfiguration requires a framegrabber reset.
std::thread m_fgThread
A separate thread for the actual framegrabbings.
pid_t m_fgThreadID
The ID of the framegrabber thread.
int appLogic()
Checks the framegrabber thread.
int m_fgThreadPrio
Priority of the framegrabber thread, should normally be > 00.
cbIndexT m_latencyCircBuffMaxLength
Maximum length of the latency measurement circular buffers.
mx::sigproc::circularBufferIndex< timespec, cbIndexT > m_atimes
std::vector< double > m_atimesD
IMAGE * m_imageStream
The ImageStreamIO shared memory buffer.
pcf::IndiProperty m_indiP_frameSize
Property used to report the current frame size.
int appStartup()
Startup function.
float m_latencyCircBuffMaxTime
Maximum time of the latency meaurement circular buffers.
uint32_t m_circBuffLength
Length of the circular buffer, in frames.
uint32_t m_height
The height of the image, once deinterlaced etc.
int setupConfig(mx::app::appConfigurator &config)
Setup the configuration system.
@ OPERATING
The device is operating, other than homing.
@ READY
The device is ready for operation, but is not operating.
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
Definition dm.hpp:24