API
 
Loading...
Searching...
No Matches
pvcamCtrl.hpp
Go to the documentation of this file.
1/** \file pvcamCtrl.hpp
2 * \brief The MagAO-X pvcam controller header file
3 *
4 * \ingroup pvcamCtrl_files
5 */
6
7#ifndef pvcamCtrl_hpp
8#define pvcamCtrl_hpp
9
10#include <string.h>
11
12// PVCAM
13#include <master.h>
14#include <pvcam.h>
15
16#include "../../libMagAOX/libMagAOX.hpp" //Note this is included on command line to trigger pch
17#include "../../magaox_git_version.h"
18
19/** \defgroup pvcamCtrl
20 * \brief The pvcam controller application for teledyne cameras
21 *
22 * <a href="../handbook/operating/software/apps/pvcamCtrl.html">Application Documentation</a>
23 *
24 * \ingroup apps
25 *
26 */
27
28#ifndef PL_ERR_LIBRARY_NOT_INITIALIZED
29 #define PL_ERR_LIBRARY_NOT_INITIALIZED ( 157 )
30#endif
31
32/// Format an error message using pvcam facilities
33/**
34 * \returns a string of format "<func> failed: <pl_error_message>. <more>"
35 */
36std::string pvcamErrMessage( const std::string &func, ///< [in] the pvcam function which failed
37 int pec, ///< [in] the code from pl_error_code
38 const std::string &more ///< [in] extra information to include
39)
40{
41 char pvmsg[ERROR_MSG_LEN];
42 pl_error_message( pec, pvmsg );
43
44 std::string msg = func + " failed: " + pvmsg;
45 if( more != "" )
46 msg += " " + more;
47
48 return msg;
49}
50
51/// Helper for logging an error from pvcam
52#define log_pvcam_software_error( func, /**< [in] the pvcam function which failed*/ \
53 more /**< [in] extra information to include*/ \
54) \
55 { \
56 int pec = pl_error_code(); \
57 log<software_error>( { __FILE__, __LINE__, 0, pec, pvcamErrMessage( func, pec, more ) } ); \
58 }
59
60/** \defgroup pvcamCtrl_files
61 * \ingroup pvcamCtrl
62 */
63
64namespace MagAOX
65{
66namespace app
67{
68
69/// The MagAO-X pvcam controller
70/**
71 * \ingroup pvcamCtrl
72 */
73class pvcamCtrl : public MagAOXApp<true>,
74 public dev::stdCamera<pvcamCtrl>,
75 public dev::frameGrabber<pvcamCtrl>,
76 public dev::dssShutter<pvcamCtrl>,
77 public dev::telemeter<pvcamCtrl>
78{
79
84
85 // Give the test harness access.
86 friend class pvcamCtrl_test;
87
88 friend class dev::stdCamera<pvcamCtrl>;
89 friend class dev::frameGrabber<pvcamCtrl>;
90 friend class dev::dssShutter<pvcamCtrl>;
91 friend class dev::telemeter<pvcamCtrl>;
92
94
95 public:
96 /** \name app::dev Configurations
97 *@{
98 */
99 static constexpr bool c_stdCamera_tempControl =
100 false; ///< app::dev config to tell stdCamera not to expose temperature controls
101
102 static constexpr bool c_stdCamera_temp = false; ///< app::dev config to tell stdCamera not to expose temperature
103
104 static constexpr bool c_stdCamera_readoutSpeed =
105 true; ///< app::dev config to tell stdCamera to expose readout speed controls
106
107 static constexpr bool c_stdCamera_vShiftSpeed =
108 false; ///< app:dev config to tell stdCamera not to expose vertical shift speed control
109
110 static constexpr bool c_stdCamera_emGain =
111 false; ///< app::dev config to tell stdCamera not to expose EM gain controls
112
113 static constexpr bool c_stdCamera_exptimeCtrl =
114 true; ///< app::dev config to tell stdCamera to expose exposure time controls
115
116 static constexpr bool c_stdCamera_fpsCtrl = true; ///< app::dev config to tell stdCamera to expose FPS controls
117
118 static constexpr bool c_stdCamera_fps = true; ///< app::dev config to tell stdCamera to expose FPS status
119
120 static constexpr bool c_stdCamera_synchro =
121 false; ///< app::dev config to tell stdCamera to not expose synchro mode controls
122
123 static constexpr bool c_stdCamera_usesModes =
124 false; ///< app:dev config to tell stdCamera not to expose mode controls
125
126 static constexpr bool c_stdCamera_usesROI = true; ///< app:dev config to tell stdCamera to expose ROI controls
127
128 static constexpr bool c_stdCamera_cropMode =
129 false; ///< app:dev config to tell stdCamera to expose Crop Mode controls
130
131 static constexpr bool c_stdCamera_hasShutter =
132 true; ///< app:dev config to tell stdCamera to expose shutter controls
133
134 static constexpr bool c_stdCamera_usesStateString =
135 false; ///< app::dev confg to tell stdCamera to expose the state string property
136
137 static constexpr bool c_frameGrabber_flippable =
138 false; ///< app:dev config to tell framegrabber this camera can be flipped
139
140 ///@}
141
142 protected:
143 /** \name Configurable Parameters
144 *@{
145 */
146
147 std::string m_serialNumber; ///< The camera serial number.
148
149 ///\todo implement config-ing of this
150 float m_tempTol{ 0.1 }; ///< Tolerance in degrees C to declare the temperature control locked.
151
153 536870912 }; ///< Max size in bytes of the circular buffer to allocate. Default is 0.5 GB.
154
155 uint32_t m_acqSleep{ 5000 }; ///< The acquisition pause time, in ns, when no frame is ready.Default is 5000.
156
157 ///@}
158
159 int16 m_handle{ -1 }; ///< Camera handle, set when camera is opened
160
161 std::string m_camName; ///< Camera name, filled in as part of opening the camera.
162
163 struct gain
164 {
165 int index;
166 std::string name;
167
169 };
170
171 struct speed
172 {
173 int index;
175 int minG;
176 int maxG;
177 std::vector<gain> gains;
178 };
179
180 struct port
181 {
182 int index;
183 int value;
184 std::string name;
185
186 std::vector<speed> speeds;
187 };
188
189 std::vector<port> m_ports;
190
191 bool m_8bit{ false };
192
193 bool m_fpsSetted{ false }; ///< Flag indicating that FPS was set, not exposure time.
194
196 uns8 *m_circBuff{ nullptr };
197
199
200 sem_t m_frSemaphore; ///< Semaphore used to signal that a frame is ready.
201 sem_t m_frDoneSemaphore; ///< Semaphore used to signal that a frame has been processed
202
205
206 public:
207 /// Default c'tor.
208 pvcamCtrl();
209
210 /// D'tor, declared and defined for noexcept.
212 {
213 }
214
215 // MagAOXApp:
216
217 virtual void setupConfig();
218
219 /// Implementation of loadConfig logic, separated for testing.
220 /** This is called by loadConfig().
221 */
222 int loadConfigImpl(
223 mx::app::appConfigurator &_config /**< [in] an application configuration from which to load values*/ );
224
225 virtual void loadConfig();
226
227 /// Startup function
228 /**
229 *
230 */
231 virtual int appStartup();
232
233 /// Implementation of the FSM for pvcamCtrl.
234 /**
235 * \returns 0 on no critical error
236 * \returns -1 on an error requiring shutdown
237 */
238 virtual int appLogic();
239
240 /// Shutdown the app.
241 /**
242 *
243 */
244 virtual int appShutdown();
245
246 // stdCamera interface:
247
248 // This must set the power-on default values of
249 /* -- m_ccdTempSetpt
250 * -- m_currentROI
251 */
252 int powerOnDefaults();
253
254 int setTempControl();
255 int setTempSetPt();
256 int setReadoutSpeed();
257 int setVShiftSpeed();
258 int setEMGain();
259 int setExpTime();
260 int setFPS();
261
262 /// Check the next ROI
263 /** Checks if the target values are valid and adjusts them to the closest valid values if needed.
264 *
265 * \returns 0 if successful
266 * \returns -1 otherwise
267 */
268 int checkNextROI();
269
270 int setNextROI();
271
272 /// Sets the shutter state, via call to dssShutter::setShutterState(int) [stdCamera interface]
273 /**
274 * \returns 0 always
275 */
276 int setShutter( int sh );
277
278 // Framegrabber interface:
280 float fps();
281 int startAcquisition();
283 int loadImageIntoStream( void *dest );
284 int reconfig();
285
286 // pvcam specific:
287 int connect();
288
289 int fillSpeedTable();
290
291 void dumpEnum( uns32 paramID, const std::string &paramMnem );
292
293 int getTemp();
294
296
298
299 /** \name Telemeter Interface
300 *
301 * @{
302 */
303 int checkRecordTimes();
304
305 int recordTelem( const telem_stdcam * );
306
307 int recordTelem( const telem_fgtimings * );
308
309 ///@}
310};
311
312pvcamCtrl::pvcamCtrl() : MagAOXApp( MAGAOX_CURRENT_SHA1, MAGAOX_REPO_MODIFIED )
313{
314 m_powerMgtEnabled = true;
315 m_powerOnWait = 15;
316
317 m_expTime = 0.01;
318 m_expTimeSet = 0.01;
319
320 m_default_x = 1599.5;
321 m_default_y = 1599.5;
322 m_default_w = 3200;
323 m_default_h = 3200;
324 m_default_bin_x = 1;
325 m_default_bin_y = 1;
326
327 m_full_x = 1599.5;
328 m_full_y = 1599.5;
329 m_full_w = 3200;
330 m_full_h = 3200;
331
338
343 m_nextROI.bin_x = m_currentROI.bin_x;
344 m_nextROI.bin_y = m_currentROI.bin_y;
345
346 m_defaultReadoutSpeed = "dynamic_range";
347 m_readoutSpeedNames = { "sensitivity", "speed", "dynamic_range", "sub_electron" };
348 m_readoutSpeedNameLabels = { "Sensitivity", "Speed", "Dynamic Range", "Sub-Electron" };
349
352
353 return;
354}
355
357{
358 config.add( "camera.serialNumber",
359 "",
360 "camera.serialNumber",
361 argType::Required,
362 "camera",
363 "serialNumber",
364 false,
365 "int",
366 "The identifying serial number of the camera." );
367
368 // put this in camera because it is a camera interface config, not a framegrabber thing per se:
369 config.add( "camera.circBuffMaxBytes",
370 "",
371 "camera.circBuffMaxBytes",
372 argType::Required,
373 "camera",
374 "circBuffMaxBytes",
375 false,
376 "int",
377 "Maximum size in bytes of the circular buffer to allocate. Default is 0.5 GB." );
378
379 STDCAMERA_SETUP_CONFIG( config );
380
382
383 config.add( "framegrabber.acqSleep",
384 "",
385 "framegrabber.acqSleep",
386 argType::Required,
387 "framegrabber",
388 "acqSleep",
389 false,
390 "int",
391 "The acquisition pause time, in ns, when no frame is ready. Default is 5000." );
392
393 shutterT::setupConfig( config );
394
395 TELEMETER_SETUP_CONFIG( config );
396}
397
398int pvcamCtrl::loadConfigImpl( mx::app::appConfigurator &_config )
399{
400 _config( m_serialNumber, "camera.serialNumber" );
401 _config( m_circBuffMaxBytes, "camera.circBuffMaxBytes" );
402
403 if( m_serialNumber == "" )
404 {
405 log<text_log>( "camera serial number not provided", logPrio::LOG_CRITICAL );
406 return -1;
407 }
408
410
417
422 m_nextROI.bin_x = m_currentROI.bin_x;
423 m_nextROI.bin_y = m_currentROI.bin_y;
424
426
427 _config( m_acqSleep, "framegrabber.acqSleep" );
429
431
432 return 0;
433}
434
436{
437 if( loadConfigImpl( config ) != 0 )
438 {
439 log<text_log>( "error loading config", logPrio::LOG_CRITICAL );
440 m_shutdown = true;
441 }
442}
443
445{
446
448
449 if( sem_init( &m_frSemaphore, 0, 0 ) < 0 )
450 {
451 return log<software_critical, -1>( { __FILE__, __LINE__, errno, 0, "Initializing frame ready semaphore" } );
452 }
453
454 if( sem_init( &m_frDoneSemaphore, 0, 0 ) < 0 )
455 {
456 return log<software_critical, -1>( { __FILE__, __LINE__, errno, 0, "Initializing frame done semaphore" } );
457 }
458
460
461 if( shutterT::appStartup() < 0 )
462 {
463 return log<software_critical, -1>( { __FILE__, __LINE__ } );
464 }
465
467
468 return 0;
469}
470
472{
473 ///\todo why do we run dev appLogics first?
474
475 if( state() == stateCodes::POWERON )
476 {
477 if( !powerOnWaitElapsed() )
478 {
479 return 0;
480 }
481
483 }
484
486
488
489 // and run dssShutter's appLogic
490 if( shutterT::appLogic() < 0 )
491 {
492 return log<software_error, -1>( { __FILE__, __LINE__ } );
493 }
494
496 {
497 m_reconfig = true; // Trigger a f.g. thread reconfig.
498
499 // Might have gotten here because of a power off.
500 if( powerState() != 1 )
501 {
502 return 0;
503 }
504
505 std::unique_lock<std::mutex> lock( m_indiMutex );
506 if( connect() < 0 )
507 {
508 if( powerState() != 1 || powerStateTarget() != 1 )
509 return 0;
511 }
512
514 return 0;
515 }
516
518 {
521 }
522
524 {
525 if( getTemp() != 0 )
526 {
527 if( powerState() != 1 || powerStateTarget() != 1 )
528 return 0;
530 return 0;
531 }
532
534
536
538 recordCamera();
539 }
540
541 return 0;
542}
543
545{
546 if( m_handle != -1 )
547 {
548 if( !pl_cam_close( m_handle ) )
549 {
550 log_pvcam_software_error( "pl_cam_close", "continuing" );
551 }
552 m_handle = -1;
553 }
554
555 if( !pl_pvcam_uninit() )
556 {
557 if( pl_error_code() != PL_ERR_LIBRARY_NOT_INITIALIZED ) /// \todo this error code is manually defined
558 {
559 log_pvcam_software_error( "pl_pvcam_uninit", "continuing" );
560 }
561 }
562
564
566
567 if( shutterT::appShutdown() < 0 )
568 {
569 log<software_error>( { __FILE__, __LINE__, "error from shutterT::appShutdown()" } );
570 }
571
573
574 return 0;
575}
576
601
603{
604 return 0;
605}
606
608{
609 return 0;
610}
611
613{
615
616 return 0;
617}
618
620{
621 return 0;
622}
623
625{
626 return 0;
627}
628
630{
632
633 if( !pl_get_param( m_handle, PARAM_EXPOSURE_TIME, ATTR_MIN, static_cast<void *>( &minExpTime ) ) )
634 {
635 log_pvcam_software_error( "pl_get_param", "PARAM_EXPOSURE_TIME ATTR_MIN" );
636 log<software_error>( { __FILE__, __LINE__, "could not set exposure time" } );
637 return -1;
638 }
639
640 if( !pl_get_param( m_handle, PARAM_EXPOSURE_TIME, ATTR_MAX, static_cast<void *>( &maxExpTime ) ) )
641 {
642 log_pvcam_software_error( "pl_get_param", "PARAM_EXPOSURE_TIME ATTR_MAX" );
643 log<software_error>( { __FILE__, __LINE__, "could not set exposure time" } );
644 return -1;
645 }
646
647 if( m_expTimeSet * 1e6 < minExpTime )
648 {
649 m_expTimeSet = (int)( minExpTime / 1e6 + 0.5 );
650 log<text_log>( "increased m_expTimeSet to: " + std::to_string( m_expTimeSet ), logPrio::LOG_INFO );
651 }
652
653 if( m_expTimeSet * 1e6 > maxExpTime )
654 {
655 m_expTimeSet = (int)( maxExpTime / 1e6 - 0.5 );
656 log<text_log>( "decreased m_expTimeSet to: " + std::to_string( m_expTimeSet ), logPrio::LOG_INFO );
657 }
658
660
661 return 0;
662}
663
665{
666 m_expTimeSet = 1.0 / m_fpsSet;
667 m_fpsSetted = true;
668 return setExpTime();
669}
670
672{
673 if( m_nextROI.w > 3200 )
674 {
675 m_nextROI.w = 3200;
676 }
677
678 int x0 = m_nextROI.x - 0.5 * m_nextROI.w;
679
680 if( x0 < 0 )
681 {
682 m_nextROI.w += x0;
683 m_nextROI.x = 0;
684 }
685
686 int x1 = m_nextROI.x + 0.5 * m_nextROI.w;
687
688 if( x1 > 3199 )
689 {
690 m_nextROI.w = 3199 - x0;
691 m_nextROI.x = x0 + 0.5 * m_nextROI.w;
692 }
693
694 if( m_nextROI.h > 3200 )
695 {
696 m_nextROI.h = 3200;
697 }
698
699 int y0 = m_nextROI.y - 0.5 * m_nextROI.h;
700
701 if( y0 < 0 )
702 {
703 m_nextROI.h += y0;
704 m_nextROI.y = 0;
705 }
706
707 int y1 = m_nextROI.y + 0.5 * m_nextROI.h;
708
709 if( y1 > 3199 )
710 {
711 m_nextROI.h = 3199 - y0;
712 m_nextROI.y = y0 + 0.5 * m_nextROI.h;
713 }
714
715 return 0;
716}
717
719{
720 // This is done in setup acq
722 return 0;
723}
724
726{
727 recordCamera( true );
729}
730
732{
733
734 recordCamera( true );
735 //-- 0: register the callback
737 false ) // Because we registered it last time we configured acq:
738 {
739 log_pvcam_software_error( "pl_cam_deregister_callback", "PL_CALLBACK_EOF" );
740 }
741
744 reinterpret_cast<void *>( &st_endOfFrameCallback ),
745 static_cast<void *>( this ) ) != true )
746 {
747 log_pvcam_software_error( "pl_cam_register_callback_ex3", "PL_CALLBACK_EOF" );
748 return -1;
749 }
750
751 //-- 1: Set speed:
752 int32 value;
753
754 if( m_readoutSpeedNameSet == "sensitivity" )
755 {
756 value = 0;
757 m_8bit = false;
758 }
759 else if( m_readoutSpeedNameSet == "speed" )
760 {
761 value = 1;
762 m_8bit = true;
763 }
764 else if( m_readoutSpeedNameSet == "dynamic_range" )
765 {
766 value = 2;
767 m_8bit = false;
768 }
769 else if( m_readoutSpeedNameSet == "sub_electron" )
770 {
771 value = 3;
772 m_8bit = false;
773 }
774 else
775 {
776 value = 2;
777 m_8bit = false;
778 m_readoutSpeedNameSet = "dynamic_range";
779 }
781
782 if( pl_set_param( m_handle, PARAM_READOUT_PORT, static_cast<void *>( &value ) ) == false )
783 {
784 log_pvcam_software_error( "pl_set_param", "PARAM_READOUT_PORT" );
785 return -1;
786 }
787
788 //-- 2: Set ROI:
790 pvROI.s1 = m_nextROI.x - 0.5 * m_nextROI.w;
791 pvROI.s2 = pvROI.s1 + m_nextROI.w - 1;
792 pvROI.sbin = m_nextROI.bin_x;
793 pvROI.p1 = m_nextROI.y - 0.5 * m_nextROI.h;
794 pvROI.p2 = pvROI.p1 + m_nextROI.h - 1;
795 pvROI.pbin = m_nextROI.bin_y;
796
798
799 m_width = ( pvROI.s2 - pvROI.s1 + 1 ) / pvROI.sbin;
800 m_height = ( pvROI.p2 - pvROI.p1 + 1 ) / pvROI.pbin;
802
803 //-- 3: Setup continuous acquisition
804 uns32 fsize;
805
806 uns32 exptime = m_expTimeSet * 1e6;
807 if( pl_exp_setup_cont( m_handle, 1, &pvROI, TIMED_MODE, exptime, &fsize, CIRC_OVERWRITE ) == false )
808 {
809 log_pvcam_software_error( "pl_exp_setup_cont", "" );
810 m_shutdown = true;
811 return -1;
812 }
813
814 if( pl_get_param( m_handle, PARAM_EXPOSURE_TIME, ATTR_CURRENT, &exptime ) == false )
815 {
816 log_pvcam_software_error( "pl_get_param", "PARAM_EXPOSURE_TIME" );
817 }
818 m_expTime = ( 1.0 * exptime ) / 1e6;
820
821 uns32 readouttime = 0;
823 {
824 log_pvcam_software_error( "pl_get_param", "PARAM_READOUT_TIME" );
825 }
826
827 long64 predelay = 0;
829 {
830 log_pvcam_software_error( "pl_get_param", "PARAM_PRE_TRIGGER_DELAY" );
831 }
832
833 long64 clearing = 0;
835 {
836 log_pvcam_software_error( "pl_get_param", "PARAM_CLEARING_TIME" );
837 }
838
839 long64 postdelay = 0;
841 {
842 log_pvcam_software_error( "pl_get_param", "PARAM_POST_TRIGGER_DELAY" );
843 }
844
845 if( m_fpsSetted )
846 {
847 if( static_cast<long64>( m_expTime * 1000000 ) < readouttime )
848 {
849 // This is as fast as we can go, set expTime to longest possible
850 m_expTimeSet = ( readouttime - 1 ) / 1e6; // Go one usec lower, which will get cleaned up by the camera
851 }
852 else
853 {
854 m_expTimeSet = ( 1.0 / m_fpsSet ) - 2.0 * postdelay / 1e9;
855
856 if( static_cast<long64>( m_expTimeSet * 1000000 ) < readouttime )
857 {
858 // again, Set expTime to longest possible
859 m_expTimeSet = ( readouttime - 1.0 ) / 1e6; // Go one usec lower, which will get cleaned up by the
860 // camera
861 }
862 }
863
864 exptime = m_expTimeSet * 1e6;
866 {
867 log_pvcam_software_error( "pl_exp_setup_cont", "" );
868 m_shutdown = true;
869 return -1;
870 }
871
872 // Note: this call apparently resets exptime to 0 ....
873 if( pl_get_param( m_handle, PARAM_EXPOSURE_TIME, ATTR_CURRENT, &exptime ) == false )
874 {
875 log_pvcam_software_error( "pl_get_param", "PARAM_EXPOSURE_TIME" );
876 }
877
878 m_expTime = ( 1.0 * exptime ) / 1e6;
879
881
882 m_fpsSetted = false;
883
885 {
886 log_pvcam_software_error( "pl_get_param", "PARAM_READOUT_TIME" );
887 }
888 }
889
890 if( static_cast<long64>( m_expTime * 1000000 ) < readouttime )
891 {
892 m_fps = 1.0 / ( readouttime / 1e6 + postdelay / 1e9 );
893 }
894 else
895 {
896 m_fps =
897 1.0 / ( m_expTime + 2.0 * postdelay / 1e9 ); // At some speeds there is an extra + ~250/1e9 that is needed.
898 }
899
900 m_fpsSet = m_fps;
901
902 //-- 4: Allocate the acq circular buffer
903 if( m_circBuff != nullptr )
904 {
905 delete[] m_circBuff;
906 m_circBuffBytes = 0;
907 m_circBuff = nullptr;
908 }
909
911
912 m_circBuff = new( std::nothrow ) uns8[m_circBuffBytes];
913
914 if( m_circBuff == nullptr )
915 {
916 log<software_critical>( { __FILE__, __LINE__, "failed to allocate acquisition circular buffer." } );
918 return -1;
919 }
920
921 recordCamera( true );
922
923 return 0;
924}
925
927{
928 return m_fps;
929}
930
932{
934 {
935 log_pvcam_software_error( "pv_exp_start_cont", "" );
936 return -1;
937 }
938
939 log<text_log>( "continuous acquisition started", logPrio::LOG_INFO );
940 return 0;
941}
942
944{
945 // See if a frame is ready without waiting
946 int rv = sem_trywait( &m_frSemaphore );
947 if( rv == 0 )
948 {
950 {
951 log<software_critical>( { __FILE__, __LINE__, errno, 0, "clock_gettime" } );
952 }
953
954 return 0;
955 }
956 else if( errno != EAGAIN )
957 {
958 log<software_critical>( { __FILE__, __LINE__, errno, 0, "sem_trywait" } );
959 return -1;
960 }
961
962 // if none is ready we pause before going on
963 mx::sys::nanoSleep( m_acqSleep );
964
965 return 1;
966}
967
969{
970
971 // Obtain a pointer to the last acquired frame
972 uns8 *frame;
973 if( pl_exp_get_latest_frame( m_handle, reinterpret_cast<void **>( &frame ) ) == false )
974 {
975 log_pvcam_software_error( "pl_exp_get_latest_frame", "" );
976 }
977
978 if( m_8bit )
979 {
980 uint16_t *dest16 = static_cast<uint16_t *>( dest );
981 uint8_t *src8 = static_cast<uint8_t *>( frame );
982 for( uint32_t i = 0; i < m_width * m_height; ++i )
983 {
984 dest16[i] = src8[i];
985 }
986 }
987 else
988 {
989 memcpy( dest, frame, m_width * m_height * 2 );
990 }
991
992 // Now tell the callback it can go on
993 if( sem_post( &m_frDoneSemaphore ) < 0 )
994 {
995 log<software_critical>( { __FILE__, __LINE__, errno, 0, "Error posting to frame done semaphore" } );
996 return -1;
997 }
998
999 return 0;
1000}
1001
1003{
1004 if( pl_exp_stop_cont( m_handle, CCS_HALT ) == false )
1005 {
1006 log_pvcam_software_error( "pl_exp_stop_cont", "" );
1007 }
1008 return 0;
1009}
1010
1012{
1013
1014 // In picam, we had to initialize every time. We'll do that here too.
1015
1016 // So close handle if it's open
1017 if( m_handle != -1 )
1018 {
1019 if( !pl_cam_close( m_handle ) )
1020 {
1021 log_pvcam_software_error( "pl_cam_close", "" );
1022 return -1;
1023 }
1024 m_handle = -1;
1025 }
1026
1027 // Uninit
1028 if( !pl_pvcam_uninit() )
1029 {
1030 if( pl_error_code() != PL_ERR_LIBRARY_NOT_INITIALIZED ) /// \todo this error code is manually defined
1031 {
1032 log_pvcam_software_error( "pl_pvcam_uninit", "continuing" );
1033 }
1034 }
1035
1036 if( !pl_pvcam_init() )
1037 {
1038 log_pvcam_software_error( "pl_pvcam_init", "" );
1039 return -1;
1040 }
1041
1043
1044 // Read the number of cameras in the system.
1045 // This will return total number of PVCAM cameras regardless of interface.
1047 {
1048 log_pvcam_software_error( "pl_cam_get_total", "" );
1049 return -1;
1050 }
1051
1052 if( nrOfCameras == 0 )
1053 {
1054 if( !stateLogged() )
1055 {
1056 log<text_log>( "Found 0 pvcam cameras.", logPrio::LOG_INFO );
1057 }
1059 return 0;
1060 }
1061
1062 log<text_log>( "Found " + std::to_string( nrOfCameras ) + " pvcam cameras.", logPrio::LOG_INFO );
1063
1064 for( int n = 0; n < nrOfCameras; ++n )
1065 {
1066 char camName[CAM_NAME_LEN]{ '\0' };
1067
1068 // Obtain PVCAM-name for this particular camera
1069 if( pl_cam_get_name( n, camName ) != PV_OK )
1070 {
1071 log_pvcam_software_error( "pl_cam_get_name", "" );
1072 return -1;
1073 }
1074
1075 int16_t handle = -1;
1076
1077 // Open to check its serial number
1079 {
1080 log_pvcam_software_error( "pl_cam_open",
1081 "" ); // We log this for now, but with 2 apps running we prob want to ignore
1082 continue;
1083 }
1084
1085 // Read the version of the Device Driver
1086
1088 if( !pl_get_param( handle, PARAM_HEAD_SER_NUM_ALPHA, ATTR_AVAIL, static_cast<void *>( &isAvailable ) ) )
1089 {
1090 log_pvcam_software_error( "pl_get_param", "PARAM_HEAD_SER_NUM_ALPHA ATTR_AVAIL" );
1091
1092 if( !pl_cam_close( handle ) )
1093 {
1094 log_pvcam_software_error( "pl_cam_close", "" );
1095 }
1096
1097 return -1;
1098 }
1099
1100 if( isAvailable )
1101 {
1102 char camSerial[MAX_ALPHA_SER_NUM_LEN]{ '\0' };
1103
1104 if( !pl_get_param( handle, PARAM_HEAD_SER_NUM_ALPHA, ATTR_CURRENT, static_cast<void *>( camSerial ) ) )
1105 {
1106 log_pvcam_software_error( "pl_get_param", "PARAM_HEAD_SER_NUM_ALPHA ATTR_CURRENT" );
1107
1108 if( !pl_cam_close( handle ) )
1109 {
1110 log_pvcam_software_error( "pl_cam_close", "" );
1111 }
1112
1113 return -1;
1114 }
1115
1116 if( camSerial == m_serialNumber )
1117 {
1118 state( stateCodes::NOTCONNECTED ); // not strictly true, but logically true until m_handle is set
1120 m_handle = handle;
1121 break;
1122 }
1123 }
1124
1125 // If we're here then either it didn't have a serial number, or it didn't have the right serial number
1126 if( !pl_cam_close( handle ) )
1127 {
1128 log_pvcam_software_error( "pl_cam_close", "" );
1129 return -1;
1130 }
1131 }
1132
1133 if( m_handle > -1 )
1134 {
1135 log<text_log>( "Opened camera " + m_serialNumber + " at " + m_camName, logPrio::LOG_INFO );
1137
1138 int32 res;
1139 uns16 idx;
1140
1141 // Set exposure resolution to usec.
1142
1143 idx = 1;
1144 if( pl_set_param( m_handle, PARAM_EXP_RES_INDEX, &idx ) == false )
1145 {
1146 log_pvcam_software_error( "pl_set_param", "PARAM_EXP_RES_INDEX" );
1147 }
1148
1149 if( pl_get_param( m_handle, PARAM_EXP_RES, ATTR_CURRENT, &res ) == false )
1150 {
1151 log_pvcam_software_error( "pl_get_param", "PARAM_EXP_RES" );
1152 }
1153
1155 {
1156 log_pvcam_software_error( "pl_get_param", "PARAM_EXP_RES_INDEX" );
1157 }
1158
1160 }
1161 else
1162 {
1163 if( !stateLogged() )
1164 {
1165 log<text_log>( "camera not found", logPrio::LOG_INFO );
1166 }
1168 }
1169
1170 return 0;
1171}
1172
1174{
1176 {
1177 return -1;
1178 }
1179
1180 uns32 nports;
1181
1182 if( pl_get_param( m_handle, PARAM_READOUT_PORT, ATTR_COUNT, static_cast<void *>( &nports ) ) == false )
1183 {
1184 log_pvcam_software_error( "pl_get_param", "PARAM_READOUT_PORT" );
1185 return -1;
1186 }
1187
1188 std::cerr << "Found " << nports << " ports\n";
1189
1190 m_ports.resize( nports );
1191
1192 for( uns32 p = 0; p < nports; ++p )
1193 {
1196 {
1197 log_pvcam_software_error( "pl_enum_str_length", "PARAM_READOUT_PORT" );
1198 return -1;
1199 }
1200
1201 char *text = new( std::nothrow ) char[strLength];
1202 if( !text )
1203 {
1204 ///\todo log this properly
1205 std::cerr << "failed to allocate string\n";
1206 return -1;
1207 }
1208
1209 int32 value;
1211 {
1212 log_pvcam_software_error( "pl_get_enum_param", "PARAM_READOUT_PORT" );
1213 delete[] text;
1214 return false;
1215 }
1216
1217 m_ports[p].index = p;
1218 m_ports[p].value = value;
1219 m_ports[p].name = text;
1220
1221 std::cerr << "Port: " << p << " name: " << text << " value: " << value << "\n";
1222
1223 delete[] text;
1224
1225 if( pl_set_param( m_handle, PARAM_READOUT_PORT, static_cast<void *>( &value ) ) == false )
1226 {
1227 log_pvcam_software_error( "pl_set_param", "PARAM_READOUT_PORT" );
1228 return -1;
1229 }
1230
1231 uns32 nspeeds;
1232 if( pl_get_param( m_handle, PARAM_SPDTAB_INDEX, ATTR_COUNT, static_cast<void *>( &nspeeds ) ) == false )
1233 {
1234 log_pvcam_software_error( "pl_get_param", "PARAM_SPDTAB_INDEX" );
1235 return -1;
1236 }
1237 std::cerr << " Speeds: " << nspeeds << "\n";
1238
1239 m_ports[p].speeds.resize( nspeeds );
1240
1241 for( uns32 s = 0; s < nspeeds; ++s )
1242 {
1243 if( pl_set_param( m_handle, PARAM_SPDTAB_INDEX, static_cast<void *>( &s ) ) == false )
1244 {
1245 log_pvcam_software_error( "pl_set_param", "PARAM_SPDTAB_INDEX" );
1246 return -1;
1247 }
1248
1249 uns16 pixtime;
1250 if( pl_get_param( m_handle, PARAM_PIX_TIME, ATTR_CURRENT, static_cast<void *>( &pixtime ) ) == false )
1251 {
1252 log_pvcam_software_error( "pl_get_param", "PARAM_PIX_TIME" );
1253 return -1;
1254 }
1255
1256 m_ports[p].speeds[s].pixTime = pixtime;
1257
1258 uns32 ngains;
1259 if( pl_get_param( m_handle, PARAM_GAIN_INDEX, ATTR_COUNT, static_cast<void *>( &ngains ) ) == false )
1260 {
1261 log_pvcam_software_error( "pl_get_param", "PARAM_GAIN_INDEX ATTR_COUNT" );
1262 return -1;
1263 }
1264
1265 int16 ming;
1266 if( pl_get_param( m_handle, PARAM_GAIN_INDEX, ATTR_MIN, static_cast<void *>( &ming ) ) == false )
1267 {
1268 log_pvcam_software_error( "pl_get_param", "PARAM_GAIN_INDEX ATTR_MIN" );
1269 return -1;
1270 }
1271
1272 int16 maxg;
1273 if( pl_get_param( m_handle, PARAM_GAIN_INDEX, ATTR_MIN, static_cast<void *>( &maxg ) ) == false )
1274 {
1275 log_pvcam_software_error( "pl_get_param", "PARAM_GAIN_INDEX ATTR_MAX" );
1276 return -1;
1277 }
1278
1279 std::cerr << " Speed: " << s << " " << " pixtime: " << pixtime << " gains: " << ngains << " [" << ming
1280 << "-" << maxg << "]\n";
1281
1282 m_ports[p].speeds[s].minG = ming;
1283 m_ports[p].speeds[s].maxG = maxg;
1284
1285 m_ports[p].speeds[s].gains.resize( ngains );
1286
1287 for( uns32 g = 0; g < ngains; ++g )
1288 {
1289
1290 int16 gg = ming + g;
1291 if( pl_set_param( m_handle, PARAM_GAIN_INDEX, static_cast<void *>( &gg ) ) == false )
1292 {
1293 log_pvcam_software_error( "pl_set_param", "PARAM_GAIN_INDEX" );
1294 return -1;
1295 }
1296
1298 if( pl_get_param( m_handle, PARAM_BIT_DEPTH, ATTR_CURRENT, static_cast<void *>( &bitdepth ) ) == false )
1299 {
1300 log_pvcam_software_error( "pl_get_param", "PARAM_BIT_DEPTH ATTR_CURRENT" );
1301 return -1;
1302 }
1303 std::cerr << " Gain: " << g << " bitdepth: " << bitdepth << "\n";
1304 }
1305 }
1306 }
1307
1308 return 0;
1309}
1310
1311void pvcamCtrl::dumpEnum( uns32 paramID, const std::string &paramMnem )
1312{
1313
1314 if( state() == stateCodes::CONNECTED )
1315 {
1316 uns32 count;
1317
1318 if( PV_OK != pl_get_param( m_handle, paramID, ATTR_COUNT, static_cast<void *>( &count ) ) )
1319 {
1320 log_pvcam_software_error( "pl_get_param", paramMnem );
1321 // TODO: Handle error
1322 return;
1323 }
1324 if( count == 0 )
1325 {
1326 std::cerr << paramMnem << ": count 0\n";
1327 return;
1328 }
1329 for( uns32 n = 0; n < count; ++n )
1330 {
1333 {
1334 log_pvcam_software_error( "pl_enum_str_length", paramMnem );
1335 // TODO: Handle error
1336 break;
1337 }
1338 char *text = new( std::nothrow ) char[strLength];
1339 if( !text )
1340 {
1341 // TODO: Handle error
1342 break;
1343 }
1344 int32 value;
1345 if( PV_OK != pl_get_enum_param( m_handle, paramID, n, &value, text, strLength ) )
1346 {
1347 log_pvcam_software_error( "pl_get_enum_param", paramMnem );
1348 // TODO: Handle error
1349 delete[] text;
1350 break;
1351 }
1352 std::cerr << paramMnem;
1353 fprintf( stderr, " item at index %u, value: %d, text: '%s'\n", n, value, text );
1354 delete[] text;
1355 }
1356 }
1357 else
1358 {
1359 std::cerr << "dumpEnum: not CONNECTED\n";
1360 }
1361}
1362
1364{
1365 if( state() == stateCodes::OPERATING )
1366 {
1367 return 0;
1368 }
1369
1371 if( !pl_get_param( m_handle, PARAM_TEMP_SETPOINT, ATTR_AVAIL, static_cast<void *>( &isAvailable ) ) )
1372 {
1373 if( powerState() != 1 || powerStateTarget() != 1 )
1374 return 0;
1375 log_pvcam_software_error( "pl_get_param", "PARAM_TEMP ATTR_AVAIL" );
1377 }
1378
1379 int16 stemp;
1380 if( isAvailable ) // Maybe this is a separate check. Don't yet know what happens when acquiring
1381 {
1382 if( !pl_get_param( m_handle, PARAM_TEMP_SETPOINT, ATTR_CURRENT, static_cast<void *>( &stemp ) ) )
1383 {
1384 if( powerState() != 1 || powerStateTarget() != 1 )
1385 return 0;
1386 log_pvcam_software_error( "pl_get_param", "PARAM_TEMP ATTR_AVAIL" );
1388 }
1389
1390 m_ccdTempSetpt = stemp / 100.0;
1391 }
1392
1393 if( !pl_get_param( m_handle, PARAM_TEMP, ATTR_AVAIL, static_cast<void *>( &isAvailable ) ) )
1394 {
1395 if( powerState() != 1 || powerStateTarget() != 1 )
1396 return 0;
1397 log_pvcam_software_error( "pl_get_param", "PARAM_TEMP ATTR_AVAIL" );
1399 }
1400
1401 int16 ctemp;
1402 if( isAvailable ) // Maybe this is a separate check. Don't yet know what happens when acquiring
1403 {
1404 if( !pl_get_param( m_handle, PARAM_TEMP, ATTR_CURRENT, static_cast<void *>( &ctemp ) ) )
1405 {
1406 if( powerState() != 1 || powerStateTarget() != 1 )
1407 return 0;
1408 log_pvcam_software_error( "pl_get_param", "PARAM_TEMP ATTR_AVAIL" );
1410 }
1411
1412 m_ccdTemp = ctemp / 100.0;
1413 }
1414
1416 {
1417 m_tempControlStatus = true;
1418 m_tempControlOnTarget = false;
1419 m_tempControlStatusStr = "UNLOCKED";
1420 }
1421 else
1422 {
1423 m_tempControlStatus = true;
1424 m_tempControlOnTarget = true;
1425 m_tempControlStatusStr = "LOCKED";
1426 }
1427
1428 return 0;
1429}
1430
1431void pvcamCtrl::st_endOfFrameCallback( FRAME_INFO *finfo, void *pvcamCtrlInst )
1432{
1433 static_cast<pvcamCtrl *>( pvcamCtrlInst )->endOfFrameCallback( finfo );
1434}
1435
1436void pvcamCtrl::endOfFrameCallback( FRAME_INFO *finfo )
1437{
1438 m_frameInfo = *finfo;
1439
1440 // 0 out the frDoneSemaphore just to be sure
1441 int semval;
1443 while( semval > 0 )
1444 {
1447 }
1448
1449 // Now tell the writer to get going
1450 if( sem_post( &m_frSemaphore ) < 0 )
1451 {
1452 log<software_critical>( { __FILE__, __LINE__, errno, 0, "Error posting to frame ready semaphore" } );
1453 return;
1454 }
1455
1456 // Now wait on the frame done semaphore
1457 timespec ts;
1458
1459 if( clock_gettime( CLOCK_REALTIME, &ts ) < 0 )
1460 {
1461 log<software_critical>( { __FILE__, __LINE__, errno, 0, "clock_gettime" } );
1462 }
1463
1464 mx::sys::timespecAddNsec( ts, 1e9 ); // We wait for up to one second for the frame done processing signal
1465
1467}
1468
1473
1475{
1476 return recordCamera( true );
1477}
1478
1480{
1481 return recordFGTimings( true );
1482}
1483
1484} // namespace app
1485} // namespace MagAOX
1486
1487#endif // pvcamCtrl_hpp
The base-class for MagAO-X applications.
Definition MagAOXApp.hpp:73
stateCodes::stateCodeT state()
Get the current state code.
int powerState()
Returns the current power state.
int m_shutdown
Flag to signal it's time to shutdown. When not 0, the main loop exits.
int powerStateTarget()
Returns the target power state.
int stateLogged()
Updates and returns the value of m_stateLogged. Will be 0 on first call after a state change,...
static int log(const typename logT::messageT &msg, logPrioT level=logPrio::LOG_DEFAULT)
Make a log entry.
bool powerOnWaitElapsed()
This method tests whether the power on wait time has elapsed.
unsigned long m_powerOnWait
Time in sec to wait for device to boot after power on.
std::mutex m_indiMutex
Mutex for locking INDI communications.
MagAO-X Uniblitz DSS Shutter interface.
void loadConfig(mx::app::appConfigurator &config)
load the configuration system results
void setupConfig(mx::app::appConfigurator &config)
Setup the configuration system.
int setShutterState(int sh)
Change shutter state.
timespec m_currImageTimestamp
The timestamp of the current image.
uint32_t m_width
The width of the image, once deinterlaced etc.
uint8_t m_dataType
The ImageStreamIO type code.
bool m_reconfig
Flag to set if a camera reconfiguration requires a framegrabber reset.
uint32_t m_height
The height of the image, once deinterlaced etc.
MagAO-X standard camera interface.
std::vector< std::string > m_readoutSpeedNames
float m_fpsSet
The commanded fps, as set by user.
float m_default_x
Power-on ROI center x coordinate.
std::string m_tempControlStatusStr
Camera specific description of temperature control status.
std::vector< std::string > m_readoutSpeedNameLabels
std::string m_defaultReadoutSpeed
The default readout speed of the camera.
float m_expTime
The current exposure time, in seconds.
int m_default_bin_x
Power-on ROI x binning.
float m_expTimeSet
The exposure time, in seconds, as set by user.
float m_full_y
The full ROI center y coordinate.
std::string m_readoutSpeedName
The current readout speed name.
float m_ccdTempSetpt
The desired temperature, in C.
bool m_tempControlStatus
Whether or not temperature control is active.
float m_full_x
The full ROI center x coordinate.
float m_ccdTemp
The current temperature, in C.
bool m_tempControlOnTarget
Whether or not the temperature control system is on its target temperature.
float m_default_y
Power-on ROI center y coordinate.
int m_default_bin_y
Power-on ROI y binning.
std::string m_readoutSpeedNameSet
The user requested readout speed name, to be set by derived()
The MagAO-X pvcam controller.
Definition pvcamCtrl.hpp:78
static constexpr bool c_stdCamera_exptimeCtrl
app::dev config to tell stdCamera to expose exposure time controls
int16 m_handle
Camera handle, set when camera is opened.
static constexpr bool c_stdCamera_usesROI
app:dev config to tell stdCamera to expose ROI controls
uint32_t m_acqSleep
The acquisition pause time, in ns, when no frame is ready.Default is 5000.
static constexpr bool c_stdCamera_emGain
app::dev config to tell stdCamera not to expose EM gain controls
static constexpr bool c_stdCamera_usesStateString
app::dev confg to tell stdCamera to expose the state string property
static constexpr bool c_stdCamera_fps
app::dev config to tell stdCamera to expose FPS status
friend class pvcamCtrl_test
Definition pvcamCtrl.hpp:86
void endOfFrameCallback(FRAME_INFO *finfo)
virtual void loadConfig()
static constexpr bool c_stdCamera_tempControl
app::dev config to tell stdCamera not to expose temperature controls
Definition pvcamCtrl.hpp:99
int recordTelem(const telem_stdcam *)
static void st_endOfFrameCallback(FRAME_INFO *finfo, void *pvcamCtrlInst)
int setShutter(int sh)
Sets the shutter state, via call to dssShutter::setShutterState(int) [stdCamera interface].
static constexpr bool c_stdCamera_usesModes
app:dev config to tell stdCamera not to expose mode controls
std::vector< port > m_ports
virtual int appShutdown()
Shutdown the app.
bool m_fpsSetted
Flag indicating that FPS was set, not exposure time.
uint32_t m_circBuffMaxBytes
Max size in bytes of the circular buffer to allocate. Default is 0.5 GB.
int checkNextROI()
Check the next ROI.
std::vector< speed > speeds
static constexpr bool c_stdCamera_hasShutter
app:dev config to tell stdCamera to expose shutter controls
static constexpr bool c_stdCamera_fpsCtrl
app::dev config to tell stdCamera to expose FPS controls
dev::stdCamera< pvcamCtrl > stdCameraT
Definition pvcamCtrl.hpp:80
dev::frameGrabber< pvcamCtrl > frameGrabberT
Definition pvcamCtrl.hpp:81
static constexpr bool c_stdCamera_temp
app::dev config to tell stdCamera not to expose temperature
pvcamCtrl()
Default c'tor.
virtual int appLogic()
Implementation of the FSM for pvcamCtrl.
sem_t m_frSemaphore
Semaphore used to signal that a frame is ready.
sem_t m_frDoneSemaphore
Semaphore used to signal that a frame has been processed.
int loadConfigImpl(mx::app::appConfigurator &_config)
Implementation of loadConfig logic, separated for testing.
~pvcamCtrl() noexcept
D'tor, declared and defined for noexcept.
std::string m_camName
Camera name, filled in as part of opening the camera.
virtual int appStartup()
Startup function.
int loadImageIntoStream(void *dest)
dev::dssShutter< pvcamCtrl > shutterT
Definition pvcamCtrl.hpp:82
static constexpr bool c_stdCamera_readoutSpeed
app::dev config to tell stdCamera to expose readout speed controls
void dumpEnum(uns32 paramID, const std::string &paramMnem)
static constexpr bool c_stdCamera_synchro
app::dev config to tell stdCamera to not expose synchro mode controls
dev::telemeter< pvcamCtrl > telemeterT
Definition pvcamCtrl.hpp:83
std::vector< gain > gains
static constexpr bool c_stdCamera_cropMode
app:dev config to tell stdCamera to expose Crop Mode controls
float m_tempTol
Tolerance in degrees C to declare the temperature control locked.
std::string m_serialNumber
The camera serial number.
virtual void setupConfig()
static constexpr bool c_stdCamera_vShiftSpeed
app:dev config to tell stdCamera not to expose vertical shift speed control
static constexpr bool c_frameGrabber_flippable
app:dev config to tell framegrabber this camera can be flipped
#define FRAMEGRABBER_SETUP_CONFIG(cfig)
Call frameGrabberT::setupConfig with error checking for frameGrabber.
#define FRAMEGRABBER_APP_LOGIC
Call frameGrabberT::appLogic with error checking for frameGrabber.
#define FRAMEGRABBER_APP_SHUTDOWN
Call frameGrabberT::appShutdown with error checking for frameGrabber.
#define FRAMEGRABBER_UPDATE_INDI
Call frameGrabberT::updateINDI with error checking for frameGrabber.
#define FRAMEGRABBER_LOAD_CONFIG(cfig)
Call frameGrabberT::loadConfig with error checking for frameGrabber.
#define FRAMEGRABBER_APP_STARTUP
Call frameGrabberT::appStartup with error checking for frameGrabber.
@ OPERATING
The device is operating, other than homing.
@ NODEVICE
No device exists for the application to control.
@ FAILURE
The application has failed, should be used when m_shutdown is set for an error.
@ ERROR
The application has encountered an error, from which it is recovering (with or without intervention)
@ READY
The device is ready for operation, but is not operating.
@ CONNECTED
The application has connected to the device or service.
@ NOTCONNECTED
The application is not connected to the device or service.
@ POWERON
The device power is on.
std::unique_lock< std::mutex > lock(m_indiMutex)
Definition dm.hpp:24
static constexpr logPrioT LOG_INFO
Informational. The info log level is the lowest level recorded during normal operations.
static constexpr logPrioT LOG_CRITICAL
The process can not continue and will shut down (fatal)
#define PL_ERR_LIBRARY_NOT_INITIALIZED
Definition pvcamCtrl.hpp:29
std::string pvcamErrMessage(const std::string &func, int pec, const std::string &more)
Format an error message using pvcam facilities.
Definition pvcamCtrl.hpp:36
#define log_pvcam_software_error(func, more)
Helper for logging an error from pvcam.
Definition pvcamCtrl.hpp:52
#define STDCAMERA_SETUP_CONFIG(cfig)
Call stdCameraT::setupConfig with error checking for stdCamera.
#define STDCAMERA_APP_LOGIC
Call stdCameraT::appLogic with error checking for stdCamera.
#define STDCAMERA_APP_STARTUP
Call stdCameraT::appStartup with error checking for stdCamera.
#define STDCAMERA_LOAD_CONFIG(cfig)
Call stdCameraT::loadConfig with error checking for stdCamera.
#define STDCAMERA_UPDATE_INDI
Call stdCameraT::updateINDI with error checking for stdCamera.
#define STDCAMERA_APP_SHUTDOWN
Call stdCameraT::appShutdown with error checking for stdCamera.
A device base class which saves telemetry.
Definition telemeter.hpp:69
int checkRecordTimes(const telT &tel, telTs... tels)
Check the time of the last record for each telemetry type and make an entry if needed.
Software CRITICAL log entry.
Software ERR log entry.
Log entry recording framegrabber timings.
Log entry recording stdcam stage specific status.
#define TELEMETER_APP_LOGIC
Call telemeter::appLogic with error checking.
#define TELEMETER_LOAD_CONFIG(cfig)
Call telemeter::loadConfig with error checking.
#define TELEMETER_APP_STARTUP
Call telemeter::appStartup with error checking.
#define TELEMETER_SETUP_CONFIG(cfig)
Call telemeter::setupConfig with error checking.
#define TELEMETER_APP_SHUTDOWN
Call telemeter::appShutdown with error checking.