API
 
Loading...
Searching...
No Matches
ocam2KCtrl.hpp
Go to the documentation of this file.
1/** \file ocam2KCtrl.hpp
2 * \brief The MagAO-X OCAM2K EMCCD camera controller.
3 *
4 * \author Jared R. Males (jaredmales@gmail.com)
5 *
6 * \ingroup ocam2KCtrl_files
7 */
8
9#ifndef ocam2KCtrl_hpp
10#define ocam2KCtrl_hpp
11
12
13#include <edtinc.h>
14
15
16
17#include "../../libMagAOX/libMagAOX.hpp" //Note this is included on command line to trigger pch
18#include "../../magaox_git_version.h"
19
20typedef MagAOX::app::MagAOXApp<true> MagAOXAppT; //This needs to be before pdvUtils.hpp for logging to work.
21
22#include "fli/ocam2_sdk.h"
23#include "ocamUtils.hpp"
24
25namespace MagAOX
26{
27namespace app
28{
29
30/** \defgroup ocam2KCtrl OCAM2K EMCCD Camera
31 * \brief Control of the OCAM2K EMCCD Camera.
32 *
33 * <a href="../handbook/operating/software/apps/ocam2KCtrl.html">Application Documentation</a>
34 *
35 * \ingroup apps
36 *
37 */
38
39/** \defgroup ocam2KCtrl_files OCAM2K EMCCD Camera Files
40 * \ingroup ocam2KCtrl
41 */
42
43/** MagAO-X application to control the OCAM 2K EMCCD
44 *
45 * \ingroup ocam2KCtrl
46 *
47 */
48class ocam2KCtrl : public MagAOXApp<>, public dev::stdCamera<ocam2KCtrl>, public dev::edtCamera<ocam2KCtrl>, public dev::frameGrabber<ocam2KCtrl>,
49 public dev::dssShutter<ocam2KCtrl>, public dev::telemeter<ocam2KCtrl>
50{
51 friend class dev::stdCamera<ocam2KCtrl>;
52 friend class dev::edtCamera<ocam2KCtrl>;
53 friend class dev::frameGrabber<ocam2KCtrl>;
54 friend class dev::dssShutter<ocam2KCtrl>;
55 friend class dev::telemeter<ocam2KCtrl>;
56
58
59public:
60 /** \name app::dev Configurations
61 *@{
62 */
63 static constexpr bool c_stdCamera_tempControl = true; ///< app::dev config to tell stdCamera to expose temperature controls
64
65 static constexpr bool c_stdCamera_temp = true; ///< app::dev config to tell stdCamera to expose temperature (ignored since tempControl==true)
66
67 static constexpr bool c_stdCamera_readoutSpeed = false; ///< app::dev config to tell stdCamera not to expose readout speed controls
68
69 static constexpr bool c_stdCamera_vShiftSpeed = false; ///< app:dev config to tell stdCamera not to expose vertical shift speed control
70
71 static constexpr bool c_stdCamera_emGain = true; ///< app::dev config to tell stdCamera to expose EM gain controls
72
73 static constexpr bool c_stdCamera_exptimeCtrl = false; ///< app::dev config to tell stdCamera not to expose exposure time controls
74
75 static constexpr bool c_stdCamera_fpsCtrl = true; ///< app::dev config to tell stdCamera to expose FPS controls
76
77 static constexpr bool c_stdCamera_fps = true; ///< app::dev config to tell stdCamera not to expose FPS status (ignored since fpsCtrl==true)
78
79 static constexpr bool c_stdCamera_synchro = true; ///< app::dev config to tell stdCamera to expose synchro mode controls
80
81 static constexpr bool c_stdCamera_usesModes = true; ///< app:dev config to tell stdCamera not to expose mode controls
82
83 static constexpr bool c_stdCamera_usesROI = false; ///< app:dev config to tell stdCamera to expose ROI controls
84
85 static constexpr bool c_stdCamera_cropMode = false; ///< app:dev config to tell stdCamera to expose Crop Mode controls
86
87 static constexpr bool c_stdCamera_hasShutter = true; ///< app:dev config to tell stdCamera to expose shutter controls
88
89 static constexpr bool c_stdCamera_usesStateString = true; ///< app::dev confg to tell stdCamera to expose the state string property
90
91 static constexpr bool c_edtCamera_relativeConfigPath = true; ///< app::dev config to tell edtCamera to use relative path to camera config file
92
93 static constexpr bool c_frameGrabber_flippable = false; ///< app:dev config to tell framegrabber these images can not be flipped
94
95 ///@}
96protected:
97
98 /** \name configurable parameters
99 *@{
100 */
101
102 //Camera:
103
104 std::string m_ocamDescrambleFile; ///< Path the OCAM 2K pixel descrambling file, relative to MagAO-X config directory.
105
106
107
108 ///@}
109
110 ocam2_id m_ocam2_id {0}; ///< OCAM SDK id.
111
112 long m_currImageNumber {-1}; ///< The current image number, retrieved from the image itself.
113
114 long m_lastImageNumber {-1}; ///< The last image number, saved from the last loop through.
115
116 bool m_protectionReset {false}; ///< Flag indicating that protection has been reset at least once.
117
118 unsigned m_protectionResetConfirmed {0}; ///< Counter indicating the number of times that the protection reset has been requested within 10 seconds, for confirmation.
119
120 double m_protectionResetReqTime {0}; ///< The time at which protection reset was requested. You have 10 seconds to confirm.
121
122 bool m_poweredOn {false};
123
124 ocamTemps m_temps; ///< Structure holding the last temperature measurement.
125
126 unsigned m_digitalBinX {1};
127
128 unsigned m_digitalBinY {1};
129
130 bool m_digitalBin {false};
131
132 mx::improc::eigenImage<int16_t> m_digitalBinWork;
133
134 std::string m_syncDevice {"fxngensync"};
135 std::string m_syncFreqProp {"C1freq"};
136
137 float m_syncFreq {0};
138
139public:
140
141 ///Default c'tor
142 ocam2KCtrl();
143
144 ///Destructor
146
147 /// Setup the configuration system (called by MagAOXApp::setup())
148 virtual void setupConfig();
149
150 /// load the configuration system results (called by MagAOXApp::setup())
151 virtual void loadConfig();
152
153 /// Startup functions
154 /** Sets up the INDI vars, and the f.g. thread.
155 *
156 */
157 virtual int appStartup();
158
159 /// Implementation of the FSM for the OCAM 2K.
160 virtual int appLogic();
161
162 /// Implementation of the on-power-off FSM logic
163 virtual int onPowerOff();
164
165 /// Implementation of the while-powered-off FSM
167
168 /// Do any needed shutdown tasks.
169 virtual int appShutdown();
170
171 /// Get the current device temperatures
172 /**
173 * \returns 0 on success
174 * \returns -1 on error
175 */
176 int getTemps();
177
178 /// Get the current frame rate.
179 /**
180 * \returns 0 on success
181 * \returns -1 on error
182 */
183 int getFPS();
184
185 /** \name stdCamera Interface
186 *
187 * @{
188 */
189
190 /// Set defaults for a power on state.
191 /**
192 * \returns 0 on success
193 * \returns -1 on error
194 */
195 int powerOnDefaults();
196
197 /// Turn temperature control on or off.
198 /** Sets temperature control on or off based on the current value of m_tempControlStatus
199 * \returns 0 on success
200 * \returns -1 on error
201 */
202 int setTempControl();
203
204 /// Set the CCD temperature setpoint [stdCamera interface].
205 /** Sets the temperature to m_ccdTempSetpt.
206 * \returns 0 on success
207 * \returns -1 on error
208 */
209 int setTempSetPt();
210
211 /// Set the frame rate. [stdCamera interface]
212 /** Sets the frame rate to m_fpsSet.
213 *
214 * \returns 0 on success
215 * \returns -1 on error
216 */
217 int setFPS();
218
219 /// Set the synchro state. [stdCamera interface]
220 /** Sets the synchro state to m_synchroSet.
221 *
222 * \returns 0 on success
223 * \returns -1 on error
224 */
225 int setSynchro();
226
227 /// Required by stdCamera, but this does not do anything for this camera [stdCamera interface]
228 /**
229 * \returns 0 always
230 */
231 int setExpTime();
232
233 /// Required by stdCamera, but this does not do anything for this camera [stdCamera interface]
234 /**
235 * \returns 0 always
236 */
237 int setNextROI();
238
239 /// Sets the shutter state, via call to dssShutter::setShutterState(int) [stdCamera interface]
240 /**
241 * \returns 0 always
242 */
243 int setShutter(int sh);
244
245 std::string stateString();
246
247 bool stateStringValid();
248
249 ///@}
250
251 /// Reset the EM Protection
252 /**
253 * \returns 0 on success
254 * \returns -1 on error
255 */
256 int resetEMProtection();
257
258 /// Get the current EM Gain.
259 /**
260 * \returns 0 on success
261 * \returns -1 on error
262 */
263 int getEMGain();
264
265 /// Set the EM gain.
266 /** Sets it to the value of stdCamera::m_emGainSet
267 *
268 * \returns 0 on success
269 * \returns -1 on error
270 */
271 int setEMGain();
272
273 /// Implementation of the framegrabber configureAcquisition interface
274 /** Sends the mode command over serial, sets the FPS, and initializes the OCAM SDK.
275 *
276 * \returns 0 on success
277 * \returns -1 on error
278 */
280
281 /// Implementation of the frameGrabber fps interface
282 /** Just returns the value of m_fps
283 */
284 float fps();
285
286 /// Implementation of the framegrabber startAcquisition interface
287 /** Initializes m_lastImageNumber, and calls edtCamera::pdvStartAcquisition
288 *
289 * \returns 0 on success
290 * \returns -1 on error
291 */
292 int startAcquisition();
293
294 /// Implementation of the framegrabber acquireAndCheckValid interface
295 /** Calls edtCamera::pdvAcquire, then analyzes the OCAM generated framenumber for skips and corruption.
296 *
297 * \returns 0 on success
298 * \returns -1 on error
299 */
301
302 /// Implementation of the framegrabber loadImageIntoStream interface
303 /** Conducts the OCAM descramble.
304 *
305 * \returns 0 on success
306 * \returns -1 on error
307 */
308 int loadImageIntoStream( void * dest /**< [in] */);
309
310 /// Implementation of the framegrabber reconfig interface
311 /** Locks the INDI mutex and calls edtCamera::pdvReconfig.
312 * \returns 0 on success
313 * \returns -1 on error
314 */
315 int reconfig();
316
317
318 //INDI:
320 //declare our properties
324
326
327public:
329
331
332 /** \name Telemeter Interface
333 *
334 * @{
335 */
336 int checkRecordTimes();
337
339
341
343
344 int recordTemps(bool force = false);
345
346 ///@}
347};
348
349inline
351{
352 //--- MagAOXApp Power Mgt. ---
353 m_powerMgtEnabled = true;
354 m_powerOnWait = 10;
355
356 //--- stdCamera ---
357 m_startupTemp = 20;
358
359 m_maxEMGain = 600;
360
361 return;
362}
363
364inline
366{
367 return;
368}
369
370inline
372{
374
376
377 config.add("camera.ocamDescrambleFile", "", "camera.ocamDescrambleFile", argType::Required, "camera", "ocamDescrambleFile", false, "string", "The path of the OCAM descramble file, relative to MagAOX/config.");
378
380
382
384}
385
386
387inline
389{
392
393 config(m_ocamDescrambleFile, "camera.ocamDescrambleFile");
394
395 if(m_maxEMGain < 1)
396 {
397 m_maxEMGain = 1;
398 log<text_log>("maxEMGain set to 1");
399 }
400
401 if(m_maxEMGain > 600)
402 {
403 m_maxEMGain = 600;
404 log<text_log>("maxEMGain set to 600");
405 }
406
410}
411
412inline
414{
415 REG_INDI_NEWPROP_NOCB(m_indiP_temps, "temps", pcf::IndiProperty::Number);
416 m_indiP_temps.add (pcf::IndiElement("cpu"));
417 m_indiP_temps["cpu"].set(0);
418 m_indiP_temps.add (pcf::IndiElement("power"));
419 m_indiP_temps["power"].set(0);
420 m_indiP_temps.add (pcf::IndiElement("bias"));
421 m_indiP_temps["bias"].set(0);
422 m_indiP_temps.add (pcf::IndiElement("water"));
423 m_indiP_temps["water"].set(0);
424 m_indiP_temps.add (pcf::IndiElement("left"));
425 m_indiP_temps["left"].set(0);
426 m_indiP_temps.add (pcf::IndiElement("right"));
427 m_indiP_temps["right"].set(0);
428 m_indiP_temps.add (pcf::IndiElement("cooling"));
429 m_indiP_temps["cooling"].set(0);
430
431 REG_INDI_NEWPROP_NOCB(m_indiP_emProt, "emProtection", pcf::IndiProperty::Text);
432 m_indiP_emProt.add(pcf::IndiElement("status"));
433 m_indiP_emProt["status"].set("UNKNOWN");
434 m_indiP_emProt.setState(INDI_IDLE);
435
436 createStandardIndiRequestSw( m_indiP_emProtReset, "emProtectionReset", "Reset", "EM Protection");
438
440
442 {
444 }
445
447 {
449 }
450
452 {
454 }
455
457 {
459 }
460
463 {
464 return log<software_error,-1>({__FILE__,__LINE__});
465 }
466
467 return 0;
468
469}
470
471
472
473inline
475{
476
477 //and run stdCamera's appLogic
479 {
480 return log<software_error, -1>({__FILE__, __LINE__});
481 }
482
483 //and run edtCamera's appLogic
485 {
486 return log<software_error, -1>({__FILE__, __LINE__});
487 }
488
489 //first run frameGrabber's appLogic to see if the f.g. thread has exited.
491 {
492 return log<software_error, -1>({__FILE__, __LINE__});
493 }
494
495 //and run dssShutter's appLogic
497 {
498 return log<software_error, -1>({__FILE__, __LINE__});
499 }
500
501 if( state() == stateCodes::POWERON) return 0;
502
504 {
506
507 std::string response;
508
509 //Might have gotten here because of a power off.
510 if(MagAOXAppT::m_powerState == 0) return 0;
511
512 int ret = pdvSerialWriteRead( response, "fps"); //m_pdv, "fps", m_readTimeout);
513 if( ret == 0)
514 {
516 }
517 else
518 {
519 sleep(1);
520 return 0;
521 }
522 }
523
525 {
526 //Get a lock
527 std::unique_lock<std::mutex> lock(m_indiMutex);
528
529 if( getFPS() == 0 )
530 {
533
534 if(m_poweredOn && m_ccdTempSetpt > -999)
535 {
536 m_poweredOn = false;
537 if(setTempSetPt() < 0)
538 {
539 if(powerState() != 1 || powerStateTarget() != 1) return 0;
541 }
542 }
543
544 //We always have to set synchro on connecting, b/c otherwise we don't know the state.
545 m_synchroSet = false;
546 if( setSynchro() != 0 )
547 {
548 log<software_error>({__FILE__, __LINE__, "error from setSynchro on CONNECT"});
549 }
550 }
551 else
552 {
553 if(powerState() != 1 || powerStateTarget() != 1) return 0;
556 }
557 }
558
560 {
561 //Get a lock if we can
562 std::unique_lock<std::mutex> lock(m_indiMutex, std::try_to_lock);
563
564 //but don't wait for it, just go back around.
565 if(!lock.owns_lock()) return 0;
566
567 if(getTemps() < 0)
568 {
569 if(powerState() != 1 || powerStateTarget() != 1) return 0;
572 return 0;
573 }
574
575 if(getFPS() < 0)
576 {
577 if(powerState() != 1 || powerStateTarget() != 1) return 0;
578
580 return 0;
581 }
582
584 {
585 if( mx::sys::get_curr_time() - m_protectionResetReqTime > 10.0)
586 {
588 updateIfChanged(m_indiP_emProt, "status", std::string("UNCONFIRMED"));
589 log<text_log>("protection reset request not confirmed", logPrio::LOG_NOTICE);
590 }
591 }
592
593 if(getEMGain () < 0)
594 {
595 if(MagAOXAppT::m_powerState == 0) return 0;
596
598 return 0;
599 }
600
602 {
605 return 0;
606 }
607
609 {
612 return 0;
613 }
614
616 {
619 return 0;
620 }
621
623 {
625 return 0;
626 }
627
628 }
629
630 ///\todo Fall through check?
631
632 return 0;
633
634}
635
636inline
638{
640
641 std::lock_guard<std::mutex> lock(m_indiMutex);
642
643 updateIfChanged(m_indiP_emProt, "status", std::string("UNKNOWN"), INDI_IDLE);
644
646
654
656 {
658 }
659
661 {
663 }
664
666 {
668 }
669
671 {
673 }
674
675 //Setting m_poweredOn
676 m_poweredOn = true;
677
678
679 return 0;
680}
681
682inline
684{
685 std::lock_guard<std::mutex> lock(m_indiMutex);
686
688 {
690 }
691
693 {
695 }
696
698 {
700 }
701
702 return 0;
703}
704
705inline
722
723
724inline
726{
727 std::string response;
728
729 if( pdvSerialWriteRead( response, "temp") == 0)
730 {
732
733 if(parseTemps( temps, response ) < 0)
734 {
735 if(powerState() != 1 || powerStateTarget() != 1) return -1;
736
740 m_tempControlStatus = false;
741 m_tempControlStatusStr = "UNKNOWN";
742
743 recordTemps();
744 recordCamera();
745
746 std::cerr << "Temp. parse error. Response:\n" << response << std::endl;
747
748 //We don't trust the temps, but don't reconfig just for this.
749 return log<software_error, 0>({__FILE__, __LINE__, "Temp. parse error"});
750 }
751
752 m_temps = temps;
753
754 //stdCamera temp control:
757
758 //Detect that temperature control is off
759 if(m_temps.COOLING_POWER < 5)
760 {
761 if( m_temps.CCD - m_temps.SET > 2.99 )
762 {
763 m_tempControlStatus = false;
764 }
765 }
766 else m_tempControlStatus = true;
767
768 if(m_tempControlStatus == true)
769 {
770 if(fabs(m_temps.CCD - m_temps.SET) < 1.0)
771 {
772 m_tempControlStatusStr = "ON TARGET";
774 }
775 else
776 {
777 m_tempControlStatusStr = "OFF TARGET";
778 m_tempControlOnTarget = false;
779 }
780 }
781 else
782 {
783 m_tempControlStatusStr = "TEMP OFF";
784 m_tempControlOnTarget = false;
785 }
786
787
788 //Telemeter:
789 recordTemps();
790 recordCamera();
791
799 return 0;
800
801 }
802 else
803 {
804 if(powerState() != 1 || powerStateTarget() != 1) return -1;
805 return log<software_error,-1>({__FILE__, __LINE__});
806 }
807}
808
809inline
811{
812 //Camera boots up with this true in most cases.
814 m_tempControlStatus =false;
815
816 return 0;
817}
818
819inline
821{
822 std::string response;
823
824 std::string command;
825
826 std::string comStr = "temp ";
828 {
829 command = "on";
831 m_tempControlStatus = true;
832 }
833 else
834 {
835 if( m_ccdTemp > 19) //19 is 20 with a 1 C slop
836 {
837 command = "off";
839 m_tempControlStatus = false;
840 }
841 else
842 {
843 return log<text_log,-1>("Can not turn temp control off when not at 20 C or higher", logPrio::LOG_ERROR);
844 }
845 }
846
847 comStr += command;
848
850 {
851 std::cerr << "response: " << response << "\n";
852 ///\todo check response
853 log<text_log,0>({"Set temperature control to " + command});
854 }
855 else
856 {
857 if(powerState() != 1 || powerStateTarget() != 1) return -1;
858 return log<software_error,-1>({__FILE__, __LINE__});
859 }
860
862 {
863 return setTempSetPt();
864 }
865
866 recordCamera();
867
868 return 0;
869}
870
871inline
873{
874 std::string response;
875
876 std::string tempStr = std::to_string( m_ccdTempSetpt );
877
878 ///\todo make more configurable
879 if(m_ccdTempSetpt >= 30 || m_ccdTempSetpt < -50)
880 {
881 return log<text_log,-1>({"attempt to set temperature outside valid range: " + tempStr}, logPrio::LOG_ERROR);
882 }
883
884 if( pdvSerialWriteRead( response, "temp " + tempStr) == 0)
885 {
886 std::cerr << "response: " << response << "\n";
887
888 recordCamera();
889
890 ///\todo check response
891 std::cerr << "temp " << tempStr << " response: " << response << "\n";
892
893 return log<text_log,0>({"set temperature: " + tempStr});
894 }
895 else
896 {
897 if(powerState() != 1 || powerStateTarget() != 1) return -1;
898 return log<software_error,-1>({__FILE__, __LINE__});
899 }
900
901}
902
903inline
905{
906 if(!m_synchro)
907 {
908 std::string response;
909
910 if( pdvSerialWriteRead( response, "fps") == 0) // m_pdv, "fps", m_readTimeout) == 0)
911 {
912 float fps;
913 if(parseFPS( fps, response ) < 0)
914 {
915 if(powerState() != 1 || powerStateTarget() != 1) return -1;
916
917 std::cerr << "fps parse error. Response:\n" << response << "\n";
918
919 return log<software_error, 0>({__FILE__, __LINE__, "fps parse error"});
920 }
921 m_fps = fps;
922
923 recordCamera();
924
925 return 0;
926
927 }
928 else
929 {
930 return log<software_error,-1>({__FILE__, __LINE__});
931 }
932 }
933 else
934 {
936 recordCamera();
937
938 return 0;
939 }
940}
941
942inline
944{
945 std::string fpsStr= std::to_string(m_fpsSet);
946
947 if(!m_synchro)
948 {
949 std::string response;
950
951 if( pdvSerialWriteRead( response, "fps " + fpsStr ) == 0)
952 {
953 ///\todo check response
954 std::cerr << "fps " << fpsStr << " response: " << response << "\n";
955 log<text_log>({"set fps: " + fpsStr});
956
957 //We always want to reset the latency circular buffers
958 ///\todo verify that this works!!
960 m_reconfig = true;
961
962 }
963 else
964 {
965 if(powerState() != 1 || powerStateTarget() != 1) return -1;
966 return log<software_error,-1>({__FILE__, __LINE__});
967 }
968 }
969 else
970 {
971 std::cerr << "setting: " << fpsStr << "\n";
972
973 pcf::IndiProperty ipFreq(pcf::IndiProperty::Number);
974
975 ipFreq.setDevice(m_syncDevice);
976 ipFreq.setName(m_syncFreqProp);
977 ipFreq.add(pcf::IndiElement("target"));
978
979 ipFreq["target"] = fpsStr;
980
982 }
983
984
985
986 return 0;
987}
988
989inline
991{
992 std::string response;
993
994 //First set the actual FPS to 0 to get to max
995 std::string fpsStr= std::to_string(0);
996 if( pdvSerialWriteRead( response, "fps " + fpsStr ) == 0)
997 {
998 ///\todo check response
999 std::cerr << "fps " << fpsStr << " response: " << response << "\n";
1000 log<text_log>({"set fps: " + fpsStr});
1001 }
1002 else
1003 {
1004 if(powerState() != 1 || powerStateTarget() != 1) return -1;
1005 return log<software_error,-1>({__FILE__, __LINE__});
1006 }
1007
1008 //Now actually turn synchro on
1009 std::string sStr;
1010 if(m_synchroSet) sStr = "on";
1011 else sStr = "off";
1012 if( pdvSerialWriteRead( response, "synchro " + sStr ) == 0)
1013 {
1014 ///\todo check response
1015 std::cerr << "synchro " << sStr << " resonse: " << response << "\n";
1016 log<text_log>({"set synchro: " + sStr});
1017
1019
1020 if(m_synchro == false)
1021 {
1022 updateSwitchIfChanged(m_indiP_synchro, "toggle", pcf::IndiElement::Off, INDI_IDLE);
1023 }
1024 else
1025 {
1026 updateSwitchIfChanged(m_indiP_synchro, "toggle", pcf::IndiElement::On, INDI_OK);
1027 }
1028 }
1029 else
1030 {
1031 if(powerState() != 1 || powerStateTarget() != 1) return -1;
1032 return log<software_error,-1>({__FILE__, __LINE__});
1033 }
1034
1035 //Finally we set the FPS of the synchro device
1036 return setFPS();
1037}
1038
1039inline
1041{
1042 return 0;
1043}
1044
1045inline
1047{
1048 return 0;
1049}
1050
1051inline
1056
1057inline
1059{
1060 std::string ss;
1061
1062 ss += m_modeName + "_";
1063 ss += std::to_string(m_fps) + "_";
1064 ss += std::to_string(m_emGain) + "_";
1065 ss += std::to_string(m_ccdTempSetpt);
1066
1067 return ss;
1068}
1069
1070inline
1072{
1074 {
1075 return true;
1076 }
1077 else return false;
1078}
1079
1080inline
1082{
1083 std::string response;
1084
1085 if( pdvSerialWriteRead( response, "protection reset") == 0)
1086 {
1087 std::cerr << "\n******************************************\n";
1088 std::cerr << "protection reset:\n";
1089 std::cerr << response << "\n";
1090 std::cerr << "\n******************************************\n";
1091 ///\todo check response.
1092
1093 updateIfChanged(m_indiP_emProt, "status", std::string("RESET"), INDI_OK);
1094
1095 log<text_log>("overillumination protection has been reset", logPrio::LOG_NOTICE);
1096
1098
1099 m_protectionReset = true;
1100
1101 return 0;
1102
1103 }
1104 else
1105 {
1106 if(powerState() != 1 || powerStateTarget() != 1) return -1;
1107 return log<software_error,-1>({__FILE__, __LINE__});
1108 }
1109}
1110
1111inline
1113{
1114 std::string response;
1115
1116 if( pdvSerialWriteRead( response, "gain") == 0)
1117 {
1118 unsigned emGain;
1119 if(parseEMGain( emGain, response ) < 0)
1120 {
1121 if(powerState() != 1 || powerStateTarget() != 1) return -1;
1122
1123 if(response.find("HV") != std::string::npos)
1124 {
1125 m_emGain = 1;
1126 updateIfChanged(m_indiP_emProt, "status", std::string("TRIPPED"), INDI_ALERT);
1127 return log<software_warning, -1>({__FILE__, __LINE__, "EM Gain tripped!"});
1128
1129 }
1130
1131 std::cerr << "EM Gain parse error, response:\n" << response << "\n";
1132
1133 return log<software_error, -1>({__FILE__, __LINE__, "EM Gain parse error"});
1134 }
1135
1136 m_emGain = emGain;
1137
1138 return 0;
1139
1140 }
1141 else
1142 {
1143 if(powerState() != 1 || powerStateTarget() != 1) return -1;
1144 return log<software_error,-1>({__FILE__, __LINE__});
1145 }
1146}
1147
1148inline
1150{
1151 std::string response;
1152
1153 if(m_protectionReset == false)
1154 {
1155 log<text_log>("Attempt to set EM gain before protection reset", logPrio::LOG_NOTICE);
1156
1157 if(m_emGainSet > 1) //we allow setting to 1 for safety
1158 {
1159 return 0;
1160 }
1161 }
1162
1163 unsigned emg = m_emGainSet; //a float
1164
1166 {
1167 log<text_log>("Attempt to set EM gain to " + std::to_string(emg) + " outside limits refused", logPrio::LOG_WARNING);
1168 return 0;
1169 }
1170
1171 std::string emgStr= std::to_string(emg);
1172 if( pdvSerialWriteRead( response, "gain " + emgStr ) == 0) //m_pdv, "gain " + emgStr, m_readTimeout) == 0)
1173 {
1174 ///\todo check response
1175 std::cerr << "gain " << emgStr << " response: " << emgStr << "\n";
1176
1177 log<text_log>({"set EM Gain: " + emgStr});
1178
1179 return 0;
1180 }
1181 else
1182 {
1183 if(powerState() != 1 || powerStateTarget() != 1) return -1;
1184 return log<software_error,-1>({__FILE__, __LINE__});
1185 }
1186
1187}
1188
1189inline
1191{
1192 //lock mutex
1193 std::unique_lock<std::mutex> lock(m_indiMutex);
1194
1195 //Send command to camera to place it in the correct mode
1196 std::string response;
1197 if( pdvSerialWriteRead( response, m_cameraModes[m_modeName].m_serialCommand) != 0) //m_pdv, m_cameraModes[m_modeName].m_serialCommand, m_readTimeout) != 0)
1198 {
1199 if(powerState() != 1 || powerStateTarget() != 1) return -1;
1200
1201 log<software_error>({__FILE__, __LINE__, "Error sending command to set mode"});
1202 sleep(1);
1203 return -1;
1204 }
1205
1206 m_currentROI.x = 119.5;
1207 m_currentROI.y = 119.5;
1208 m_currentROI.w = 240;
1209 m_currentROI.h = 240;
1210 m_currentROI.bin_x = m_cameraModes[m_modeName].m_binningX;
1211 m_currentROI.bin_y = m_cameraModes[m_modeName].m_binningY;
1212
1213 m_digitalBinX = m_cameraModes[m_modeName].m_digitalBinX;
1214 m_digitalBinY = m_cameraModes[m_modeName].m_digitalBinY;
1215
1216 if(m_digitalBinX > 1)
1217 {
1218 m_digitalBin = true;
1219 std::cerr << "digital binning!\n";
1220 }
1221 else
1222 {
1223 m_digitalBin = false;
1224 }
1225
1226 recordCamera();
1227
1228 ///\todo check response of pdvSerialWriteRead
1229 log<text_log>("camera configured with: " +m_cameraModes[m_modeName].m_serialCommand);
1230
1231 if(m_fpsSet > 0) setFPS();
1232
1233 log<text_log>("Send command to set mode: " + m_cameraModes[m_modeName].m_serialCommand);
1234 log<text_log>("Response was: " + response);
1235
1236 if(setSynchro() < 0)
1237 {
1238 log<software_error>({__FILE__, __LINE__, "Error setting synchro during configureAcquisition"});
1239 }
1240
1241 /* Initialize the OCAM2 SDK
1242 */
1243
1244 if(m_ocam2_id > 0)
1245 {
1247 }
1248 ocam2_rc rc;
1249 ocam2_mode mode;
1250
1251 int OCAM_SZw;
1252 int OCAM_SZh;
1253 if(m_raw_height == 121)
1254 {
1255 mode = OCAM2_NORMAL;
1256 OCAM_SZw = 240;
1257 OCAM_SZh = 240;
1258 }
1259 else if (m_raw_height == 62)
1260 {
1261 mode = OCAM2_BINNING;
1262 OCAM_SZw = 120;
1263 OCAM_SZh = 120;
1264 }
1265 else if (m_raw_height == 41)
1266 {
1267 mode = OCAM2_BINNING1x3;
1268 OCAM_SZw = 240;
1269 OCAM_SZh = 80;
1270 }
1271 else if (m_raw_height == 31)
1272 {
1273 mode = OCAM2_BINNING1x4;
1274 OCAM_SZw = 240;
1275 OCAM_SZh = 60;
1276 }
1277 else
1278 {
1279 log<text_log>("Unrecognized OCAM2 mode.", logPrio::LOG_ERROR);
1280 return -1;
1281 }
1282
1284
1285 std::cerr << "ocamDescrambleFile: " << ocamDescrambleFile << std::endl;
1286
1287 rc=ocam2_init(mode, ocamDescrambleFile.c_str(), &m_ocam2_id);
1288
1289 if (rc != OCAM2_OK)
1290 {
1291 if(powerState() != 1 || powerStateTarget() != 1) return -1;
1292 log<text_log>("ocam2_init error. Failed to initialize OCAM SDK with descramble file: " + ocamDescrambleFile, logPrio::LOG_ERROR);
1293 return -1;
1294 }
1295
1296
1297 log<text_log>("OCAM2K initialized. id: " + std::to_string(m_ocam2_id));
1298
1299 log<text_log>(std::string("OCAM2K mode is:") + ocam2_modeStr(ocam2_getMode(m_ocam2_id)));
1300
1301
1302
1303
1304 if(m_digitalBin)
1305 {
1306 std::cerr << "and digital binning!\n";
1308
1311 }
1312 else
1313 {
1314 m_width = OCAM_SZw;
1316 }
1317
1319
1321
1322 return 0;
1323}
1324
1325inline
1327{
1328 return m_fps;
1329}
1330
1331inline
1338
1339inline
1341{
1343
1344 /* Removed all pdv timeout and overrun checking, since we can rely on frame number from the camera
1345 to detect missed and corrupted frames.
1346
1347 See ef0dd24 for last version with full checks in it.
1348 */
1349
1350 //Get the image number to see if this is valid.
1351 //This is how it is in the ocam2_sdk:
1352 unsigned currImageNumber = (reinterpret_cast<int *>(m_image_p))[OCAM2_IMAGE_NB_OFFSET/4]; /* int offset */
1354
1355 //For the first loop after a restart
1356 if( m_lastImageNumber == -1 )
1357 {
1359 }
1360
1362 {
1363 //Detect exact condition of a wraparound on the unsigned int.
1364 // Yes, this can only happen once every 13.72 days at 3622 fps
1365 // But just in case . . .
1366 if(m_lastImageNumber != std::numeric_limits<unsigned int>::max() && m_currImageNumber != 0)
1367 {
1368 //The far more likely case is a problem...
1369
1370 //If a reasonably small number of frames skipped, then we trust the image number
1372 {
1373 //This we handle as a non-timeout -- report how many frames were skipped
1375
1376 log<text_log>("frames skipped: " + std::to_string(framesSkipped), logPrio::LOG_ERROR);
1377
1378 m_lastImageNumber = -1;
1380 m_reconfig = 1;
1381 return 1;
1382
1383 }
1384 else //but if it's any bigger or < 0, it's probably garbage
1385 {
1386 if(powerState() != 1 || powerStateTarget() != 1) return -1;
1387
1388 ///\todo need frame corrupt log type
1389 log<text_log>("frame number possibly corrupt: " + std::to_string(m_currImageNumber) + " - " + std::to_string(m_lastImageNumber), logPrio::LOG_ERROR);
1390
1392 m_reconfig = 1;
1393
1394 //Reset the counters.
1395 m_lastImageNumber = -1;
1396 return 1;
1397
1398 }
1399 }
1400 }
1402 return 0;
1403}
1404
1405inline
1407{
1408 unsigned currImageNumber = 0;
1409
1410 if(!m_digitalBin)
1411 {
1412 ocam2_descramble(m_ocam2_id, &currImageNumber, reinterpret_cast<int16_t *>(dest), reinterpret_cast<int16_t *>(m_image_p));
1413 //memcpy(dest, m_image_p, 120*120*2); //This is about 10 usec faster -- but we have to descramble.
1414 }
1415 else
1416 {
1418
1419 mx::improc::eigenMap<int16_t> destMap( reinterpret_cast<int16_t*>(dest), m_width, m_height);
1420
1421 if(m_digitalBinX > 1)
1422 {
1423 for(int cc = 0; cc < destMap.cols(); ++cc)
1424 {
1425 for(int rr = 0; rr < destMap.rows(); ++rr)
1426 {
1428
1429 for(unsigned b = 1; b < m_digitalBinX; ++b)
1430 {
1432 }
1433 }
1434 }
1435 }
1436 }
1437
1438
1439 return 0;
1440}
1441
1442inline
1444{
1445 //lock mutex
1446 std::unique_lock<std::mutex> lock(m_indiMutex);
1447
1449 if(rv < 0) return rv;
1451 return 0;
1452}
1453
1455{
1456 if(MagAOXAppT::m_powerState == 0) return 0;
1457
1458 if (ipRecv.getName() != m_indiP_emProtReset.getName())
1459 {
1460 log<software_error>({__FILE__,__LINE__, "wrong INDI property received."});
1461 return -1;
1462 }
1463
1464 if(!ipRecv.find("request"))
1465 {
1466 return 0;
1467 }
1468
1469 if( ipRecv["request"].getSwitchState() == pcf::IndiElement::Off )
1470 {
1471 return 0;
1472 }
1473
1474 std::unique_lock<std::mutex> lock(m_indiMutex);
1475
1477 {
1478 updateIfChanged(m_indiP_emProt, "status", std::string("CONFIRM"), INDI_BUSY);
1479
1481
1482 m_protectionResetReqTime = mx::sys::get_curr_time();
1483
1484 log<text_log>("protection reset requested", logPrio::LOG_NOTICE);
1485
1486 return 0;
1487 }
1488
1489
1490 //If here, this is a confirmation.
1491 return resetEMProtection();
1492}
1493
1495{
1497
1498 if(!ipRecv.find("current"))
1499 {
1500 return -1;
1501 }
1502
1503 m_syncFreq = ipRecv["current"].get<double>();
1504
1505 if(m_synchro && m_syncFreq != m_fps)
1506 {
1507 recordCamera(true);
1508 m_fps = m_syncFreq;
1509
1510 //We always want to reset the latency circular buffers
1512 m_reconfig = true;
1513 }
1514
1515 return 0;
1516}
1517
1518inline
1523
1524inline
1526{
1527 return recordTemps(true);
1528}
1529
1530inline
1532{
1533 return recordCamera(true);
1534}
1535
1536inline
1538{
1539 return recordFGTimings(true);
1540}
1541
1542inline
1544{
1545 static ocamTemps lastTemps;
1546
1547 if(!(lastTemps == m_temps) || force)
1548 {
1551 }
1552
1553 return 0;
1554}
1555
1556}//namespace app
1557} //namespace MagAOX
1558
1559#endif
The base-class for MagAO-X applications.
Definition MagAOXApp.hpp:73
void updateIfChanged(pcf::IndiProperty &p, const std::string &el, const T &newVal, pcf::IndiProperty::PropertyStateType ipState=pcf::IndiProperty::Ok)
Update an INDI property element value if it has changed.
int createStandardIndiRequestSw(pcf::IndiProperty &prop, const std::string &name, const std::string &label="", const std::string &group="")
Create a standard R/W INDI switch with a single request element.
stateCodes::stateCodeT state()
Get the current state code.
int registerIndiPropertyNew(pcf::IndiProperty &prop, int(*)(void *, const pcf::IndiProperty &))
Register an INDI property which is exposed for others to request a New Property for.
int powerState()
Returns the current power state.
int m_powerState
Current power state, 1=On, 0=Off, -1=Unk.
int powerStateTarget()
Returns the target power state.
void updateSwitchIfChanged(pcf::IndiProperty &p, const std::string &el, const pcf::IndiElement::SwitchStateType &newVal, pcf::IndiProperty::PropertyStateType ipState=pcf::IndiProperty::Ok)
Update an INDI switch element value if it has changed.
static int log(const typename logT::messageT &msg, logPrioT level=logPrio::LOG_DEFAULT)
Make a log entry.
std::string m_configDir
The path to configuration files for MagAOX.
Definition MagAOXApp.hpp:85
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.
int sendNewProperty(const pcf::IndiProperty &ipSend, const std::string &el, const T &newVal)
Send a newProperty command to another device (using the INDI Client interface)
MagAO-X Uniblitz DSS Shutter interface.
void loadConfig(mx::app::appConfigurator &config)
load the configuration system results
int appShutdown()
applogic shutdown
void setupConfig(mx::app::appConfigurator &config)
Setup the configuration system.
MagAO-X EDT framegrabber interface.
Definition edtCamera.hpp:57
u_char * m_image_p
The image data grabbed.
Definition edtCamera.hpp:80
int m_raw_height
The height of the frame, according to the framegrabber.
Definition edtCamera.hpp:86
int pdvSerialWriteRead(std::string &response, const std::string &command)
Send a serial command over cameralink and retrieve the response.
void setupConfig(mx::app::appConfigurator &config)
Setup the configuration system.
void loadConfig(mx::app::appConfigurator &config)
load the configuration system results
int appShutdown()
Application the shutdown.
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.
int loadConfig(mx::app::appConfigurator &config)
load the configuration system results
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.
int setupConfig(mx::app::appConfigurator &config)
Setup the configuration system.
MagAO-X standard camera interface.
float m_maxEMGain
The configurable maximum EM gain. To be enforced in derivedT.
float m_fpsSet
The commanded fps, as set by user.
bool m_synchroSet
Target status of m_synchro.
std::string m_tempControlStatusStr
Camera specific description of temperature control status.
float m_emGain
The camera's current EM gain (if available).
float m_emGainSet
The camera's EM gain, as set by the user.
std::string m_nextMode
The mode to be set by the next reconfiguration.
float m_ccdTempSetpt
The desired temperature, in C.
bool m_tempControlStatus
Whether or not temperature control is active.
int setupConfig(mx::app::appConfigurator &config)
Setup the configuration system.
std::string m_modeName
The current mode name.
float m_startupTemp
The temperature to set after a power-on. Set to <= -999 to not use [default].
int loadConfig(mx::app::appConfigurator &config)
load the configuration system results
bool m_synchro
Status of synchronization, true is on, false is off.
float m_ccdTemp
The current temperature, in C.
bool m_tempControlOnTarget
Whether or not the temperature control system is on its target temperature.
cameraConfigMap m_cameraModes
Map holding the possible camera mode configurations.
int appShutdown()
Application shutdown.
bool m_tempControlStatusSet
Desired state of temperature control.
static constexpr bool c_stdCamera_fps
app::dev config to tell stdCamera not to expose FPS status (ignored since fpsCtrl==true)
pcf::IndiProperty m_indiP_temps
static constexpr bool c_stdCamera_usesModes
app:dev config to tell stdCamera not to expose mode controls
int resetEMProtection()
Reset the EM Protection.
bool m_protectionReset
Flag indicating that protection has been reset at least once.
int setShutter(int sh)
Sets the shutter state, via call to dssShutter::setShutterState(int) [stdCamera interface].
mx::improc::eigenImage< int16_t > m_digitalBinWork
virtual int appLogic()
Implementation of the FSM for the OCAM 2K.
int recordTelem(const ocam_temps *)
static constexpr bool c_stdCamera_usesStateString
app::dev confg to tell stdCamera to expose the state string property
static constexpr bool c_stdCamera_exptimeCtrl
app::dev config to tell stdCamera not to expose exposure time controls
int setNextROI()
Required by stdCamera, but this does not do anything for this camera [stdCamera interface].
virtual void setupConfig()
Setup the configuration system (called by MagAOXApp::setup())
pcf::IndiProperty m_indiP_emProtReset
~ocam2KCtrl() noexcept
Destructor.
virtual int onPowerOff()
Implementation of the on-power-off FSM logic.
std::string m_ocamDescrambleFile
Path the OCAM 2K pixel descrambling file, relative to MagAO-X config directory.
ocamTemps m_temps
Structure holding the last temperature measurement.
static constexpr bool c_edtCamera_relativeConfigPath
app::dev config to tell edtCamera to use relative path to camera config file
int powerOnDefaults()
Set defaults for a power on state.
int setSynchro()
Set the synchro state. [stdCamera interface].
virtual int appShutdown()
Do any needed shutdown tasks.
static constexpr bool c_stdCamera_synchro
app::dev config to tell stdCamera to expose synchro mode controls
int getEMGain()
Get the current EM Gain.
unsigned m_protectionResetConfirmed
Counter indicating the number of times that the protection reset has been requested within 10 seconds...
virtual void loadConfig()
load the configuration system results (called by MagAOXApp::setup())
int setExpTime()
Required by stdCamera, but this does not do anything for this camera [stdCamera interface].
float fps()
Implementation of the frameGrabber fps interface.
int loadImageIntoStream(void *dest)
Implementation of the framegrabber loadImageIntoStream interface.
int setTempControl()
Turn temperature control on or off.
pcf::IndiProperty m_indiP_emProt
int reconfig()
Implementation of the framegrabber reconfig interface.
static constexpr bool c_stdCamera_emGain
app::dev config to tell stdCamera to expose EM gain controls
long m_currImageNumber
The current image number, retrieved from the image itself.
int setTempSetPt()
Set the CCD temperature setpoint [stdCamera interface].
static constexpr bool c_stdCamera_vShiftSpeed
app:dev config to tell stdCamera not to expose vertical shift speed control
int startAcquisition()
Implementation of the framegrabber startAcquisition interface.
static constexpr bool c_stdCamera_tempControl
app::dev config to tell stdCamera to expose temperature controls
int setEMGain()
Set the EM gain.
int setFPS()
Set the frame rate. [stdCamera interface].
static constexpr bool c_stdCamera_temp
app::dev config to tell stdCamera to expose temperature (ignored since tempControl==true)
static constexpr bool c_stdCamera_usesROI
app:dev config to tell stdCamera to expose ROI controls
int recordTemps(bool force=false)
int getFPS()
Get the current frame rate.
int configureAcquisition()
Implementation of the framegrabber configureAcquisition interface.
double m_protectionResetReqTime
The time at which protection reset was requested. You have 10 seconds to confirm.
ocam2KCtrl()
Default c'tor.
int acquireAndCheckValid()
Implementation of the framegrabber acquireAndCheckValid interface.
static constexpr bool c_stdCamera_fpsCtrl
app::dev config to tell stdCamera to expose FPS controls
static constexpr bool c_stdCamera_cropMode
app:dev config to tell stdCamera to expose Crop Mode controls
static constexpr bool c_frameGrabber_flippable
app:dev config to tell framegrabber these images can not be flipped
static constexpr bool c_stdCamera_readoutSpeed
app::dev config to tell stdCamera not to expose readout speed controls
ocam2_id m_ocam2_id
OCAM SDK id.
virtual int whilePowerOff()
Implementation of the while-powered-off FSM.
long m_lastImageNumber
The last image number, saved from the last loop through.
int getTemps()
Get the current device temperatures.
virtual int appStartup()
Startup functions.
static constexpr bool c_stdCamera_hasShutter
app:dev config to tell stdCamera to expose shutter controls
pcf::IndiProperty m_indiP_syncFreq
#define INDI_NEWCALLBACK_DEFN(class, prop)
Define the callback for a new property request.
#define REG_INDI_NEWPROP_NOCB(prop, propName, type)
Register a NEW INDI property with the class, with no callback.
#define INDI_NEWCALLBACK(prop)
Get the name of the static callback wrapper for a new property.
#define INDI_SETCALLBACK_DECL(class, prop)
Declare the callback for a set property request, and declare and define the static wrapper.
#define INDI_SETCALLBACK_DEFN(class, prop)
Define the callback for a set property request.
#define REG_INDI_SETPROP(prop, devName, propName)
Register a SET INDI property with the class, using the standard callback name.
#define INDI_NEWCALLBACK_DECL(class, prop)
Declare the callback for a new property request, and declare and define the static wrapper.
@ OPERATING
The device is operating, other than homing.
@ 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.
#define INDI_VALIDATE_CALLBACK_PROPS(prop1, prop2)
Standard check for matching INDI properties in a callback.
#define INDI_IDLE
Definition indiUtils.hpp:27
#define INDI_BUSY
Definition indiUtils.hpp:29
#define INDI_ALERT
Definition indiUtils.hpp:30
#define INDI_OK
Definition indiUtils.hpp:28
int parseFPS(float &fps, const std::string &fstr)
Parse the FPS response.
Definition ocamUtils.hpp:94
int parseTemps(ocamTemps &temps, const std::string &tstr)
Parse the OCAM temp query and fill the ocamTemps structure.
Definition ocamUtils.hpp:64
const pcf::IndiProperty & ipRecv
int parseEMGain(unsigned &emGain, const std::string &fstr)
Parse the EM gain response.
std::unique_lock< std::mutex > lock(m_indiMutex)
Definition dm.hpp:24
static constexpr logPrioT LOG_NOTICE
A normal but significant condition.
static constexpr logPrioT LOG_WARNING
A condition has occurred which may become an error, but the process continues.
static constexpr logPrioT LOG_ERROR
An error has occured which the software will attempt to correct.
MagAOX::app::MagAOXApp< true > MagAOXAppT
ocam2_rc ocam2_init(ocam2_mode mode, const char *descrbFile, ocam2_id *id)
Create a camera instance with the provided mode.
Definition ocam2_sdk.c:431
void ocam2_descramble(ocam2_id id, unsigned int *number, short *image, const short *imageRaw)
Create a camera instance with the provided mode.
Definition ocam2_sdk.c:418
const char * ocam2_modeStr(ocam2_mode mode)
Return a description text for camera mode.
Definition ocam2_sdk.c:77
ocam2_rc ocam2_exit(ocam2_id id)
Clear a camera instance.
Definition ocam2_sdk.c:490
ocam2_mode ocam2_getMode(ocam2_id id)
Return the camera mode.
Definition ocam2_sdk.c:69
The purpose of the libocam2sdk library is to provide an easy way to achieve ocam2 specific opération....
int ocam2_id
Library camera identifier.
Definition ocam2_sdk.h:268
ocam2_rc
Enum of ocam2 library return code.
Definition ocam2_sdk.h:258
@ OCAM2_OK
Definition ocam2_sdk.h:260
@ OCAM2_BINNING
Definition ocam2_sdk.h:236
@ OCAM2_BINNING1x3
Definition ocam2_sdk.h:244
@ OCAM2_BINNING1x4
Definition ocam2_sdk.h:246
@ OCAM2_NORMAL
Definition ocam2_sdk.h:226
enum workMode ocam2_mode
typedef of ocam2 camera mode
#define OCAM2_IMAGE_NB_OFFSET
Definition ocam2_sdk.h:208
Utilities for the OCAM camera.
A device base class which saves telemetry.
Definition telemeter.hpp:69
int appShutdown()
Perform telemeter application shutdown.
int loadConfig(appConfigurator &config)
Load the device section from an application configurator.
int setupConfig(appConfigurator &config)
Setup an application configurator for the device section.
Structure to hold the OCAM camera temperature readings returned by the device.
Definition ocamUtils.hpp:20
float WATER
Cooling water temperature.
Definition ocamUtils.hpp:25
int setInvalid()
Set all values to the invalid value, -999.
Definition ocamUtils.hpp:43
float COOLING_POWER
the cooling power in 100 mw.
Definition ocamUtils.hpp:29
float BIAS
Bias temperature.
Definition ocamUtils.hpp:24
float SET
The CCD set temeperature.
Definition ocamUtils.hpp:28
float CCD
The detector temperature.
Definition ocamUtils.hpp:21
float POWER
Power supply temperature.
Definition ocamUtils.hpp:23
float CPU
The CPU temperature.
Definition ocamUtils.hpp:22
float LEFT
The left amplifier temperature.
Definition ocamUtils.hpp:26
float RIGHT
The right amplifier temperature.
Definition ocamUtils.hpp:27
Log entry recording the build-time git state.
Software CRITICAL log entry.
Software ERR log entry.
Software WARN log entry.
Log entry recording framegrabber timings.
Log entry recording stdcam stage specific status.
A simple text log, a string-type log.
Definition text_log.hpp:24