API
 
Loading...
Searching...
No Matches
stdCamera.hpp
Go to the documentation of this file.
1/** \file stdCamera.hpp
2 * \brief Standard camera interface
3 *
4 * \author Jared R. Males (jaredmales@gmail.com)
5 *
6 * \ingroup app_files
7 */
8
9#ifndef stdCamera_hpp
10#define stdCamera_hpp
11
12#include <string>
13#include <unordered_map>
14
15#include <mx/app/application.hpp>
16
17#include "../MagAOXApp.hpp"
18
19namespace MagAOX
20{
21namespace app
22{
23namespace dev
24{
25
26#define CAMCTRL_E_NOCONFIGS ( -10 )
27
28/// A camera configuration
29/** a.k.a. a mode
30 */
32{
33 std::string m_configFile; ///< The file to use for this mode, e.g. an EDT configuration file.
34 std::string m_serialCommand; ///< The command to send to the camera to place it in this mode.
35 unsigned m_centerX{ 0 };
36 unsigned m_centerY{ 0 };
37 unsigned m_sizeX{ 0 };
38 unsigned m_sizeY{ 0 };
39 unsigned m_binningX{ 0 };
40 unsigned m_binningY{ 0 };
41
42 unsigned m_digitalBinX{ 0 };
43 unsigned m_digitalBinY{ 0 };
44
45 float m_maxFPS{ 0 };
46};
47
48typedef std::unordered_map<std::string, cameraConfig> cameraConfigMap;
49
50/// Load the camera configurations contained in the app configuration into a map
51int loadCameraConfig( cameraConfigMap &ccmap, ///< [out] the map in which to place the configurations found in config
52 mx::app::appConfigurator &config ///< [in] the application configuration structure
53);
54
55/// MagAO-X standard camera interface
56/** Implements the standard interface to a MagAO-X camera. The derived class `derivedT` must
57 * meet the following requirements:
58 *
59 * - The derived class `derivedT` must be a `MagAOXApp<true>`
60 *
61 * - Must declare this class a friend like so:
62 * \code
63 * friend class dev::stdCamera<DERIVEDNAME>; //replace DERIVEDNAME with derivedT class name
64 * \endcode
65 *
66 * - Must declare the following typedef:
67 * \code
68 * typedef dev::stdCamera<DERIVEDNAME> stdCameraT; //replace DERIVEDNAME with derivedT class name
69 * \endcode
70 *
71 * - Must declare a series of `static constexpr` flags to manage static compile-time configuration. Each of these
72 * flags must be defined in derivedT to be either true or false.
73 *
74 * - Temperature Control and Status:
75 *
76 * - A static configuration variable must be defined in derivedT as
77 * \code
78 * static constexpr bool c_stdCamera_tempControl = true; //or: false
79 * \endcode
80 * which determines whether or not temperature controls are exposed.
81 *
82 * - If `(c_stdCamera_tempControl == true)` then the derived class must implement the following interfaces
83 * \code
84 * int setTempControl(); // set temp control status according to m_tempControlStatusSet
85 * int setTempSetPt(); // set the temperature set point accordin to m_ccdTempSetpt
86 * \endcode
87 *
88 * - A static configuration variable must be defined in derivedT as
89 * \code
90 * static constexpr bool c_stdCamera_temp = true; //or: false
91 * \endcode
92 * which determines whether or not temperature reporting is exposed. Note that if
93 * `(c_stdCamera_tempControl == true)`, then the behavior is as if `c_stdCamera_temp == true`,
94 * but thus constexpr must still be defined.
95 *
96 * - If either `(c_stdCamera_tempControl == true)` or `c_stdCamera_temp == true` then the INDI property
97 * "temp_ccd" will be updated from the value of \ref m_ccdTemp.
98 *
99 * - Readout Speed:
100 *
101 * - A static configuration variable must be defined in derivedT as
102 * \code
103 * static constexpr bool c_stdCamera_readoutSpeed = true; //or: false
104 * \endcode
105 * which determines whether or not readout speed controls are exposed. If true, then the implementation should
106 * populate
107 * \ref m_readoutSpeedNames and \ref m_readoutSpeedNameLabels (vectors of strings) on construction to the
108 * allowed values. This facility is normally used to control both amplifier and readout/adc speed with names like
109 * "ccd_1MHz" and "emccd_17MHz".
110 *
111 * - If used (and true) then the following interface must be implemented:
112 * \code
113 * int setReadoutSpeed(); // configures camera using m_readoutSpeedNameSet
114 * \endcode
115 * This configures the camera according to \ref m_readoutSpeedNameSet.
116 * The implementation must also manage \ref m_readoutSpeedName, keeping it up to date with the current setting.
117 *
118 * - If true, the configuration setting "camera.defaultReadoutSpeed"
119 * is also exposed, and \ref m_defaultReadoutSpeed will be set according to it. The implementation can
120 * set a sensible default on construction.
121 *
122 * - Vertical Shift Speed:
123 *
124 * - A static configuration variable must be defined in derivedT as
125 * \code
126 * static constexpr bool c_stdCamera_vShiftSpeed = true; //or: false
127 * \endcode
128 * which determines whether or not vertical shift speed controls are exposed.
129 *
130 * - If true, then the implementation should populate \ref m_vShiftSpeedNames and \ref m_vShiftSpeedLabels
131 * (vectors of strings) on construction to the allowed values. This
132 * facility is normally used with names like "0_3us" and "1_3us".
133 *
134 * - If true then the following interface must be defined:
135 * \code
136 * int setVShiftSpeed(); // configures camera according to m_vShiftSpeedNameSet
137 * \endcode
138 * function must be defined which sets the camera according to \ref m_vShiftSpeedNameSet.
139 * The implementation must also manage m_vShiftSpeedName, keeping it up to date.
140 *
141 * - The configuration setting "camera.defaultVShiftSpeed"
142 * is also exposed, and \ref m_defaultVShiftSpeed will be set accordingly. derivedT can set a sensible default
143 * on constuction.
144 *
145 * - Exposure Time:
146 * - A static configuration variable must be defined in derivedT as
147 * \code
148 * static constexpr bool c_stdCamera_exptimeCtrl = true; //or: false
149 * \endcode
150 * - If true, the following interface must be implemented:
151 * \code
152 * int setExpTime(); // set camera exposure time according to m_expTimeSet.
153 * \endcode
154 * to configure the camera according to \ref m_expTimeSet. derivedT must also keep \ref m_expTime up to date.
155 *
156 * - Frames per Second (FPS) Control and Status:
157 * - A static configuration variable must be defined in derivedT as
158 * \code
159 * static constexpr bool c_stdCamera_fpsCtrl = true; //or: false
160 * \endcode
161 *
162 * - If that is set to true the derivedT must implement
163 * \code
164 * int setFPS(); // set camera FS according to m_fps
165 * \endcode
166 * to configure the camera according to \ref m_fpsSet.
167 *
168 * - A static configuration variable must be defined in derivedT as
169 * \code
170 * static constexpr bool c_stdCamera_fps = true; //or: false
171 * \endcode
172 * Note that the value of c_stdCamera_fps does not matter if c_stdCamera_fpsCtrl == true.
173 *
174 * - If either `c_stdCamera_fpsCtrl == true` or `c_stdCamera_fps == true` then derivedT must also
175 * keep \ref m_fps up to date.
176 *
177 * - Synchro Control:
178 *
179 * - A static configuration variable must be defined in derivedT as
180 * \code
181 * static constexpr bool c_stdCamera_synchro = true; //or: false
182 * \endcode
183 * - If that is set to true the derivedT must implement
184 * \code
185 * int setSynchro(); // configure the camera based m_synchroSet.
186 * \endcode
187 * to configure the camera based on \ref m_synchroSet. The implementation should also keep
188 * \ref m_synchro up to date.
189 *
190 * - EM Gain:
191 * - A static configuration variable must be defined in derivedT as
192 * \code
193 * static constexpr bool c_stdCamera_emGain = true; //or: false
194 * \endcode
195 * which determines whether or not EM gain controls are exposed.
196 *
197 * - If the camera uses EM Gain, then a function
198 * \code
199 * int setEMGain(); // set EM gain based on m_emGainSet.
200 * \endcode
201 * must be defined which sets the camera EM Gain to \ref m_emGainSet.
202 *
203 * - If true the implementation must keep \ref m_emGain up to date.
204 *
205 * - If true the value of \ref m_maxEMGain should be set by the implementation and managed
206 * as needed. Additionally the configuration setting "camera.maxEMGain" is exposed.
207 *
208 * - Camera Modes:
209 *
210 * - A static configuration variable must be defined in derivedT as
211 * \code
212 * static constexpr bool c_stdCamera_usesModes= true; //or: false
213 * \endcode
214 *
215 * - If true, then modes are read from the configuration file. See \ref loadCameraConfig()
216 *
217 * - If true, then the configuration setting "camera.startupMode" is exposed, which sets the mode at startup by
218 * its name.
219 *
220 * - Regions of Interest
221 *
222 * - A static configuration variable must be defined in derivedT as
223 * \code
224 * static constexpr bool c_stdCamera_usesROI = true; //or: false
225 * \endcode
226 *
227 * - The default values of m_currentROI should be set before calling stdCamera::appStartup().
228 *
229 * - The derived class must implement:
230 * \code
231 * int checkNextROI(); // verifies m_nextROI values and modifies to closest valid values if needed
232 * int setNextROI(); // sets the ROI to the new target values.
233 * \endcode
234 *
235 * - Crop Mode ROIs:
236 *
237 * - A static configuration variable must be defined in derivedT as
238 * \code
239 * static constexpr bool c_stdCamera_cropMode = true; //or: false
240 * \endcode
241 *
242 * - If true the derived class must implement
243 * \code
244 * int setCropMode(); // set crop mode according to m_cropModeSet
245 * \endcode
246 * which changes the crop mode according to \ref m_cropModeSet.
247 *
248 * - `derivedT` must also maintain the value of \ref m_cropMode.
249 *
250 * - Shutters:
251 *
252 * - A static configuration variable must be defined in derivedT as
253 * \code
254 * static constexpr bool c_stdCamera_hasShutter = true; //or: false
255 * \endcode
256 *
257 * - If true the following interface must be implemented:
258 * \code
259 * int setShutter(int); // shut the shutter if 0, open the shutter otherwise.
260 * \endcode
261 * which shuts the shutter if the argument is 0, opens it otherwise.
262 *
263 * - State:
264 *
265 * - A static configuration variable must be defined in derivedT as
266 * \code
267 * static constexpr bool c_stdCamera_usesStateString = true; //or: false
268 * \endcode
269 * which determines whether the class provides a state string for dark management.
270 *
271 * - If true, the following functions must be defined in derivedT:
272 * \code
273 * std::string stateString(); //String capturing the current state. Must not include "__T".
274 * bool stateStringValid(); //Whether or not the current state string is valid, i.e. not changing.
275 * \endcode
276 *
277 *
278 * - The derived class must implement:
279 * \code
280 * int powerOnDefaults(); // called on power-on after powerOnWaitElapsed has occurred.
281 * \endcode
282 *
283 * - Calls to this class's setupConfig(), loadConfig(), appStartup(), appLogic(), appShutdown()
284 * onPowerOff(), and whilePowerOff(), must be placed in the derived class's functions of the same name.
285 *
286 * \ingroup appdev
287 */
288template <class derivedT>
290{
291 protected:
292 /** \name Configurable Parameters
293 * @{
294 */
295
296 cameraConfigMap m_cameraModes; ///< Map holding the possible camera mode configurations
297
298 std::string m_startupMode; ///< The camera mode to load during first init after a power-on.
299
300 float m_startupTemp{ -999 }; ///< The temperature to set after a power-on. Set to <= -999 to not use [default].
301
302 std::string m_defaultReadoutSpeed; ///< The default readout speed of the camera.
303 std::string m_defaultVShiftSpeed; ///< The default readout speed of the camera.
304
305 ///@}
306
307 /** \name Temperature Control Interface
308 * @{
309 */
310
311 float m_minTemp{ -60 };
312 float m_maxTemp{ 30 };
313 float m_stepTemp{ 0 };
314
315 float m_ccdTemp{ -999 }; ///< The current temperature, in C
316
317 float m_ccdTempSetpt{ -999 }; ///< The desired temperature, in C
318
319 bool m_tempControlStatus{ false }; ///< Whether or not temperature control is active
320 bool m_tempControlStatusSet{ false }; ///< Desired state of temperature control
321
322 bool m_tempControlOnTarget{ false }; ///< Whether or not the temperature control system is on its target temperature
323
324 std::string m_tempControlStatusStr; ///< Camera specific description of temperature control status.
325
326 pcf::IndiProperty m_indiP_temp;
327 pcf::IndiProperty m_indiP_tempcont;
328 pcf::IndiProperty m_indiP_tempstat;
329
330 ///@}
331
332 /** \name Readout Control
333 * @{
334 */
335
336 std::vector<std::string> m_readoutSpeedNames;
337 std::vector<std::string> m_readoutSpeedNameLabels;
338
339 std::string m_readoutSpeedName; ///< The current readout speed name
340 std::string m_readoutSpeedNameSet; ///< The user requested readout speed name, to be set by derived()
341
342 std::vector<std::string> m_vShiftSpeedNames;
343 std::vector<std::string> m_vShiftSpeedNameLabels;
344
345 std::string m_vShiftSpeedName; ///< The current vshift speed name
346 std::string m_vShiftSpeedNameSet; ///< The user requested vshift speed name, to be set by derived()
347
348 float m_adcSpeed{ 0 };
349 float m_vshiftSpeed{ 0 };
350
351 float m_emGain{ 1 }; ///< The camera's current EM gain (if available).
352 float m_emGainSet{ 1 }; ///< The camera's EM gain, as set by the user.
353 float m_maxEMGain{ 1 }; ///< The configurable maximum EM gain. To be enforced in derivedT.
354
355 pcf::IndiProperty m_indiP_readoutSpeed;
356 pcf::IndiProperty m_indiP_vShiftSpeed;
357
358 pcf::IndiProperty m_indiP_emGain;
359
360 ///@}
361
362 /** \name Exposure Control
363 * @{
364 */
365 float m_minExpTime{ 0 }; ///< The minimum exposure time, used for INDI attributes
366 float m_maxExpTime{ std::numeric_limits<float>::max() }; ///< The maximum exposure time, used for INDI attributes
367 float m_stepExpTime{ 0 }; ///< The maximum exposure time stepsize, used for INDI attributes
368
369 float m_expTime{ 0 }; ///< The current exposure time, in seconds.
370 float m_expTimeSet{ 0 }; ///< The exposure time, in seconds, as set by user.
371
372 float m_minFPS{ 0 }; ///< The minimum FPS, used for INDI attributes
373 float m_maxFPS{ std::numeric_limits<float>::max() }; ///< The maximum FPS, used for INDI attributes
374 float m_stepFPS{ 0 }; ///< The FPS step size, used for INDI attributes
375
376 float m_fps{ 0 }; ///< The current FPS.
377 float m_fpsSet{ 0 }; ///< The commanded fps, as set by user.
378
379 pcf::IndiProperty m_indiP_exptime;
380
381 pcf::IndiProperty m_indiP_fps;
382
383 ///@}
384
385 /** \name External Synchronization
386 * @{
387 */
388 bool m_synchroSet{ false }; ///< Target status of m_synchro
389
390 bool m_synchro{ false }; ///< Status of synchronization, true is on, false is off.
391
392 pcf::IndiProperty m_indiP_synchro;
393
394 ///@}
395
396 /** \name Modes
397 *
398 * @{
399 */
400 std::string m_modeName; ///< The current mode name
401
402 std::string m_nextMode; ///< The mode to be set by the next reconfiguration
403
404 pcf::IndiProperty m_indiP_mode; ///< Property used to report the current mode
405
406 pcf::IndiProperty
407 m_indiP_reconfig; ///< Request switch which forces the framegrabber to go through the reconfigure process.
408
409 ///@}
410
411 /** \name ROIs
412 * ROI controls are exposed if derivedT::c_stdCamera_usesROI==true
413 * @{
414 */
415 struct roi
416 {
417 float x{ 0 };
418 float y{ 0 };
419 int w{ 0 };
420 int h{ 0 };
421 int bin_x{ 0 };
422 int bin_y{ 0 };
423 };
424
428
429 float m_minROIx{ 0 };
430 float m_maxROIx{ 1023 };
431 float m_stepROIx{ 0 };
432
433 float m_minROIy{ 0 };
434 float m_maxROIy{ 1023 };
435 float m_stepROIy{ 0 };
436
438 int m_maxROIWidth{ 1024 };
440
442 int m_maxROIHeight{ 1024 };
444
448
452
453 float m_default_x{ 0 }; ///< Power-on ROI center x coordinate.
454 float m_default_y{ 0 }; ///< Power-on ROI center y coordinate.
455 int m_default_w{ 0 }; ///< Power-on ROI width.
456 int m_default_h{ 0 }; ///< Power-on ROI height.
457 int m_default_bin_x{ 1 }; ///< Power-on ROI x binning.
458 int m_default_bin_y{ 1 }; ///< Power-on ROI y binning.
459
460 float m_full_x{ 0 }; ///< The full ROI center x coordinate.
461 float m_full_y{ 0 }; ///< The full ROI center y coordinate.
462 int m_full_w{ 0 }; ///< The full ROI width.
463 int m_full_h{ 0 }; ///< The full ROI height.
464 int m_full_bin_x{ 1 }; ///< The x-binning in the full ROI.
465 int m_full_bin_y{ 1 }; ///< The y-binning in the full ROI.
466
467 float m_full_currbin_x{ 0 }; ///< The current-binning full ROI center x coordinate.
468 float m_full_currbin_y{ 0 }; ///< The current-binning full ROI center y coordinate.
469 int m_full_currbin_w{ 0 }; ///< The current-binning full ROI width.
470 int m_full_currbin_h{ 0 }; ///< The current-binning full ROI height.
471
472 pcf::IndiProperty m_indiP_roi_x; ///< Property used to set the ROI x center coordinate
473 pcf::IndiProperty m_indiP_roi_y; ///< Property used to set the ROI x center coordinate
474 pcf::IndiProperty m_indiP_roi_w; ///< Property used to set the ROI width
475 pcf::IndiProperty m_indiP_roi_h; ///< Property used to set the ROI height
476 pcf::IndiProperty m_indiP_roi_bin_x; ///< Property used to set the ROI x binning
477 pcf::IndiProperty m_indiP_roi_bin_y; ///< Property used to set the ROI y binning
478
479 pcf::IndiProperty m_indiP_fullROI; ///< Property used to preset the full ROI dimensions.
480
481 pcf::IndiProperty m_indiP_roi_check; ///< Property used to trigger checking the target ROI
482
483 pcf::IndiProperty m_indiP_roi_set; ///< Property used to trigger setting the ROI
484
485 pcf::IndiProperty m_indiP_roi_full; ///< Property used to trigger setting the full ROI.
486 pcf::IndiProperty m_indiP_roi_fullbin; ///< Property used to trigger setting the full in current binning ROI.
487 pcf::IndiProperty m_indiP_roi_loadlast; ///< Property used to trigger loading the last ROI as the target.
488 pcf::IndiProperty m_indiP_roi_last; ///< Property used to trigger setting the last ROI.
489 pcf::IndiProperty m_indiP_roi_default; ///< Property used to trigger setting the default and startup ROI.
490
491 ///@}
492
493 /** \name Crop Mode
494 * Crop mode controls are exposed if derivedT::c_stdCamera_cropMode==true
495 * @{
496 */
497 bool m_cropMode{ false }; ///< Status of crop mode ROIs, if enabled for this camera.
498 bool m_cropModeSet{ false }; ///< Desired status of crop mode ROIs, if enabled for this camera.
499
500 pcf::IndiProperty m_indiP_cropMode; ///< Property used to toggle crop mode on and off.
501 ///@}
502
503 /** \name Shutter Control
504 * Shutter controls are exposed if derivedT::c_stdCamera_hasShutter == true.
505 * @{
506 */
507 std::string m_shutterStatus{ "UNKNOWN" };
508 int m_shutterState{ -1 }; /// State of the shutter. 0 = shut, 1 = open, -1 = unknown.
509
510 pcf::IndiProperty m_indiP_shutterStatus; ///< Property to report shutter status
511 pcf::IndiProperty m_indiP_shutter; ///< Property used to control the shutter, a switch.
512
513 ///@}
514
515 /** \name State String
516 * The State string is exposed if derivedT::c_stdCamera_usesStateString is true.
517 * @{
518 */
519 pcf::IndiProperty m_indiP_stateString;
520 ///@}
521
522 public:
523 /// Destructor.
524 ~stdCamera() noexcept;
525
526 /// Setup the configuration system
527 /**
528 * This should be called in `derivedT::setupConfig` as
529 * \code
530 stdCamera<derivedT>::setupConfig(config);
531 \endcode
532 * with appropriate error checking.
533 */
534 int setupConfig( mx::app::appConfigurator &config /**< [out] the derived classes configurator*/ );
535
536 /// load the configuration system results
537 /**
538 * This should be called in `derivedT::loadConfig` as
539 * \code
540 stdCamera<derivedT>::loadConfig(config);
541 \endcode
542 * with appropriate error checking.
543 */
544 int loadConfig( mx::app::appConfigurator &config /**< [in] the derived classes configurator*/ );
545
546 protected:
547 // workers to create indi variables if needed
548 int createReadoutSpeed( const mx::meta::trueFalseT<true> &t );
549
550 int createReadoutSpeed( const mx::meta::trueFalseT<false> &f );
551
552 int createVShiftSpeed( const mx::meta::trueFalseT<true> &t );
553
554 int createVShiftSpeed( const mx::meta::trueFalseT<false> &f );
555
556 public:
557 /// Startup function
558 /**
559 * This should be called in `derivedT::appStartup` as
560 * \code
561 stdCamera<derivedT>::appStartup();
562 \endcode
563 * with appropriate error checking.
564 *
565 * You should set the default/startup values of m_currentROI as well as the min/max/step values for the ROI
566 parameters
567 * before calling this function.
568 *
569 * \returns 0 on success
570 * \returns -1 on error, which is logged.
571 */
573
574 /// Application logic
575 /** Checks the stdCamera thread
576 *
577 * This should be called from the derived's appLogic() as in
578 * \code
579 stdCamera<derivedT>::appLogic();
580 \endcode
581 * with appropriate error checking.
582 *
583 * \returns 0 on success
584 * \returns -1 on error, which is logged.
585 */
586 int appLogic();
587
588 /// Actions on power off
589 /**
590 * This should be called from the derived's onPowerOff() as in
591 * \code
592 stdCamera<derivedT>::onPowerOff();
593 \endcode
594 * with appropriate error checking.
595 *
596 * The INDI mutex should be locked before calling.
597 *
598 * \returns 0 on success
599 * \returns -1 on error, which is logged.
600 */
602
603 /// Actions while powered off
604 /**
605 * This should be called from the derived's whilePowerOff() as in
606 * \code
607 stdCamera<derivedT>::whilePowerOff();
608 \endcode
609 * with appropriate error checking.
610 *
611 * \returns 0 on success
612 * \returns -1 on error, which is logged.
613 */
615
616 /// Application shutdown
617 /** Shuts down the stdCamera thread
618 *
619 * \code
620 stdCamera<derivedT>::appShutdown();
621 \endcode
622 * with appropriate error checking.
623 *
624 * \returns 0 on success
625 * \returns -1 on error, which is logged.
626 */
628
629 protected:
630 /** \name INDI
631 *
632 *@{
633 */
634 public:
635 /// The static callback function to be registered for stdCamera properties
636 /** Calls newCallback_stdCamera
637 *
638 * \returns 0 on success.
639 * \returns -1 on error.
640 */
642 void *app, ///< [in] a pointer to this, will be static_cast-ed to derivedT.
643 const pcf::IndiProperty &ipRecv ///< [in] the INDI property sent with the the new property request.
644 );
645
646 /// The callback function for stdCamera properties
647 /** Dispatches to the relevant handler
648 *
649 * \returns 0 on success.
650 * \returns -1 on error.
651 */
653 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
654
655 /// Interface to setTempSetPt when the derivedT has temperature control
656 /** Tag-dispatch resolution of c_stdCamera_tempControl==true will call this function.
657 * Calls derivedT::setTempSetPt.
658 */
659 int setTempSetPt( const mx::meta::trueFalseT<true> &t );
660
661 /// Interface to setTempSetPt when the derivedT does not have temperature control
662 /** Tag-dispatch resolution of c_stdCamera_tempControl==false will call this function.
663 * Prevents requiring derivedT::setTempSetPt.
664 */
665 int setTempSetPt( const mx::meta::trueFalseT<false> &f );
666
667 /// Callback to process a NEW CCD temp request
668 /**
669 * \returns 0 on success.
670 * \returns -1 on error.
671 */
673 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
674
675 /// Interface to setTempControl when the derivedT has temperature control
676 /** Tag-dispatch resolution of c_stdCamera_tempControl==true will call this function.
677 * Calls derivedT::setTempControl.
678 */
679 int setTempControl( const mx::meta::trueFalseT<true> &t );
680
681 /// Interface to setTempControl when the derivedT does not have temperature control
682 /** Tag-dispatch resolution of c_stdCamera_tempControl==false will call this function.
683 * Prevents requiring derivedT::setTempControl.
684 */
685 int setTempControl( const mx::meta::trueFalseT<false> &f );
686
687 /// Callback to process a NEW CCD temp control request
688 /**
689 * \returns 0 on success.
690 * \returns -1 on error.
691 */
693 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
694
695 /// Interface to setReadoutSpeed when the derivedT has readout speed control
696 /** Tag-dispatch resolution of c_stdCamera_readoutSpeed==true will call this function.
697 * Calls derivedT::setReadoutSpeed.
698 */
699 int setReadoutSpeed( const mx::meta::trueFalseT<true> &t );
700
701 /// Interface to setReadoutSpeed when the derivedT does not have readout speed control
702 /** Tag-dispatch resolution of c_stdCamera_readoutSpeed==false will call this function.
703 * Just returns 0.
704 */
705 int setReadoutSpeed( const mx::meta::trueFalseT<false> &f );
706
707 /// Callback to process a NEW readout speed request
708 /**
709 * \returns 0 on success.
710 * \returns -1 on error.
711 */
713 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
714
715 /// Interface to setVShiftSpeed when the derivedT has vshift speed control
716 /** Tag-dispatch resolution of c_stdCamera_vShiftSpeed==true will call this function.
717 * Calls derivedT::setVShiftSpeed.
718 */
719 int setVShiftSpeed( const mx::meta::trueFalseT<true> &t );
720
721 /// Interface to setVShiftSpeed when the derivedT does not have vshift speed control
722 /** Tag-dispatch resolution of c_stdCamera_vShiftSpeed==false will call this function.
723 * Just returns 0.
724 */
725 int setVShiftSpeed( const mx::meta::trueFalseT<false> &f );
726
727 /// Callback to process a NEW vshift speed request
728 /**
729 * \returns 0 on success.
730 * \returns -1 on error.
731 */
733 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
734
735 /// Interface to setEMGain when the derivedT has EM Gain
736 /** Tag-dispatch resolution of c_stdCamera_emGain==true will call this function.
737 * Calls derivedT::setEMGain.
738 */
739 int setEMGain( const mx::meta::trueFalseT<true> &t );
740
741 /// Interface to setEMGain when the derivedT does not have EM Gain
742 /** Tag-dispatch resolution of c_stdCamera_emGain==false will call this function.
743 * This prevents requiring derivedT to have its own setEMGain().
744 */
745 int setEMGain( const mx::meta::trueFalseT<false> &f );
746
747 /// Callback to process a NEW EM gain request
748 /**
749 * \returns 0 on success.
750 * \returns -1 on error.
751 */
753 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
754
755 /// Interface to setExpTime when the derivedT uses exposure time controls
756 /** Tag-dispatch resolution of c_stdCamera_exptimeCtrl==true will call this function.
757 * Calls derivedT::setExpTime.
758 */
759 int setExpTime( const mx::meta::trueFalseT<true> &t );
760
761 /// Interface to setExptime when the derivedT does not use exposure time controls.
762 /** Tag-dispatch resolution of c_stdCamera_exptimeCtrl==false will call this function.
763 * This prevents requiring derivedT to have its own setExpTime().
764 */
765 int setExpTime( const mx::meta::trueFalseT<false> &f );
766
767 /// Callback to process a NEW exposure time request
768 /**
769 * \returns 0 on success.
770 * \returns -1 on error.
771 */
773 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
774
775 /// Interface to setFPS when the derivedT uses FPS controls
776 /** Tag-dispatch resolution of c_stdCamera_fpsCtrl==true will call this function.
777 * Calls derivedT::setFPS.
778 */
779 int setFPS( const mx::meta::trueFalseT<true> &t );
780
781 /// Interface to setFPS when the derivedT does not use FPS controls.
782 /** Tag-dispatch resolution of c_stdCamera_hasFPS==false will call this function.
783 * This prevents requiring derivedT to have its own setFPS().
784 */
785 int setFPS( const mx::meta::trueFalseT<false> &f );
786
787 /// Callback to process a NEW fps request
788 /**
789 * \returns 0 on success.
790 * \returns -1 on error.
791 */
793 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
794
795 /// Interface to setSynchro when the derivedT has synchronization
796 /** Tag-dispatch resolution of c_stdCamera_synchro==true will call this function.
797 * Calls derivedT::setSynchro.
798 */
799 int setSynchro( const mx::meta::trueFalseT<true> &t );
800
801 /// Interface to setSynchro when the derivedT does not have synchronization
802 /** Tag-dispatch resolution of c_stdCamera_ynchro==false will call this function.
803 * This prevents requiring derivedT to have its own setSynchro().
804 */
805 int setSynchro( const mx::meta::trueFalseT<false> &f );
806
807 /// Callback to process a NEW synchro request
808 /**
809 * \returns 0 on success.
810 * \returns -1 on error.
811 */
813 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
814
815 /// Callback to process a NEW mode request
816 /**
817 * \returns 0 on success.
818 * \returns -1 on error.
819 */
821 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
822
823 /// Callback to process a NEW reconfigure request
824 /**
825 * \returns 0 on success.
826 * \returns -1 on error.
827 */
829 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
830
831 /// Interface to setCropMode when the derivedT has crop mode
832 /** Tag-dispatch resolution of c_stdCamera_cropMode==true will call this function.
833 * Calls derivedT::setCropMode.
834 */
835 int setCropMode( const mx::meta::trueFalseT<true> &t );
836
837 /// Interface to setCropMode when the derivedT does not have crop mode
838 /** Tag-dispatch resolution of c_stdCamera_cropMode==false will call this function.
839 * This prevents requiring derivedT to have its own setCropMode().
840 */
841 int setCropMode( const mx::meta::trueFalseT<false> &f );
842
843 /// Callback to process a NEW cropMode request
844 /**
845 * \returns 0 on success.
846 * \returns -1 on error.
847 */
849 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
850
851 /// Callback to process a NEW roi_x request
852 /**
853 * \returns 0 on success.
854 * \returns -1 on error.
855 */
857 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
858
859 /// Callback to process a NEW roi_y request
860 /**
861 * \returns 0 on success.
862 * \returns -1 on error.
863 */
865 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
866
867 /// Callback to process a NEW roi_w request
868 /**
869 * \returns 0 on success.
870 * \returns -1 on error.
871 */
873 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
874
875 /// Callback to process a NEW roi_h request
876 /**
877 * \returns 0 on success.
878 * \returns -1 on error.
879 */
881 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
882
883 /// Callback to process a NEW bin_x request
884 /**
885 * \returns 0 on success.
886 * \returns -1 on error.
887 */
889 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
890
891 /// Callback to process a NEW bin_y request
892 /**
893 * \returns 0 on success.
894 * \returns -1 on error.
895 */
897 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
898
899 /// Interface to checkNextROI when the derivedT uses ROIs
900 /** Tag-dispatch resolution of c_stdCamera_usesROI==true will call this function.
901 * Calls derivedT::checkNextROI.
902 */
903 int checkNextROI( const mx::meta::trueFalseT<true> &t );
904
905 /// Interface to checkNextROI when the derivedT does not use ROIs.
906 /** Tag-dispatch resolution of c_stdCamera_usesROI==false will call this function.
907 * This prevents requiring derivedT to have its own checkNextROI().
908 */
909 int checkNextROI( const mx::meta::trueFalseT<false> &f );
910
911 /// Callback to process a NEW roi_check request
912 /**
913 * \returns 0 on success.
914 * \returns -1 on error.
915 */
917 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
918
919 /// Interface to setNextROI when the derivedT uses ROIs
920 /** Tag-dispatch resolution of c_stdCamera_usesROI==true will call this function.
921 * Calls derivedT::setNextROI.
922 */
923 int setNextROI( const mx::meta::trueFalseT<true> &t );
924
925 /// Interface to setNextROI when the derivedT does not use ROIs.
926 /** Tag-dispatch resolution of c_stdCamera_usesROI==false will call this function.
927 * This prevents requiring derivedT to have its own setNextROI().
928 */
929 int setNextROI( const mx::meta::trueFalseT<false> &f );
930
931 /// Callback to process a NEW roi_set request
932 /**
933 * \returns 0 on success.
934 * \returns -1 on error.
935 */
937 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
938
939 /// Callback to process a NEW roi_full request
940 /**
941 * \returns 0 on success.
942 * \returns -1 on error.
943 */
945 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
946
947 /// Callback to process a NEW roi_fullbin request
948 /**
949 * \returns 0 on success.
950 * \returns -1 on error.
951 */
953 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
954
955 /// Callback to process a NEW roi_loadlast request
956 /**
957 * \returns 0 on success.
958 * \returns -1 on error.
959 */
961 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
962
963 /// Callback to process a NEW roi_last request
964 /**
965 * \returns 0 on success.
966 * \returns -1 on error.
967 */
969 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
970
971 /// Callback to process a NEW roi_default request
972 /**
973 * \returns 0 on success.
974 * \returns -1 on error.
975 */
977 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
978
979 /// Interface to setShutter when the derivedT has a shutter
980 /** Tag-dispatch resolution of c_stdCamera_hasShutter==true will call this function.
981 * Calls derivedT::setShutter.
982 */
983 int setShutter( int ss, const mx::meta::trueFalseT<true> &t );
984
985 /// Interface to setShutter when the derivedT does not have a shutter.
986 /** Tag-dispatch resolution of c_stdCamera_hasShutter==false will call this function.
987 * This prevents requiring derivedT to have its own setShutter().
988 */
989 int setShutter( int ss, const mx::meta::trueFalseT<false> &f );
990
991 /// Callback to process a NEW shutter request
992 /**
993 * \returns 0 on success.
994 * \returns -1 on error.
995 */
997 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/ );
998
999 /// Interface to stateString when the derivedT provides it
1000 /** Tag-dispatch resolution of c_stdCamera_usesStateString==true will call this function.
1001 * Calls derivedT::stateString.
1002 */
1003 std::string stateString( const mx::meta::trueFalseT<true> &t );
1004
1005 /// Interface to stateString when the derivedT does not provide it
1006 /** Tag-dispatch resolution of c_stdCamera_usesStateString==false will call this function.
1007 * returns "".
1008 */
1009 std::string stateString( const mx::meta::trueFalseT<false> &f );
1010
1011 /// Interface to stateStringValid when the derivedT provides it
1012 /** Tag-dispatch resolution of c_stdCamera_usesStateString==true will call this function.
1013 * Calls derivedT::stateStringValid.
1014 */
1015 bool stateStringValid( const mx::meta::trueFalseT<true> &t );
1016
1017 /// Interface to stateStringValid when the derivedT does not provide it
1018 /** Tag-dispatch resolution of c_stdCamera_usesStateString==false will call this function.
1019 * returns false.
1020 */
1021 bool stateStringValid( const mx::meta::trueFalseT<false> &f );
1022
1023 /// Update the INDI properties for this device controller
1024 /** You should call this once per main loop.
1025 * It is not called automatically.
1026 *
1027 * \returns 0 on success.
1028 * \returns -1 on error.
1029 */
1031
1032 ///@}
1033
1034 /** \name Telemeter Interface
1035 * @{
1036 */
1037
1038 int recordCamera( bool force = false );
1039
1040 ///@}
1041
1042 private:
1043 derivedT &derived()
1044 {
1045 return *static_cast<derivedT *>( this );
1046 }
1047};
1048
1049template <class derivedT>
1051{
1052 return;
1053}
1054
1055template <class derivedT>
1056int stdCamera<derivedT>::setupConfig( mx::app::appConfigurator &config )
1057{
1058 if( derivedT::c_stdCamera_tempControl )
1059 {
1060 config.add( "camera.startupTemp",
1061 "",
1062 "camera.startupTemp",
1063 argType::Required,
1064 "camera",
1065 "startupTemp",
1066 false,
1067 "float",
1068 "The temperature setpoint to set after a power-on [C]. Default is 20 C." );
1069 }
1070
1071 if( derivedT::c_stdCamera_readoutSpeed )
1072 {
1073 config.add( "camera.defaultReadoutSpeed",
1074 "",
1075 "camera.defaultReadoutSpeed",
1076 argType::Required,
1077 "camera",
1078 "defaultReadoutSpeed",
1079 false,
1080 "string",
1081 "The default amplifier and readout speed." );
1082 }
1083
1084 if( derivedT::c_stdCamera_vShiftSpeed )
1085 {
1086 config.add( "camera.defaultVShiftSpeed",
1087 "",
1088 "camera.defaultVShiftSpeed",
1089 argType::Required,
1090 "camera",
1091 "defaultVShiftSpeed",
1092 false,
1093 "string",
1094 "The default vertical shift speed." );
1095 }
1096
1097 if( derivedT::c_stdCamera_emGain )
1098 {
1099 config.add( "camera.maxEMGain",
1100 "",
1101 "camera.maxEMGain",
1102 argType::Required,
1103 "camera",
1104 "maxEMGain",
1105 false,
1106 "unsigned",
1107 "The maximum EM gain which can be set by the user." );
1108 }
1109
1110 if( derivedT::c_stdCamera_usesModes )
1111 {
1112 config.add( "camera.startupMode",
1113 "",
1114 "camera.startupMode",
1115 argType::Required,
1116 "camera",
1117 "startupMode",
1118 false,
1119 "string",
1120 "The mode to set upon power on or application startup." );
1121 }
1122
1123 if( derivedT::c_stdCamera_usesROI )
1124 {
1125 config.add( "camera.default_x",
1126 "",
1127 "camera.default_x",
1128 argType::Required,
1129 "camera",
1130 "default_x",
1131 false,
1132 "float",
1133 "The default ROI x position." );
1134 config.add( "camera.default_y",
1135 "",
1136 "camera.default_y",
1137 argType::Required,
1138 "camera",
1139 "default_y",
1140 false,
1141 "float",
1142 "The default ROI y position." );
1143 config.add( "camera.default_w",
1144 "",
1145 "camera.default_w",
1146 argType::Required,
1147 "camera",
1148 "default_w",
1149 false,
1150 "int",
1151 "The default ROI width." );
1152 config.add( "camera.default_h",
1153 "",
1154 "camera.default_h",
1155 argType::Required,
1156 "camera",
1157 "default_h",
1158 false,
1159 "int",
1160 "The default ROI height." );
1161 config.add( "camera.default_bin_x",
1162 "",
1163 "camera.default_bin_x",
1164 argType::Required,
1165 "camera",
1166 "default_bin_x",
1167 false,
1168 "int",
1169 "The default ROI x binning." );
1170 config.add( "camera.default_bin_y",
1171 "",
1172 "camera.default_bin_y",
1173 argType::Required,
1174 "camera",
1175 "default_bin_y",
1176 false,
1177 "int",
1178 "The default ROI y binning." );
1179 }
1180
1181 return 0;
1182}
1183
1184template <class derivedT>
1185int stdCamera<derivedT>::loadConfig( mx::app::appConfigurator &config )
1186{
1187 if( derivedT::c_stdCamera_tempControl )
1188 {
1189 config( m_startupTemp, "camera.startupTemp" );
1190 }
1191
1192 if( derivedT::c_stdCamera_readoutSpeed )
1193 {
1194 config( m_defaultReadoutSpeed, "camera.defaultReadoutSpeed" );
1195 }
1196
1197 if( derivedT::c_stdCamera_vShiftSpeed )
1198 {
1199 config( m_defaultVShiftSpeed, "camera.defaultVShiftSpeed" );
1200 }
1201
1202 if( derivedT::c_stdCamera_emGain )
1203 {
1204 config( m_maxEMGain, "camera.maxEMGain" );
1205 }
1206
1207 if( derivedT::c_stdCamera_usesModes )
1208 {
1209 int rv = loadCameraConfig( m_cameraModes, config );
1210
1211 if( rv < 0 )
1212 {
1213 if( rv == CAMCTRL_E_NOCONFIGS )
1214 {
1215 derivedT::template log<text_log>( "No camera configurations found.", logPrio::LOG_CRITICAL );
1216 }
1217 }
1218
1219 config( m_startupMode, "camera.startupMode" );
1220 }
1221
1222 if( derivedT::c_stdCamera_usesROI )
1223 {
1224 config( m_default_x, "camera.default_x" );
1225 config( m_default_y, "camera.default_y" );
1226 config( m_default_w, "camera.default_w" );
1227 config( m_default_h, "camera.default_h" );
1228 config( m_default_bin_x, "camera.default_bin_x" );
1229 config( m_default_bin_y, "camera.default_bin_y" );
1230 }
1231
1232 return 0;
1233}
1234
1235template <class derivedT>
1236int stdCamera<derivedT>::createReadoutSpeed( const mx::meta::trueFalseT<true> &t )
1237{
1238 static_cast<void>( t );
1239
1240 derived().createStandardIndiSelectionSw(
1241 m_indiP_readoutSpeed, "readout_speed", m_readoutSpeedNames, "Readout Speed" );
1242
1243 // Set the labes if provided
1244 if( m_readoutSpeedNameLabels.size() == m_readoutSpeedNames.size() )
1245 {
1246 for( size_t n = 0; n < m_readoutSpeedNames.size(); ++n )
1247 m_indiP_readoutSpeed[m_readoutSpeedNames[n]].setLabel( m_readoutSpeedNameLabels[n] );
1248 }
1249
1250 derived().registerIndiPropertyNew( m_indiP_readoutSpeed, st_newCallBack_stdCamera );
1251
1252 return 0;
1253}
1254
1255template <class derivedT>
1256int stdCamera<derivedT>::createReadoutSpeed( const mx::meta::trueFalseT<0> &f )
1257{
1258 static_cast<void>( f );
1259
1260 return 0;
1261}
1262
1263template <class derivedT>
1264int stdCamera<derivedT>::createVShiftSpeed( const mx::meta::trueFalseT<true> &t )
1265{
1266 static_cast<void>( t );
1267
1268 derived().createStandardIndiSelectionSw(
1269 m_indiP_vShiftSpeed, "vshift_speed", m_vShiftSpeedNames, "Vert. Shift Speed" );
1270
1271 if( m_vShiftSpeedNameLabels.size() == m_vShiftSpeedNames.size() )
1272 {
1273 for( size_t n = 0; n < m_vShiftSpeedNames.size(); ++n )
1274 m_indiP_vShiftSpeed[m_vShiftSpeedNames[n]].setLabel( m_vShiftSpeedNameLabels[n] );
1275 }
1276
1277 derived().registerIndiPropertyNew( m_indiP_vShiftSpeed, st_newCallBack_stdCamera );
1278
1279 return 0;
1280}
1281
1282template <class derivedT>
1283int stdCamera<derivedT>::createVShiftSpeed( const mx::meta::trueFalseT<0> &f )
1284{
1285 static_cast<void>( f );
1286
1287 return 0;
1288}
1289
1290template <class derivedT>
1292{
1293
1294 if( derivedT::c_stdCamera_tempControl )
1295 {
1296 // The min/max/step values should be set in derivedT before this is called.
1297 derived().createStandardIndiNumber(
1298 m_indiP_temp, "temp_ccd", m_minTemp, m_maxTemp, m_stepTemp, "%0.1f", "CCD Temperature", "CCD Temperature" );
1299 m_indiP_temp["current"].set( m_ccdTemp );
1300 m_indiP_temp["target"].set( m_ccdTempSetpt );
1301 if( derived().registerIndiPropertyNew( m_indiP_temp, st_newCallBack_stdCamera ) < 0 )
1302 {
1303#ifndef STDCAMERA_TEST_NOLOG
1304 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1305#endif
1306 return -1;
1307 }
1308
1309 derived().createStandardIndiToggleSw(
1310 m_indiP_tempcont, "temp_controller", "CCD Temperature", "Control On/Off" );
1311 m_indiP_tempcont["toggle"].set( pcf::IndiElement::Off );
1312 if( derived().registerIndiPropertyNew( m_indiP_tempcont, st_newCallBack_stdCamera ) < 0 )
1313 {
1314#ifndef STDCAMERA_TEST_NOLOG
1315 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1316#endif
1317 return -1;
1318 }
1319
1320 derived().createROIndiText(
1321 m_indiP_tempstat, "temp_control", "status", "CCD Temperature", "", "CCD Temperature" );
1322 if( derived().registerIndiPropertyReadOnly( m_indiP_tempstat ) < 0 )
1323 {
1324#ifndef STDCAMERA_TEST_NOLOG
1325 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1326#endif
1327 return -1;
1328 }
1329 }
1330 else if( derivedT::c_stdCamera_temp )
1331 {
1332 derived().createROIndiNumber( m_indiP_temp, "temp_ccd", "CCD Temperature", "CCD Temperature" );
1333 m_indiP_temp.add( pcf::IndiElement( "current" ) );
1334 m_indiP_temp["current"].set( m_ccdTemp );
1335 if( derived().registerIndiPropertyReadOnly( m_indiP_temp ) < 0 )
1336 {
1337#ifndef STDCAMERA_TEST_NOLOG
1338 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1339#endif
1340 return -1;
1341 }
1342 }
1343
1344 if( derivedT::c_stdCamera_readoutSpeed )
1345 {
1346 mx::meta::trueFalseT<derivedT::c_stdCamera_readoutSpeed> tf;
1347 if( createReadoutSpeed( tf ) < 0 )
1348 {
1349#ifndef STDCAMERA_TEST_NOLOG
1350 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1351#endif
1352 return -1;
1353 }
1354 }
1355
1356 if( derivedT::c_stdCamera_vShiftSpeed )
1357 {
1358 mx::meta::trueFalseT<derivedT::c_stdCamera_vShiftSpeed> tf;
1359 if( createVShiftSpeed( tf ) < 0 )
1360 {
1361#ifndef STDCAMERA_TEST_NOLOG
1362 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1363#endif
1364 return -1;
1365 }
1366 }
1367
1368 if( derivedT::c_stdCamera_emGain )
1369 {
1370 derived().createStandardIndiNumber( m_indiP_emGain, "emgain", 0, 1000, 1, "%0.3f" );
1371 if( derived().registerIndiPropertyNew( m_indiP_emGain, st_newCallBack_stdCamera ) < 0 )
1372 {
1373#ifndef STDCAMERA_TEST_NOLOG
1374 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1375#endif
1376 return -1;
1377 }
1378 }
1379
1380 if( derivedT::c_stdCamera_exptimeCtrl )
1381 {
1382 derived().createStandardIndiNumber(
1383 m_indiP_exptime, "exptime", m_minExpTime, m_maxExpTime, m_stepExpTime, "%0.3f" );
1384 if( derived().registerIndiPropertyNew( m_indiP_exptime, st_newCallBack_stdCamera ) < 0 )
1385 {
1386#ifndef STDCAMERA_TEST_NOLOG
1387 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1388#endif
1389 return -1;
1390 }
1391 }
1392
1393 if( derivedT::c_stdCamera_fpsCtrl )
1394 {
1395 derived().createStandardIndiNumber( m_indiP_fps, "fps", m_minFPS, m_maxFPS, m_stepFPS, "%0.2f" );
1396 if( derived().registerIndiPropertyNew( m_indiP_fps, st_newCallBack_stdCamera ) < 0 )
1397 {
1398#ifndef STDCAMERA_TEST_NOLOG
1399 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1400#endif
1401 return -1;
1402 }
1403 }
1404 else if( derivedT::c_stdCamera_fps )
1405 {
1406 derived().createROIndiNumber( m_indiP_fps, "fps" );
1407 m_indiP_fps.add( pcf::IndiElement( "current" ) );
1408 m_indiP_fps["current"].setMin( m_minFPS );
1409 m_indiP_fps["current"].setMax( m_maxFPS );
1410 m_indiP_fps["current"].setStep( m_stepFPS );
1411 m_indiP_fps["current"].setFormat( "%0.2f" );
1412
1413 if( derived().registerIndiPropertyReadOnly( m_indiP_fps ) < 0 )
1414 {
1415#ifndef STDCAMERA_TEST_NOLOG
1416 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1417#endif
1418 return -1;
1419 }
1420 }
1421
1422 if( derivedT::c_stdCamera_synchro )
1423 {
1424 derived().createStandardIndiToggleSw( m_indiP_synchro, "synchro", "Synchronization", "Synchronization" );
1425 if( derived().registerIndiPropertyNew( m_indiP_synchro, st_newCallBack_stdCamera ) < 0 )
1426 {
1427#ifndef STDCAMERA_TEST_NOLOG
1428 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1429#endif
1430 return -1;
1431 }
1432 }
1433
1434 if( derivedT::c_stdCamera_usesModes )
1435 {
1436 std::vector<std::string> modeNames;
1437 for( auto it = m_cameraModes.begin(); it != m_cameraModes.end(); ++it )
1438 {
1439 modeNames.push_back( it->first );
1440 }
1441
1442 if( derived().createStandardIndiSelectionSw( m_indiP_mode, "mode", modeNames ) < 0 )
1443 {
1444 derivedT::template log<software_critical>( { __FILE__, __LINE__ } );
1445 return -1;
1446 }
1447 if( derived().registerIndiPropertyNew( m_indiP_mode, st_newCallBack_stdCamera ) < 0 )
1448 {
1449#ifndef STDCAMERA_TEST_NOLOG
1450 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1451#endif
1452 return -1;
1453 }
1454 }
1455
1456 derived().createStandardIndiRequestSw( m_indiP_reconfig, "reconfigure" );
1457 if( derived().registerIndiPropertyNew( m_indiP_reconfig, st_newCallBack_stdCamera ) < 0 )
1458 {
1459#ifndef STDCAMERA_TEST_NOLOG
1460 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1461#endif
1462 return -1;
1463 }
1464
1465 if( derivedT::c_stdCamera_usesROI )
1466 {
1467 // The min/max/step values should be set in derivedT before this is called.
1468 derived().createStandardIndiNumber( m_indiP_roi_x, "roi_region_x", m_minROIx, m_maxROIx, m_stepROIx, "%0.1f" );
1469 if( derived().registerIndiPropertyNew( m_indiP_roi_x, st_newCallBack_stdCamera ) < 0 )
1470 {
1471#ifndef STDCAMERA_TEST_NOLOG
1472 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1473#endif
1474 return -1;
1475 }
1476
1477 derived().createStandardIndiNumber( m_indiP_roi_y, "roi_region_y", m_minROIy, m_maxROIy, m_stepROIy, "%0.1f" );
1478 if( derived().registerIndiPropertyNew( m_indiP_roi_y, st_newCallBack_stdCamera ) < 0 )
1479 {
1480#ifndef STDCAMERA_TEST_NOLOG
1481 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1482#endif
1483 return -1;
1484 }
1485
1486 derived().createStandardIndiNumber(
1487 m_indiP_roi_w, "roi_region_w", m_minROIWidth, m_maxROIWidth, m_stepROIWidth, "%d" );
1488 if( derived().registerIndiPropertyNew( m_indiP_roi_w, st_newCallBack_stdCamera ) < 0 )
1489 {
1490#ifndef STDCAMERA_TEST_NOLOG
1491 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1492#endif
1493 return -1;
1494 }
1495
1496 derived().createStandardIndiNumber(
1497 m_indiP_roi_h, "roi_region_h", m_minROIHeight, m_maxROIHeight, m_stepROIHeight, "%d" );
1498 if( derived().registerIndiPropertyNew( m_indiP_roi_h, st_newCallBack_stdCamera ) < 0 )
1499 {
1500#ifndef STDCAMERA_TEST_NOLOG
1501 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1502#endif
1503 return -1;
1504 }
1505
1506 derived().createStandardIndiNumber(
1507 m_indiP_roi_bin_x, "roi_region_bin_x", m_minROIBinning_x, m_maxROIBinning_x, m_stepROIBinning_x, "%f" );
1508 if( derived().registerIndiPropertyNew( m_indiP_roi_bin_x, st_newCallBack_stdCamera ) < 0 )
1509 {
1510#ifndef STDCAMERA_TEST_NOLOG
1511 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1512#endif
1513 return -1;
1514 }
1515
1516 derived().createStandardIndiNumber(
1517 m_indiP_roi_bin_y, "roi_region_bin_y", m_minROIBinning_y, m_maxROIBinning_y, m_stepROIBinning_y, "%f" );
1518 if( derived().registerIndiPropertyNew( m_indiP_roi_bin_y, st_newCallBack_stdCamera ) < 0 )
1519 {
1520#ifndef STDCAMERA_TEST_NOLOG
1521 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1522#endif
1523 return -1;
1524 }
1525
1526 derived().createROIndiNumber( m_indiP_fullROI, "roi_full_region" );
1527 m_indiP_fullROI.add( pcf::IndiElement( "x" ) );
1528 m_indiP_fullROI["x"] = 0;
1529 m_indiP_fullROI.add( pcf::IndiElement( "y" ) );
1530 m_indiP_fullROI["y"] = 0;
1531 m_indiP_fullROI.add( pcf::IndiElement( "w" ) );
1532 m_indiP_fullROI["w"] = 0;
1533 m_indiP_fullROI.add( pcf::IndiElement( "h" ) );
1534 m_indiP_fullROI["h"] = 0;
1535 if( derived().registerIndiPropertyReadOnly( m_indiP_fullROI ) < 0 )
1536 {
1537#ifndef STDCAMERA_TEST_NOLOG
1538 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1539#endif
1540 return -1;
1541 }
1542
1543 derived().createStandardIndiRequestSw( m_indiP_roi_check, "roi_region_check" );
1544 if( derived().registerIndiPropertyNew( m_indiP_roi_check, st_newCallBack_stdCamera ) < 0 )
1545 {
1546#ifndef STDCAMERA_TEST_NOLOG
1547 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1548#endif
1549 return -1;
1550 }
1551
1552 derived().createStandardIndiRequestSw( m_indiP_roi_set, "roi_set" );
1553 if( derived().registerIndiPropertyNew( m_indiP_roi_set, st_newCallBack_stdCamera ) < 0 )
1554 {
1555#ifndef STDCAMERA_TEST_NOLOG
1556 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1557#endif
1558 return -1;
1559 }
1560
1561 derived().createStandardIndiRequestSw( m_indiP_roi_full, "roi_set_full" );
1562 if( derived().registerIndiPropertyNew( m_indiP_roi_full, st_newCallBack_stdCamera ) < 0 )
1563 {
1564#ifndef STDCAMERA_TEST_NOLOG
1565 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1566#endif
1567 return -1;
1568 }
1569
1570 derived().createStandardIndiRequestSw( m_indiP_roi_fullbin, "roi_set_full_bin" );
1571 if( derived().registerIndiPropertyNew( m_indiP_roi_fullbin, st_newCallBack_stdCamera ) < 0 )
1572 {
1573#ifndef STDCAMERA_TEST_NOLOG
1574 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1575#endif
1576 return -1;
1577 }
1578
1579 derived().createStandardIndiRequestSw( m_indiP_roi_loadlast, "roi_load_last" );
1580 if( derived().registerIndiPropertyNew( m_indiP_roi_loadlast, st_newCallBack_stdCamera ) < 0 )
1581 {
1582#ifndef STDCAMERA_TEST_NOLOG
1583 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1584#endif
1585 return -1;
1586 }
1587
1588 derived().createStandardIndiRequestSw( m_indiP_roi_last, "roi_set_last" );
1589 if( derived().registerIndiPropertyNew( m_indiP_roi_last, st_newCallBack_stdCamera ) < 0 )
1590 {
1591#ifndef STDCAMERA_TEST_NOLOG
1592 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1593#endif
1594 return -1;
1595 }
1596
1597 derived().createStandardIndiRequestSw( m_indiP_roi_default, "roi_set_default" );
1598 if( derived().registerIndiPropertyNew( m_indiP_roi_default, st_newCallBack_stdCamera ) < 0 )
1599 {
1600#ifndef STDCAMERA_TEST_NOLOG
1601 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1602#endif
1603 return -1;
1604 }
1605 }
1606
1607 if( derivedT::c_stdCamera_cropMode )
1608 {
1609 derived().createStandardIndiToggleSw( m_indiP_cropMode, "roi_crop_mode", "Crop Mode", "Crop Mode" );
1610 if( derived().registerIndiPropertyNew( m_indiP_cropMode, st_newCallBack_stdCamera ) < 0 )
1611 {
1612#ifndef STDCAMERA_TEST_NOLOG
1613 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1614#endif
1615 return -1;
1616 }
1617 }
1618
1619 // Set up INDI for shutter
1620 if( derivedT::c_stdCamera_hasShutter )
1621 {
1622 derived().createROIndiText(
1623 m_indiP_shutterStatus, "shutter_status", "status", "Shutter Status", "Shutter", "Status" );
1624 m_indiP_shutterStatus["status"] = m_shutterStatus;
1625 if( derived().registerIndiPropertyReadOnly( m_indiP_shutterStatus ) < 0 )
1626 {
1627#ifndef STDCAMERA_TEST_NOLOG
1628 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1629#endif
1630 return -1;
1631 }
1632
1633 derived().createStandardIndiToggleSw( m_indiP_shutter, "shutter", "Shutter", "Shutter" );
1634 if( derived().registerIndiPropertyNew( m_indiP_shutter, st_newCallBack_stdCamera ) < 0 )
1635 {
1636#ifndef STDCAMERA_TEST_NOLOG
1637 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1638#endif
1639 return -1;
1640 }
1641 }
1642
1643 if( derivedT::c_stdCamera_usesStateString )
1644 {
1645 derived().createROIndiText( m_indiP_stateString, "state_string", "current", "State String", "State", "String" );
1646 m_indiP_stateString.add( pcf::IndiElement( "valid" ) );
1647 m_indiP_stateString["valid"] = "no";
1648 if( derived().registerIndiPropertyReadOnly( m_indiP_stateString ) < 0 )
1649 {
1650#ifndef STDCAMERA_TEST_NOLOG
1651 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1652#endif
1653 return -1;
1654 }
1655 }
1656
1657 return 0;
1658} // int stdCamera<derivedT>::appStartup()
1659
1660template <class derivedT>
1662{
1663 try
1664 {
1665
1666 if( derived().state() == stateCodes::POWERON )
1667 {
1668 if( derived().powerOnWaitElapsed() )
1669 {
1670 derived().state( stateCodes::NOTCONNECTED );
1671
1672 // Set power-on defaults
1673 derived().powerOnDefaults();
1674
1675 if( derivedT::c_stdCamera_tempControl )
1676 {
1677 // then set startupTemp if configured
1678 if( m_startupTemp > -999 )
1679 m_ccdTempSetpt = m_startupTemp;
1680 derived().updateIfChanged( m_indiP_temp, "target", m_ccdTempSetpt, INDI_IDLE );
1681 }
1682
1683 if( derivedT::c_stdCamera_usesROI )
1684 {
1685 // m_currentROI should be set to default/startup values in derivedT::powerOnDefaults
1686 m_nextROI.x = m_currentROI.x;
1687 m_nextROI.y = m_currentROI.y;
1688 m_nextROI.w = m_currentROI.w;
1689 m_nextROI.h = m_currentROI.h;
1690 m_nextROI.bin_x = m_currentROI.bin_x;
1691 m_nextROI.bin_y = m_currentROI.bin_y;
1692
1693 derived().updateIfChanged( m_indiP_roi_x, "current", m_currentROI.x, INDI_IDLE );
1694 derived().updateIfChanged( m_indiP_roi_x, "target", m_nextROI.x, INDI_IDLE );
1695
1696 derived().updateIfChanged( m_indiP_roi_y, "current", m_currentROI.y, INDI_IDLE );
1697 derived().updateIfChanged( m_indiP_roi_y, "target", m_nextROI.y, INDI_IDLE );
1698
1699 derived().updateIfChanged( m_indiP_roi_w, "current", m_currentROI.w, INDI_IDLE );
1700 derived().updateIfChanged( m_indiP_roi_w, "target", m_nextROI.w, INDI_IDLE );
1701
1702 derived().updateIfChanged( m_indiP_roi_h, "current", m_currentROI.h, INDI_IDLE );
1703 derived().updateIfChanged( m_indiP_roi_h, "target", m_nextROI.h, INDI_IDLE );
1704
1705 derived().updateIfChanged( m_indiP_roi_bin_x, "current", m_currentROI.bin_x, INDI_IDLE );
1706 derived().updateIfChanged( m_indiP_roi_bin_x, "target", m_nextROI.bin_x, INDI_IDLE );
1707
1708 derived().updateIfChanged( m_indiP_roi_bin_y, "current", m_currentROI.bin_y, INDI_IDLE );
1709 derived().updateIfChanged( m_indiP_roi_bin_y, "target", m_nextROI.bin_y, INDI_IDLE );
1710 }
1711
1712 if( derivedT::c_stdCamera_hasShutter )
1713 {
1714 if( m_shutterStatus == "OPERATING" )
1715 {
1716 derived().updateIfChanged( m_indiP_shutterStatus, "status", m_shutterStatus, INDI_BUSY );
1717 }
1718 if( m_shutterStatus == "POWERON" || m_shutterStatus == "READY" )
1719 {
1720 derived().updateIfChanged( m_indiP_shutterStatus, "status", m_shutterStatus, INDI_OK );
1721 }
1722 else
1723 {
1724 derived().updateIfChanged( m_indiP_shutterStatus, "status", m_shutterStatus, INDI_IDLE );
1725 }
1726
1727 if( m_shutterState == 1 )
1728 {
1729 derived().updateSwitchIfChanged( m_indiP_shutter, "toggle", pcf::IndiElement::On, INDI_OK );
1730 }
1731 else
1732 {
1733 derived().updateSwitchIfChanged( m_indiP_shutter, "toggle", pcf::IndiElement::Off, INDI_IDLE );
1734 }
1735 }
1736
1737 return 0;
1738 }
1739 else
1740 {
1741 return 0;
1742 }
1743 }
1744 else if( derived().state() == stateCodes::READY || derived().state() == stateCodes::OPERATING )
1745 {
1746 if( derivedT::c_stdCamera_usesROI )
1747 {
1748 derived().updateIfChanged( m_indiP_roi_x, "current", m_currentROI.x, INDI_IDLE );
1749 derived().updateIfChanged( m_indiP_roi_x, "target", m_nextROI.x, INDI_IDLE );
1750
1751 derived().updateIfChanged( m_indiP_roi_y, "current", m_currentROI.y, INDI_IDLE );
1752 derived().updateIfChanged( m_indiP_roi_y, "target", m_nextROI.y, INDI_IDLE );
1753
1754 derived().updateIfChanged( m_indiP_roi_w, "current", m_currentROI.w, INDI_IDLE );
1755 derived().updateIfChanged( m_indiP_roi_w, "target", m_nextROI.w, INDI_IDLE );
1756
1757 derived().updateIfChanged( m_indiP_roi_h, "current", m_currentROI.h, INDI_IDLE );
1758 derived().updateIfChanged( m_indiP_roi_h, "target", m_nextROI.h, INDI_IDLE );
1759
1760 derived().updateIfChanged( m_indiP_roi_bin_x, "current", m_currentROI.bin_x, INDI_IDLE );
1761 derived().updateIfChanged( m_indiP_roi_bin_x, "target", m_nextROI.bin_x, INDI_IDLE );
1762
1763 derived().updateIfChanged( m_indiP_roi_bin_y, "current", m_currentROI.bin_y, INDI_IDLE );
1764 derived().updateIfChanged( m_indiP_roi_bin_y, "target", m_nextROI.bin_y, INDI_IDLE );
1765 }
1766 }
1767
1768 return 0;
1769 }
1770 catch( const std::exception &e )
1771 {
1772 return derivedT::template log<software_error, -1>(
1773 { __FILE__, __LINE__, std::string( "Exception caught: " ) + e.what() } );
1774 }
1775}
1776
1777template <class derivedT>
1779{
1780 if( !derived().m_indiDriver )
1781 return 0;
1782
1783 if( derivedT::c_stdCamera_usesModes )
1784 {
1785 for( auto it = m_cameraModes.begin(); it != m_cameraModes.end(); ++it )
1786 {
1787 derived().updateSwitchIfChanged( m_indiP_mode, it->first, pcf::IndiElement::Off, INDI_IDLE );
1788 }
1789 }
1790
1791 if( derivedT::c_stdCamera_usesROI )
1792 {
1793 indi::updateIfChanged( m_indiP_roi_x, "current", std::string( "" ), derived().m_indiDriver, INDI_IDLE );
1794 indi::updateIfChanged( m_indiP_roi_x, "target", std::string( "" ), derived().m_indiDriver, INDI_IDLE );
1795
1796 indi::updateIfChanged( m_indiP_roi_y, "current", std::string( "" ), derived().m_indiDriver, INDI_IDLE );
1797 indi::updateIfChanged( m_indiP_roi_y, "target", std::string( "" ), derived().m_indiDriver, INDI_IDLE );
1798
1799 indi::updateIfChanged( m_indiP_roi_w, "current", std::string( "" ), derived().m_indiDriver, INDI_IDLE );
1800 indi::updateIfChanged( m_indiP_roi_w, "target", std::string( "" ), derived().m_indiDriver, INDI_IDLE );
1801
1802 indi::updateIfChanged( m_indiP_roi_h, "current", std::string( "" ), derived().m_indiDriver, INDI_IDLE );
1803 indi::updateIfChanged( m_indiP_roi_h, "target", std::string( "" ), derived().m_indiDriver, INDI_IDLE );
1804
1805 indi::updateIfChanged( m_indiP_roi_bin_x, "current", std::string( "" ), derived().m_indiDriver, INDI_IDLE );
1806 indi::updateIfChanged( m_indiP_roi_bin_x, "target", std::string( "" ), derived().m_indiDriver, INDI_IDLE );
1807
1808 indi::updateIfChanged( m_indiP_roi_bin_y, "current", std::string( "" ), derived().m_indiDriver, INDI_IDLE );
1809 indi::updateIfChanged( m_indiP_roi_bin_y, "target", std::string( "" ), derived().m_indiDriver, INDI_IDLE );
1810 }
1811
1812 // Shutters can be independent pieces of hardware . . .
1813 if( derivedT::c_stdCamera_hasShutter )
1814 {
1815 if( m_shutterStatus == "OPERATING" )
1816 {
1817 derived().updateIfChanged( m_indiP_shutterStatus, "status", m_shutterStatus, INDI_BUSY );
1818 }
1819 if( m_shutterStatus == "POWERON" || m_shutterStatus == "READY" )
1820 {
1821 derived().updateIfChanged( m_indiP_shutterStatus, "status", m_shutterStatus, INDI_OK );
1822 }
1823 else
1824 {
1825 derived().updateIfChanged( m_indiP_shutterStatus, "status", m_shutterStatus, INDI_IDLE );
1826 }
1827
1828 if( m_shutterState == 0 )
1829 {
1830 derived().updateSwitchIfChanged( m_indiP_shutter, "toggle", pcf::IndiElement::On, INDI_OK );
1831 }
1832 else
1833 {
1834 derived().updateSwitchIfChanged( m_indiP_shutter, "toggle", pcf::IndiElement::Off, INDI_IDLE );
1835 }
1836 }
1837
1838 return 0;
1839}
1840
1841template <class derivedT>
1843{
1844 // Shutters can be independent pieces of hardware . . .
1845 if( derivedT::c_stdCamera_hasShutter )
1846 {
1847 if( m_shutterStatus == "OPERATING" )
1848 {
1849 derived().updateIfChanged( m_indiP_shutterStatus, "status", m_shutterStatus, INDI_BUSY );
1850 }
1851 if( m_shutterStatus == "POWERON" || m_shutterStatus == "READY" )
1852 {
1853 derived().updateIfChanged( m_indiP_shutterStatus, "status", m_shutterStatus, INDI_OK );
1854 }
1855 else
1856 {
1857 derived().updateIfChanged( m_indiP_shutterStatus, "status", m_shutterStatus, INDI_IDLE );
1858 }
1859
1860 if( m_shutterState == 0 )
1861 {
1862 derived().updateSwitchIfChanged( m_indiP_shutter, "toggle", pcf::IndiElement::On, INDI_OK );
1863 }
1864 else
1865 {
1866 derived().updateSwitchIfChanged( m_indiP_shutter, "toggle", pcf::IndiElement::Off, INDI_IDLE );
1867 }
1868 }
1869
1870 return 0;
1871}
1872
1873template <class derivedT>
1875{
1876 return 0;
1877}
1878
1879template <class derivedT>
1880int stdCamera<derivedT>::st_newCallBack_stdCamera( void *app, const pcf::IndiProperty &ipRecv )
1881{
1882 derivedT *_app = static_cast<derivedT *>( app );
1883 return _app->newCallBack_stdCamera( ipRecv );
1884}
1885
1886template <class derivedT>
1888{
1889
1890 if( ipRecv.getDevice() != derived().configName() )
1891 {
1892#ifndef XWCTEST_INDI_CALLBACK_VALIDATION
1893 derivedT::template log<software_error>( { __FILE__, __LINE__, "unknown INDI property" } );
1894#endif
1895
1896 return -1;
1897 }
1898
1899 std::string name = ipRecv.getName();
1900
1901 if( name == "reconfigure" )
1902 return newCallBack_reconfigure( ipRecv );
1903 else if( derivedT::c_stdCamera_temp && name == "temp_ccd" )
1904 return newCallBack_temp( ipRecv );
1905 else if( derivedT::c_stdCamera_tempControl && name == "temp_ccd" )
1906 return newCallBack_temp( ipRecv );
1907 else if( derivedT::c_stdCamera_tempControl && name == "temp_controller" )
1908 return newCallBack_temp_controller( ipRecv );
1909 else if( derivedT::c_stdCamera_readoutSpeed && name == "readout_speed" )
1910 return newCallBack_readoutSpeed( ipRecv );
1911 else if( derivedT::c_stdCamera_vShiftSpeed && name == "vshift_speed" )
1912 return newCallBack_vShiftSpeed( ipRecv );
1913 else if( derivedT::c_stdCamera_emGain && name == "emgain" )
1914 return newCallBack_emgain( ipRecv );
1915 else if( derivedT::c_stdCamera_exptimeCtrl && name == "exptime" )
1916 return newCallBack_exptime( ipRecv );
1917 else if( derivedT::c_stdCamera_fpsCtrl && name == "fps" )
1918 return newCallBack_fps( ipRecv );
1919 else if( derivedT::c_stdCamera_synchro && name == "synchro" )
1920 return newCallBack_synchro( ipRecv );
1921 else if( derivedT::c_stdCamera_usesModes && name == "mode" )
1922 return newCallBack_mode( ipRecv );
1923 else if( derivedT::c_stdCamera_cropMode && name == "roi_crop_mode" )
1924 return newCallBack_cropMode( ipRecv );
1925 else if( derivedT::c_stdCamera_usesROI && name == "roi_region_x" )
1926 return newCallBack_roi_x( ipRecv );
1927 else if( derivedT::c_stdCamera_usesROI && name == "roi_region_y" )
1928 return newCallBack_roi_y( ipRecv );
1929 else if( derivedT::c_stdCamera_usesROI && name == "roi_region_w" )
1930 return newCallBack_roi_w( ipRecv );
1931 else if( derivedT::c_stdCamera_usesROI && name == "roi_region_h" )
1932 return newCallBack_roi_h( ipRecv );
1933 else if( derivedT::c_stdCamera_usesROI && name == "roi_region_bin_x" )
1934 return newCallBack_roi_bin_x( ipRecv );
1935 else if( derivedT::c_stdCamera_usesROI && name == "roi_region_bin_y" )
1936 return newCallBack_roi_bin_y( ipRecv );
1937 else if( derivedT::c_stdCamera_usesROI && name == "roi_region_check" )
1938 return newCallBack_roi_check( ipRecv );
1939 else if( derivedT::c_stdCamera_usesROI && name == "roi_set" )
1940 return newCallBack_roi_set( ipRecv );
1941 else if( derivedT::c_stdCamera_usesROI && name == "roi_set_full" )
1942 return newCallBack_roi_full( ipRecv );
1943 else if( derivedT::c_stdCamera_usesROI && name == "roi_set_full_bin" )
1944 return newCallBack_roi_fullbin( ipRecv );
1945 else if( derivedT::c_stdCamera_usesROI && name == "roi_load_last" )
1946 return newCallBack_roi_loadlast( ipRecv );
1947 else if( derivedT::c_stdCamera_usesROI && name == "roi_set_last" )
1948 return newCallBack_roi_last( ipRecv );
1949 else if( derivedT::c_stdCamera_usesROI && name == "roi_set_default" )
1950 return newCallBack_roi_default( ipRecv );
1951 else if( derivedT::c_stdCamera_hasShutter && name == "shutter" )
1952 return newCallBack_shutter( ipRecv );
1953
1954#ifndef XWCTEST_INDI_CALLBACK_VALIDATION
1955 derivedT::template log<software_error>( { __FILE__, __LINE__, "unknown INDI property" } );
1956#endif
1957
1958 return -1;
1959}
1960
1961template <class derivedT>
1962int stdCamera<derivedT>::setTempSetPt( const mx::meta::trueFalseT<true> &t )
1963{
1964 static_cast<void>( t );
1965 return derived().setTempSetPt();
1966}
1967
1968template <class derivedT>
1969int stdCamera<derivedT>::setTempSetPt( const mx::meta::trueFalseT<false> &f )
1970{
1971 static_cast<void>( f );
1972 return 0;
1973}
1974
1975template <class derivedT>
1976int stdCamera<derivedT>::newCallBack_temp( const pcf::IndiProperty &ipRecv )
1977{
1978 if( derivedT::c_stdCamera_tempControl )
1979 {
1980#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
1981 return 0;
1982#endif
1983
1984 float target;
1985
1986 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
1987
1988 if( derived().indiTargetUpdate( m_indiP_temp, target, ipRecv, true ) < 0 )
1989 {
1990 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
1991 return -1;
1992 }
1993
1994 m_ccdTempSetpt = target;
1995
1996 mx::meta::trueFalseT<derivedT::c_stdCamera_tempControl> tf;
1997 return setTempSetPt( tf );
1998 }
1999 else
2000 {
2001 return 0;
2002 }
2003}
2004
2005template <class derivedT>
2006int stdCamera<derivedT>::setTempControl( const mx::meta::trueFalseT<true> &t )
2007{
2008 static_cast<void>( t );
2009 return derived().setTempControl();
2010}
2011
2012template <class derivedT>
2013int stdCamera<derivedT>::setTempControl( const mx::meta::trueFalseT<false> &f )
2014{
2015 static_cast<void>( f );
2016 return 0;
2017}
2018
2019template <class derivedT>
2021{
2022 if( derivedT::c_stdCamera_tempControl )
2023 {
2024#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2025 return 0;
2026#endif
2027
2028 if( !ipRecv.find( "toggle" ) )
2029 return 0;
2030
2031 m_tempControlStatusSet = false;
2032
2033 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2034
2035 if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::On )
2036 {
2037 m_tempControlStatusSet = true;
2038 derived().updateSwitchIfChanged( m_indiP_tempcont, "toggle", pcf::IndiElement::On, INDI_BUSY );
2039 }
2040 else if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::Off )
2041 {
2042 m_tempControlStatusSet = false;
2043 derived().updateSwitchIfChanged( m_indiP_tempcont, "toggle", pcf::IndiElement::Off, INDI_BUSY );
2044 }
2045
2046 mx::meta::trueFalseT<derivedT::c_stdCamera_emGain> tf;
2047 return setTempControl( tf );
2048 }
2049 else
2050 {
2051 return 0;
2052 }
2053}
2054
2055template <class derivedT>
2056int stdCamera<derivedT>::setReadoutSpeed( const mx::meta::trueFalseT<true> &t )
2057{
2058 static_cast<void>( t );
2059 return derived().setReadoutSpeed();
2060}
2061
2062template <class derivedT>
2063int stdCamera<derivedT>::setReadoutSpeed( const mx::meta::trueFalseT<false> &f )
2064{
2065 static_cast<void>( f );
2066 return 0;
2067}
2068
2069template <class derivedT>
2071{
2072 if( derivedT::c_stdCamera_readoutSpeed )
2073 {
2074#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2075 return 0;
2076#endif
2077
2078 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2079
2080 std::string newspeed;
2081
2082 for( size_t i = 0; i < m_readoutSpeedNames.size(); ++i )
2083 {
2084 if( !ipRecv.find( m_readoutSpeedNames[i] ) )
2085 continue;
2086
2087 if( ipRecv[m_readoutSpeedNames[i]].getSwitchState() == pcf::IndiElement::On )
2088 {
2089 if( newspeed != "" )
2090 {
2091 derivedT::template log<text_log>( "More than one readout speed selected", logPrio::LOG_ERROR );
2092 return -1;
2093 }
2094
2095 newspeed = m_readoutSpeedNames[i];
2096 }
2097 }
2098
2099 if( newspeed == "" )
2100 {
2101 // We do a reset
2102 m_readoutSpeedNameSet = m_readoutSpeedName;
2103 }
2104 else
2105 {
2106 m_readoutSpeedNameSet = newspeed;
2107 }
2108
2109 mx::meta::trueFalseT<derivedT::c_stdCamera_readoutSpeed> tf;
2110 return setReadoutSpeed( tf );
2111 }
2112
2113 return 0;
2114}
2115
2116template <class derivedT>
2117int stdCamera<derivedT>::setVShiftSpeed( const mx::meta::trueFalseT<true> &t )
2118{
2119 static_cast<void>( t );
2120 return derived().setVShiftSpeed();
2121}
2122
2123template <class derivedT>
2124int stdCamera<derivedT>::setVShiftSpeed( const mx::meta::trueFalseT<false> &f )
2125{
2126 static_cast<void>( f );
2127 return 0;
2128}
2129
2130template <class derivedT>
2132{
2133 if( derivedT::c_stdCamera_vShiftSpeed )
2134 {
2135#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2136 return 0;
2137#endif
2138
2139 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2140
2141 std::string newspeed;
2142
2143 for( size_t i = 0; i < m_vShiftSpeedNames.size(); ++i )
2144 {
2145 if( !ipRecv.find( m_vShiftSpeedNames[i] ) )
2146 continue;
2147
2148 if( ipRecv[m_vShiftSpeedNames[i]].getSwitchState() == pcf::IndiElement::On )
2149 {
2150 if( newspeed != "" )
2151 {
2152 derivedT::template log<text_log>( "More than one vShift speed selected", logPrio::LOG_ERROR );
2153 return -1;
2154 }
2155
2156 newspeed = m_vShiftSpeedNames[i];
2157 }
2158 }
2159
2160 if( newspeed == "" )
2161 {
2162 // We do a reset
2163 m_vShiftSpeedNameSet = m_vShiftSpeedName;
2164 }
2165 else
2166 {
2167 m_vShiftSpeedNameSet = newspeed;
2168 }
2169
2170 mx::meta::trueFalseT<derivedT::c_stdCamera_vShiftSpeed> tf;
2171 return setVShiftSpeed( tf );
2172 }
2173
2174 return 0;
2175}
2176
2177template <class derivedT>
2178int stdCamera<derivedT>::setEMGain( const mx::meta::trueFalseT<true> &t )
2179{
2180 static_cast<void>( t );
2181 return derived().setEMGain();
2182}
2183
2184template <class derivedT>
2185int stdCamera<derivedT>::setEMGain( const mx::meta::trueFalseT<false> &f )
2186{
2187 static_cast<void>( f );
2188 return 0;
2189}
2190
2191template <class derivedT>
2193{
2194 if( derivedT::c_stdCamera_emGain )
2195 {
2196#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2197 return 0;
2198#endif
2199
2200 float target;
2201
2202 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2203
2204 if( derived().indiTargetUpdate( m_indiP_emGain, target, ipRecv, true ) < 0 )
2205 {
2206 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
2207 return -1;
2208 }
2209
2210 m_emGainSet = target;
2211
2212 mx::meta::trueFalseT<derivedT::c_stdCamera_emGain> tf;
2213 return setEMGain( tf );
2214 }
2215
2216 return 0;
2217}
2218
2219template <class derivedT>
2220int stdCamera<derivedT>::setExpTime( const mx::meta::trueFalseT<true> &t )
2221{
2222 static_cast<void>( t );
2223 return derived().setExpTime();
2224}
2225
2226template <class derivedT>
2227int stdCamera<derivedT>::setExpTime( const mx::meta::trueFalseT<false> &f )
2228{
2229 static_cast<void>( f );
2230 return 0;
2231}
2232
2233template <class derivedT>
2235{
2236 if( derivedT::c_stdCamera_exptimeCtrl )
2237 {
2238#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2239 return 0;
2240#endif
2241
2242 float target;
2243
2244 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2245
2246 if( derived().indiTargetUpdate( m_indiP_exptime, target, ipRecv, true ) < 0 )
2247 {
2248 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
2249 return -1;
2250 }
2251
2252 m_expTimeSet = target;
2253
2254 mx::meta::trueFalseT<derivedT::c_stdCamera_exptimeCtrl> tf;
2255 return setExpTime( tf );
2256 }
2257
2258 return 0;
2259}
2260
2261template <class derivedT>
2262int stdCamera<derivedT>::setFPS( const mx::meta::trueFalseT<true> &t )
2263{
2264 static_cast<void>( t );
2265 return derived().setFPS();
2266}
2267
2268template <class derivedT>
2269int stdCamera<derivedT>::setFPS( const mx::meta::trueFalseT<false> &f )
2270{
2271 static_cast<void>( f );
2272 return 0;
2273}
2274
2275template <class derivedT>
2276int stdCamera<derivedT>::newCallBack_fps( const pcf::IndiProperty &ipRecv )
2277{
2278 if( derivedT::c_stdCamera_fpsCtrl )
2279 {
2280#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2281 return 0;
2282#endif
2283
2284 float target;
2285
2286 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2287
2288 if( derived().indiTargetUpdate( m_indiP_fps, target, ipRecv, true ) < 0 )
2289 {
2290 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
2291 return -1;
2292 }
2293
2294 m_fpsSet = target;
2295
2296 mx::meta::trueFalseT<derivedT::c_stdCamera_fpsCtrl> tf;
2297 return setFPS( tf );
2298 }
2299
2300 return 0;
2301}
2302
2303template <class derivedT>
2304int stdCamera<derivedT>::setSynchro( const mx::meta::trueFalseT<true> &t )
2305{
2306 static_cast<void>( t );
2307 return derived().setSynchro();
2308}
2309
2310template <class derivedT>
2311int stdCamera<derivedT>::setSynchro( const mx::meta::trueFalseT<false> &f )
2312{
2313 static_cast<void>( f );
2314 return 0;
2315}
2316
2317template <class derivedT>
2319{
2320 if( derivedT::c_stdCamera_synchro )
2321 {
2322#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2323 return 0;
2324#endif
2325
2326 if( !ipRecv.find( "toggle" ) )
2327 return 0;
2328
2329 if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::Off )
2330 {
2331 m_synchroSet = false;
2332 }
2333
2334 if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::On )
2335 {
2336 m_synchroSet = true;
2337 }
2338
2339 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2340
2341 mx::meta::trueFalseT<derivedT::c_stdCamera_synchro> tf;
2342 return setSynchro( tf );
2343 }
2344
2345 return 0;
2346}
2347
2348template <class derivedT>
2349int stdCamera<derivedT>::newCallBack_mode( const pcf::IndiProperty &ipRecv )
2350{
2351 if( derivedT::c_stdCamera_usesModes )
2352 {
2353#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2354 return 0;
2355#endif
2356
2357 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2358
2359 if( ipRecv.getName() != m_indiP_mode.getName() )
2360 {
2361 derivedT::template log<software_error>( { __FILE__, __LINE__, "invalid indi property received" } );
2362 return -1;
2363 }
2364
2365 // look for selected mode switch which matches a known mode. Make sure only one is selected.
2366 std::string newName = "";
2367 for( auto it = m_cameraModes.begin(); it != m_cameraModes.end(); ++it )
2368 {
2369 if( !ipRecv.find( it->first ) )
2370 continue;
2371
2372 if( ipRecv[it->first].getSwitchState() == pcf::IndiElement::On )
2373 {
2374 if( newName != "" )
2375 {
2376 derivedT::template log<text_log>( "More than one camera mode selected", logPrio::LOG_ERROR );
2377 return -1;
2378 }
2379
2380 newName = it->first;
2381 }
2382 }
2383
2384 if( newName == "" )
2385 {
2386 return 0;
2387 }
2388
2389 // Now signal the f.g. thread to reconfigure
2390 m_nextMode = newName;
2391 derived().m_reconfig = true;
2392
2393 return 0;
2394 }
2395 else
2396 {
2397 return 0;
2398 }
2399}
2400
2401template <class derivedT>
2403{
2404#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2405 return 0;
2406#endif
2407
2408 if( !ipRecv.find( "request" ) )
2409 return 0;
2410
2411 if( ipRecv["request"].getSwitchState() == pcf::IndiElement::On )
2412 {
2413 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2414
2416 m_indiP_reconfig, "request", pcf::IndiElement::Off, derived().m_indiDriver, INDI_IDLE );
2417
2418 m_nextMode = m_modeName;
2419 derived().m_reconfig = true;
2420 return 0;
2421 }
2422
2423 return 0;
2424}
2425
2426template <class derivedT>
2427int stdCamera<derivedT>::setCropMode( const mx::meta::trueFalseT<true> &t )
2428{
2429 static_cast<void>( t );
2430 return derived().setCropMode();
2431}
2432
2433template <class derivedT>
2434int stdCamera<derivedT>::setCropMode( const mx::meta::trueFalseT<false> &f )
2435{
2436 static_cast<void>( f );
2437 return 0;
2438}
2439
2440template <class derivedT>
2442{
2443 if( derivedT::c_stdCamera_cropMode )
2444 {
2445#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2446 return 0;
2447#endif
2448
2449 if( !ipRecv.find( "toggle" ) )
2450 return 0;
2451
2452 if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::Off )
2453 {
2454 m_cropModeSet = false;
2455 }
2456
2457 if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::On )
2458 {
2459 m_cropModeSet = true;
2460 }
2461
2462 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2463
2464 mx::meta::trueFalseT<derivedT::c_stdCamera_cropMode> tf;
2465 return setCropMode( tf );
2466 }
2467
2468 return 0;
2469}
2470
2471///\todo why don't these check if usesROI is true?
2472template <class derivedT>
2473int stdCamera<derivedT>::newCallBack_roi_x( const pcf::IndiProperty &ipRecv )
2474{
2475#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2476 return 0;
2477#endif
2478
2479 float target;
2480
2481 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2482
2483 if( derived().indiTargetUpdate( m_indiP_roi_x, target, ipRecv, false ) < 0 )
2484 {
2485 m_nextROI.x = m_currentROI.x;
2486 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
2487 return -1;
2488 }
2489
2490 m_nextROI.x = target;
2491
2492 return 0;
2493}
2494
2495template <class derivedT>
2496int stdCamera<derivedT>::newCallBack_roi_y( const pcf::IndiProperty &ipRecv )
2497{
2498#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2499 return 0;
2500#endif
2501
2502 float target;
2503
2504 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2505
2506 if( derived().indiTargetUpdate( m_indiP_roi_y, target, ipRecv, false ) < 0 )
2507 {
2508 m_nextROI.y = m_currentROI.y;
2509 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
2510 return -1;
2511 }
2512
2513 m_nextROI.y = target;
2514
2515 return 0;
2516}
2517
2518template <class derivedT>
2519int stdCamera<derivedT>::newCallBack_roi_w( const pcf::IndiProperty &ipRecv )
2520{
2521#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2522 return 0;
2523#endif
2524
2525 int target;
2526
2527 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2528
2529 if( derived().indiTargetUpdate( m_indiP_roi_w, target, ipRecv, false ) < 0 )
2530 {
2531 m_nextROI.w = m_currentROI.w;
2532 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
2533 return -1;
2534 }
2535
2536 m_nextROI.w = target;
2537
2538 return 0;
2539}
2540
2541template <class derivedT>
2542int stdCamera<derivedT>::newCallBack_roi_h( const pcf::IndiProperty &ipRecv )
2543{
2544#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2545 return 0;
2546#endif
2547
2548 int target;
2549
2550 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2551
2552 if( derived().indiTargetUpdate( m_indiP_roi_h, target, ipRecv, false ) < 0 )
2553 {
2554 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
2555 m_nextROI.h = m_currentROI.h;
2556 return -1;
2557 }
2558
2559 m_nextROI.h = target;
2560
2561 return 0;
2562}
2563
2564template <class derivedT>
2566{
2567#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2568 return 0;
2569#endif
2570
2571 int target;
2572
2573 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2574
2575 if( derived().indiTargetUpdate( m_indiP_roi_bin_x, target, ipRecv, false ) < 0 )
2576 {
2577 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
2578 m_nextROI.bin_x = m_currentROI.bin_x;
2579 return -1;
2580 }
2581
2582 m_nextROI.bin_x = target;
2583
2584 return 0;
2585}
2586
2587template <class derivedT>
2589{
2590#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2591 return 0;
2592#endif
2593
2594 int target;
2595
2596 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2597
2598 if( derived().indiTargetUpdate( m_indiP_roi_bin_y, target, ipRecv, false ) < 0 )
2599 {
2600 derivedT::template log<software_error>( { __FILE__, __LINE__ } );
2601 m_nextROI.bin_y = m_currentROI.bin_y;
2602 return -1;
2603 }
2604
2605 m_nextROI.bin_y = target;
2606
2607 return 0;
2608}
2609
2610template <class derivedT>
2611int stdCamera<derivedT>::checkNextROI( const mx::meta::trueFalseT<true> &t )
2612{
2613 static_cast<void>( t );
2614 return derived().checkNextROI();
2615}
2616
2617template <class derivedT>
2618int stdCamera<derivedT>::checkNextROI( const mx::meta::trueFalseT<false> &f )
2619{
2620 static_cast<void>( f );
2621 return 0;
2622}
2623
2624template <class derivedT>
2626{
2627 if( derivedT::c_stdCamera_usesROI )
2628 {
2629#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2630 return 0;
2631#endif
2632
2633 if( !ipRecv.find( "request" ) )
2634 return 0;
2635
2636 if( ipRecv["request"].getSwitchState() == pcf::IndiElement::On )
2637 {
2638 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2639
2641 m_indiP_roi_check, "request", pcf::IndiElement::Off, derived().m_indiDriver, INDI_IDLE );
2642
2643 mx::meta::trueFalseT<derivedT::c_stdCamera_usesROI> tf;
2644 return checkNextROI( tf );
2645 }
2646
2647 return 0;
2648 }
2649
2650 return 0;
2651}
2652
2653template <class derivedT>
2654int stdCamera<derivedT>::setNextROI( const mx::meta::trueFalseT<true> &t )
2655{
2656 static_cast<void>( t );
2657 return derived().setNextROI();
2658}
2659
2660template <class derivedT>
2661int stdCamera<derivedT>::setNextROI( const mx::meta::trueFalseT<false> &f )
2662{
2663 static_cast<void>( f );
2664 return 0;
2665}
2666
2667template <class derivedT>
2669{
2670 if( derivedT::c_stdCamera_usesROI )
2671 {
2672#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2673 return 0;
2674#endif
2675
2676 if( !ipRecv.find( "request" ) )
2677 return 0;
2678
2679 if( ipRecv["request"].getSwitchState() == pcf::IndiElement::On )
2680 {
2681 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2682
2684 m_indiP_roi_set, "request", pcf::IndiElement::Off, derived().m_indiDriver, INDI_IDLE );
2685
2686 m_lastROI = m_currentROI;
2687
2688 mx::meta::trueFalseT<derivedT::c_stdCamera_usesROI> tf;
2689 return setNextROI( tf );
2690 }
2691
2692 return 0;
2693 }
2694
2695 return 0;
2696}
2697
2698template <class derivedT>
2700{
2701 if( derivedT::c_stdCamera_usesROI )
2702 {
2703#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2704 return 0;
2705#endif
2706
2707 if( !ipRecv.find( "request" ) )
2708 return 0;
2709
2710 if( ipRecv["request"].getSwitchState() == pcf::IndiElement::On )
2711 {
2712 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2713
2715 m_indiP_roi_full, "request", pcf::IndiElement::Off, derived().m_indiDriver, INDI_IDLE );
2716
2717 m_nextROI.x = m_full_x;
2718 m_nextROI.y = m_full_y;
2719 m_nextROI.w = m_full_w;
2720 m_nextROI.h = m_full_h;
2721 m_nextROI.bin_x = m_full_bin_x;
2722 m_nextROI.bin_y = m_full_bin_y;
2723 m_lastROI = m_currentROI;
2724 mx::meta::trueFalseT<derivedT::c_stdCamera_usesROI> tf;
2725 return setNextROI( tf );
2726 }
2727
2728 return 0;
2729 }
2730
2731 return 0;
2732}
2733
2734template <class derivedT>
2736{
2737 if( derivedT::c_stdCamera_usesROI )
2738 {
2739#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2740 return 0;
2741#endif
2742
2743 if( !ipRecv.find( "request" ) )
2744 return 0;
2745
2746 if( ipRecv["request"].getSwitchState() == pcf::IndiElement::On )
2747 {
2748 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2749
2751 m_indiP_roi_fullbin, "request", pcf::IndiElement::Off, derived().m_indiDriver, INDI_IDLE );
2752
2753 bool reset = false;
2754
2755 if( m_full_currbin_x == 0 ) // still defaulted
2756 {
2757 derivedT::template log<text_log>( "current-binning full ROI not implemented for this camera",
2759 m_full_currbin_x = m_full_x;
2760 m_full_currbin_y = m_full_y;
2761 m_full_currbin_w = m_full_w;
2762 m_full_currbin_h = m_full_h;
2763 reset = true;
2764 }
2765
2766 m_nextROI.x = m_full_currbin_x;
2767 m_nextROI.y = m_full_currbin_y;
2768 m_nextROI.w = m_full_currbin_w;
2769 m_nextROI.h = m_full_currbin_h;
2770 if( reset )
2771 {
2772 // Use full binning
2773 m_nextROI.bin_x = m_full_bin_x;
2774 m_nextROI.bin_y = m_full_bin_y;
2775
2776 // restore defaults for next time
2777 m_full_currbin_x = 0;
2778 m_full_currbin_y = 0;
2779 m_full_currbin_w = 0;
2780 m_full_currbin_h = 0;
2781 }
2782 else
2783 {
2784 m_nextROI.bin_x = m_currentROI.bin_x;
2785 m_nextROI.bin_y = m_currentROI.bin_y;
2786 }
2787
2788 m_lastROI = m_currentROI;
2789 mx::meta::trueFalseT<derivedT::c_stdCamera_usesROI> tf;
2790 return setNextROI( tf );
2791 }
2792
2793 return 0;
2794 }
2795
2796 return 0;
2797}
2798
2799template <class derivedT>
2801{
2802 if( derivedT::c_stdCamera_usesROI )
2803 {
2804#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2805 return 0;
2806#endif
2807
2808 if( !ipRecv.find( "request" ) )
2809 return 0;
2810
2811 if( ipRecv["request"].getSwitchState() == pcf::IndiElement::On )
2812 {
2813 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2814
2816 m_indiP_roi_loadlast, "request", pcf::IndiElement::Off, derived().m_indiDriver, INDI_IDLE );
2817
2818 m_nextROI = m_lastROI;
2819 return 0;
2820 }
2821
2822 return 0;
2823 }
2824
2825 return 0;
2826}
2827
2828template <class derivedT>
2830{
2831 if( derivedT::c_stdCamera_usesROI )
2832 {
2833#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2834 return 0;
2835#endif
2836 if( !ipRecv.find( "request" ) )
2837 return 0;
2838
2839 if( ipRecv["request"].getSwitchState() == pcf::IndiElement::On )
2840 {
2841 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2842
2844 m_indiP_roi_last, "request", pcf::IndiElement::Off, derived().m_indiDriver, INDI_IDLE );
2845
2846 m_nextROI = m_lastROI;
2847 m_lastROI = m_currentROI;
2848 mx::meta::trueFalseT<derivedT::c_stdCamera_usesROI> tf;
2849 return setNextROI( tf );
2850 }
2851
2852 return 0;
2853 }
2854
2855 return 0;
2856}
2857
2858template <class derivedT>
2860{
2861 if( derivedT::c_stdCamera_usesROI )
2862 {
2863#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2864 return 0;
2865#endif
2866
2867 if( !ipRecv.find( "request" ) )
2868 return 0;
2869
2870 if( ipRecv["request"].getSwitchState() == pcf::IndiElement::On )
2871 {
2872 std::unique_lock<std::mutex> lock( derived().m_indiMutex );
2873
2875 m_indiP_roi_default, "request", pcf::IndiElement::Off, derived().m_indiDriver, INDI_IDLE );
2876
2877 m_nextROI.x = m_default_x;
2878 m_nextROI.y = m_default_y;
2879 m_nextROI.w = m_default_w;
2880 m_nextROI.h = m_default_h;
2881 m_nextROI.bin_x = m_default_bin_x;
2882 m_nextROI.bin_y = m_default_bin_y;
2883 m_lastROI = m_currentROI;
2884 mx::meta::trueFalseT<derivedT::c_stdCamera_usesROI> tf;
2885 return setNextROI( tf );
2886 }
2887
2888 return 0;
2889 }
2890
2891 return 0;
2892}
2893
2894template <class derivedT>
2895int stdCamera<derivedT>::setShutter( int ss, const mx::meta::trueFalseT<true> &t )
2896{
2897 static_cast<void>( t );
2898 return derived().setShutter( ss );
2899}
2900
2901template <class derivedT>
2902int stdCamera<derivedT>::setShutter( int ss, const mx::meta::trueFalseT<false> &f )
2903{
2904 static_cast<void>( ss );
2905 static_cast<void>( f );
2906 return 0;
2907}
2908
2909template <class derivedT>
2911{
2912 if( derivedT::c_stdCamera_hasShutter )
2913 {
2914#ifdef XWCTEST_INDI_CALLBACK_VALIDATION
2915 return 0;
2916#endif
2917
2918 if( !ipRecv.find( "toggle" ) )
2919 return 0;
2920
2921 mx::meta::trueFalseT<derivedT::c_stdCamera_hasShutter> tf;
2922
2923 if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::Off )
2924 {
2925 setShutter( 1, tf );
2926 }
2927
2928 if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::On )
2929 {
2930 setShutter( 0, tf );
2931 }
2932
2933 return 0;
2934 }
2935 return 0;
2936}
2937
2938template <class derivedT>
2939std::string stdCamera<derivedT>::stateString( const mx::meta::trueFalseT<true> &t )
2940{
2941 static_cast<void>( t );
2942 return derived().stateString();
2943}
2944
2945template <class derivedT>
2946std::string stdCamera<derivedT>::stateString( const mx::meta::trueFalseT<false> &f )
2947{
2948 static_cast<void>( f );
2949 return "";
2950}
2951
2952template <class derivedT>
2953bool stdCamera<derivedT>::stateStringValid( const mx::meta::trueFalseT<true> &t )
2954{
2955 static_cast<void>( t );
2956 return derived().stateStringValid();
2957}
2958
2959template <class derivedT>
2960bool stdCamera<derivedT>::stateStringValid( const mx::meta::trueFalseT<false> &f )
2961{
2962 static_cast<void>( f );
2963 return false;
2964}
2965
2966template <class derivedT>
2968{
2969 try
2970 {
2971 if( !derived().m_indiDriver )
2972 return 0;
2973
2974 if( derivedT::c_stdCamera_readoutSpeed )
2975 {
2977 m_indiP_readoutSpeed, m_readoutSpeedName, derived().m_indiDriver, INDI_OK );
2978 }
2979
2980 if( derivedT::c_stdCamera_vShiftSpeed )
2981 {
2983 m_indiP_vShiftSpeed, m_vShiftSpeedName, derived().m_indiDriver, INDI_OK );
2984 }
2985
2986 if( derivedT::c_stdCamera_emGain )
2987 {
2988 derived().updateIfChanged( m_indiP_emGain, "current", m_emGain, INDI_IDLE );
2989 derived().updateIfChanged( m_indiP_emGain, "target", m_emGainSet, INDI_IDLE );
2990 }
2991
2992 if( derivedT::c_stdCamera_exptimeCtrl )
2993 {
2994 derived().updateIfChanged( m_indiP_exptime, "current", m_expTime, INDI_IDLE );
2995 derived().updateIfChanged( m_indiP_exptime, "target", m_expTimeSet, INDI_IDLE );
2996 }
2997
2998 if( derivedT::c_stdCamera_fpsCtrl )
2999 {
3000 derived().updateIfChanged( m_indiP_fps, "current", m_fps, INDI_IDLE );
3001 derived().updateIfChanged( m_indiP_fps, "target", m_fpsSet, INDI_IDLE );
3002 }
3003 else if( derivedT::c_stdCamera_fps )
3004 {
3005 derived().updateIfChanged( m_indiP_fps, "current", m_fps, INDI_IDLE );
3006 }
3007
3008 if( derivedT::c_stdCamera_synchro )
3009 {
3010 if( m_synchro == false )
3011 {
3012 derived().updateSwitchIfChanged( m_indiP_synchro, "toggle", pcf::IndiElement::Off, INDI_IDLE );
3013 }
3014 else
3015 {
3016 derived().updateSwitchIfChanged( m_indiP_synchro, "toggle", pcf::IndiElement::On, INDI_OK );
3017 }
3018 }
3019
3020 if( derivedT::c_stdCamera_usesModes )
3021 {
3022 auto st = pcf::IndiProperty::Ok;
3023 if( m_nextMode != "" )
3024 st = pcf::IndiProperty::Busy;
3025
3026 for( auto it = m_cameraModes.begin(); it != m_cameraModes.end(); ++it )
3027 {
3028 if( it->first == m_modeName )
3029 derived().updateSwitchIfChanged( m_indiP_mode, it->first, pcf::IndiElement::On, st );
3030 else
3031 derived().updateSwitchIfChanged( m_indiP_mode, it->first, pcf::IndiElement::Off, st );
3032 }
3033 }
3034
3035 if( derivedT::c_stdCamera_cropMode )
3036 {
3037 if( m_cropMode == false )
3038 {
3039 derived().updateSwitchIfChanged( m_indiP_cropMode, "toggle", pcf::IndiElement::Off, INDI_IDLE );
3040 }
3041 else
3042 {
3043 derived().updateSwitchIfChanged( m_indiP_cropMode, "toggle", pcf::IndiElement::On, INDI_OK );
3044 }
3045 }
3046
3047 if( derivedT::c_stdCamera_usesROI )
3048 {
3049 // These can't change after initialization, but might not be discoverable until powered on and connected.
3050 // so we'll check every time I guess.
3051 derived().updateIfChanged( m_indiP_fullROI, "x", m_full_x, INDI_IDLE );
3052 derived().updateIfChanged( m_indiP_fullROI, "y", m_full_y, INDI_IDLE );
3053 derived().updateIfChanged( m_indiP_fullROI, "w", m_full_w, INDI_IDLE );
3054 derived().updateIfChanged( m_indiP_fullROI, "h", m_full_h, INDI_IDLE );
3055 }
3056
3057 if( derivedT::c_stdCamera_tempControl )
3058 {
3059 if( m_tempControlStatus == false )
3060 {
3061 derived().updateSwitchIfChanged( m_indiP_tempcont, "toggle", pcf::IndiElement::Off, INDI_IDLE );
3062 derived().updateIfChanged( m_indiP_temp, "current", m_ccdTemp, INDI_IDLE );
3063 derived().updateIfChanged( m_indiP_temp, "target", m_ccdTempSetpt, INDI_IDLE );
3064 derived().updateIfChanged( m_indiP_tempstat, "status", m_tempControlStatusStr, INDI_IDLE );
3065 }
3066 else
3067 {
3068 if( m_tempControlOnTarget )
3069 {
3070 derived().updateSwitchIfChanged( m_indiP_tempcont, "toggle", pcf::IndiElement::On, INDI_OK );
3071 derived().updateIfChanged( m_indiP_temp, "current", m_ccdTemp, INDI_OK );
3072 derived().updateIfChanged( m_indiP_temp, "target", m_ccdTempSetpt, INDI_OK );
3073 derived().updateIfChanged( m_indiP_tempstat, "status", m_tempControlStatusStr, INDI_OK );
3074 }
3075 else
3076 {
3077 derived().updateSwitchIfChanged( m_indiP_tempcont, "toggle", pcf::IndiElement::On, INDI_BUSY );
3078 derived().updateIfChanged( m_indiP_temp, "current", m_ccdTemp, INDI_BUSY );
3079 derived().updateIfChanged( m_indiP_temp, "target", m_ccdTempSetpt, INDI_BUSY );
3080 derived().updateIfChanged( m_indiP_tempstat, "status", m_tempControlStatusStr, INDI_BUSY );
3081 }
3082 }
3083 }
3084 else if( derivedT::c_stdCamera_temp )
3085 {
3086 derived().updateIfChanged( m_indiP_temp, "current", m_ccdTemp, INDI_IDLE );
3087 }
3088
3089 if( derivedT::c_stdCamera_hasShutter )
3090 {
3091 if( m_shutterStatus == "OPERATING" )
3092 {
3093 derived().updateIfChanged( m_indiP_shutterStatus, "status", m_shutterStatus, INDI_BUSY );
3094 }
3095 if( m_shutterStatus == "POWERON" || m_shutterStatus == "READY" )
3096 {
3097 derived().updateIfChanged( m_indiP_shutterStatus, "status", m_shutterStatus, INDI_IDLE );
3098 }
3099 else
3100 {
3101 derived().updateIfChanged( m_indiP_shutterStatus, "status", m_shutterStatus, INDI_IDLE );
3102 }
3103
3104 if( m_shutterState == 0 ) // 0 shut, 1 open
3105 {
3106 derived().updateSwitchIfChanged( m_indiP_shutter, "toggle", pcf::IndiElement::On, INDI_OK );
3107 }
3108 else
3109 {
3110 derived().updateSwitchIfChanged( m_indiP_shutter, "toggle", pcf::IndiElement::Off, INDI_IDLE );
3111 }
3112 }
3113
3114 if( derivedT::c_stdCamera_usesStateString )
3115 {
3116 mx::meta::trueFalseT<derivedT::c_stdCamera_usesStateString> tf;
3117 derived().updateIfChanged( m_indiP_stateString, "current", stateString( tf ), INDI_IDLE );
3118 if( stateStringValid( tf ) )
3119 {
3120 derived().updateIfChanged( m_indiP_stateString, "valid", "yes", INDI_IDLE );
3121 }
3122 else
3123 {
3124 derived().updateIfChanged( m_indiP_stateString, "valid", "no", INDI_IDLE );
3125 }
3126 }
3127 return 0;
3128 }
3129 catch( const std::exception &e )
3130 {
3131 return derivedT::template log<software_error, -1>(
3132 { __FILE__, __LINE__, std::string( "Exception caught: " ) + e.what() } );
3133 }
3134}
3135
3136template <class derivedT>
3138{
3139 static std::string last_mode;
3140 static roi last_roi;
3141 static float last_expTime = -1e30; // ensure first one goes
3142 static float last_fps = 0;
3143 static float last_adcSpeed = -1;
3144 static float last_emGain = -1;
3145 static float last_ccdTemp = 0;
3146 static float last_ccdTempSetpt = 0;
3147 static bool last_tempControlStatus = 0;
3148 static bool last_tempControlOnTarget = 0;
3149 static std::string last_tempControlStatusStr;
3150 static std::string last_shutterStatus;
3151 static int last_shutterState = false;
3152 static bool last_synchro = false;
3153 static float last_vshiftSpeed = -1;
3154 static bool last_cropMode = false;
3155
3156 if( force || m_modeName != last_mode || m_currentROI.x != last_roi.x || m_currentROI.y != last_roi.y ||
3157 m_currentROI.w != last_roi.w || m_currentROI.h != last_roi.h || m_currentROI.bin_x != last_roi.bin_x ||
3158 m_currentROI.bin_y != last_roi.bin_y || m_expTime != last_expTime || m_fps != last_fps ||
3159 m_emGain != last_emGain || m_adcSpeed != last_adcSpeed || m_ccdTemp != last_ccdTemp ||
3160 m_ccdTempSetpt != last_ccdTempSetpt || m_tempControlStatus != last_tempControlStatus ||
3161 m_tempControlOnTarget != last_tempControlOnTarget || m_tempControlStatusStr != last_tempControlStatusStr ||
3162 m_shutterStatus != last_shutterStatus || m_shutterState != last_shutterState || m_synchro != last_synchro ||
3163 m_vshiftSpeed != last_vshiftSpeed || m_cropMode != last_cropMode )
3164 {
3165 derived().template telem<telem_stdcam>( { m_modeName,
3166 m_currentROI.x,
3167 m_currentROI.y,
3168 m_currentROI.w,
3169 m_currentROI.h,
3170 m_currentROI.bin_x,
3171 m_currentROI.bin_y,
3172 m_expTime,
3173 m_fps,
3174 m_emGain,
3175 m_adcSpeed,
3176 m_ccdTemp,
3177 m_ccdTempSetpt,
3178 (uint8_t)m_tempControlStatus,
3179 (uint8_t)m_tempControlOnTarget,
3180 m_tempControlStatusStr,
3181 m_shutterStatus,
3182 (int8_t)m_shutterState,
3183 (uint8_t)m_synchro,
3184 m_vshiftSpeed,
3185 (uint8_t)m_cropMode } );
3186
3187 last_mode = m_modeName;
3188 last_roi = m_currentROI;
3189 last_expTime = m_expTime;
3190 last_fps = m_fps;
3191 last_emGain = m_emGain;
3192 last_adcSpeed = m_adcSpeed;
3193 last_ccdTemp = m_ccdTemp;
3194 last_ccdTempSetpt = m_ccdTempSetpt;
3195 last_tempControlStatus = m_tempControlStatus;
3196 last_tempControlOnTarget = m_tempControlOnTarget;
3197 last_tempControlStatusStr = m_tempControlStatusStr;
3198 last_shutterStatus = m_shutterStatus;
3199 last_shutterState = m_shutterState;
3200 last_synchro = m_synchro;
3201 last_vshiftSpeed = m_vshiftSpeed;
3202 last_cropMode = m_cropMode;
3203 }
3204
3205 return 0;
3206}
3207
3208
3209/// Call stdCameraT::setupConfig with error checking for stdCamera
3210/**
3211 * \param cfig the application configurator
3212 */
3213#define STDCAMERA_SETUP_CONFIG( cfig ) \
3214 if(stdCameraT::setupConfig(cfig) < 0) \
3215 { \
3216 log<software_error>({__FILE__, __LINE__, "Error from stdCameraT::setupConfig"}); \
3217 m_shutdown = true; \
3218 return; \
3219 }
3220
3221 /// Call stdCameraT::loadConfig with error checking for stdCamera
3222/** This must be inside a function that returns int, e.g. the standard loadConfigImpl.
3223 * \param cfig the application configurator
3224 */
3225#define STDCAMERA_LOAD_CONFIG( cfig ) \
3226 if(stdCameraT::loadConfig(cfig) < 0) \
3227 { \
3228 return log<software_error,-1>({__FILE__, __LINE__, "Error from stdCameraT::loadConfig"}); \
3229 }
3230
3231/// Call stdCameraT::appStartup with error checking for stdCamera
3232#define STDCAMERA_APP_STARTUP \
3233 if(stdCameraT::appStartup() < 0) \
3234 { \
3235 return log<software_error,-1>({__FILE__, __LINE__, "Error from stdCameraT::appStartup"}); \
3236 }
3237
3238/// Call stdCameraT::appLogic with error checking for stdCamera
3239#define STDCAMERA_APP_LOGIC \
3240 if(stdCameraT::appLogic() < 0) \
3241 { \
3242 return log<software_error,-1>({__FILE__, __LINE__, "Error from stdCameraT::appLogic"}); \
3243 }
3244
3245/// Call stdCameraT::updateINDI with error checking for stdCamera
3246#define STDCAMERA_UPDATE_INDI \
3247 if(stdCameraT::updateINDI() < 0) \
3248 { \
3249 return log<software_error,-1>({__FILE__, __LINE__, "Error from stdCameraT::updateINDI"}); \
3250 }
3251
3252/// Call stdCameraT::appShutdown with error checking for stdCamera
3253#define STDCAMERA_APP_SHUTDOWN \
3254 if(stdCameraT::appShutdown() < 0) \
3255 { \
3256 return log<software_error,-1>({__FILE__, __LINE__, "Error from stdCameraT::appShutdown"}); \
3257 }
3258
3259} // namespace dev
3260} // namespace app
3261} // namespace MagAOX
3262
3263#endif // stdCamera_hpp
MagAO-X standard camera interface.
float m_maxEMGain
The configurable maximum EM gain. To be enforced in derivedT.
std::vector< std::string > m_readoutSpeedNames
float m_minExpTime
The minimum exposure time, used for INDI attributes.
pcf::IndiProperty m_indiP_roi_fullbin
Property used to trigger setting the full in current binning ROI.
float m_fpsSet
The commanded fps, as set by user.
int newCallBack_temp(const pcf::IndiProperty &ipRecv)
Callback to process a NEW CCD temp request.
int newCallBack_cropMode(const pcf::IndiProperty &ipRecv)
Callback to process a NEW cropMode request.
bool m_synchroSet
Target status of m_synchro.
pcf::IndiProperty m_indiP_roi_y
Property used to set the ROI x center coordinate.
int createReadoutSpeed(const mx::meta::trueFalseT< true > &t)
int m_full_currbin_w
The current-binning full ROI width.
int setShutter(int ss, const mx::meta::trueFalseT< true > &t)
Interface to setShutter when the derivedT has a shutter.
float m_default_x
Power-on ROI center x coordinate.
std::string m_tempControlStatusStr
Camera specific description of temperature control status.
float m_emGain
The camera's current EM gain (if available).
int m_full_bin_x
The x-binning in the full ROI.
int newCallBack_readoutSpeed(const pcf::IndiProperty &ipRecv)
Callback to process a NEW readout speed request.
std::vector< std::string > m_readoutSpeedNameLabels
float m_full_currbin_x
The current-binning full ROI center x coordinate.
int m_full_w
The full ROI width.
float m_emGainSet
The camera's EM gain, as set by the user.
std::string m_vShiftSpeedNameSet
The user requested vshift speed name, to be set by derived()
bool m_cropModeSet
Desired status of crop mode ROIs, if enabled for this camera.
int newCallBack_roi_bin_y(const pcf::IndiProperty &ipRecv)
Callback to process a NEW bin_y request.
std::string m_defaultReadoutSpeed
The default readout speed of the camera.
pcf::IndiProperty m_indiP_shutterStatus
State of the shutter. 0 = shut, 1 = open, -1 = unknown.
float m_expTime
The current exposure time, in seconds.
int setExpTime(const mx::meta::trueFalseT< true > &t)
Interface to setExpTime when the derivedT uses exposure time controls.
std::string m_nextMode
The mode to be set by the next reconfiguration.
pcf::IndiProperty m_indiP_tempstat
int m_default_bin_x
Power-on ROI x binning.
int m_default_w
Power-on ROI width.
int setTempControl(const mx::meta::trueFalseT< true > &t)
Interface to setTempControl when the derivedT has temperature control.
int newCallBack_roi_h(const pcf::IndiProperty &ipRecv)
Callback to process a NEW roi_h request.
int recordCamera(bool force=false)
bool m_cropMode
Status of crop mode ROIs, if enabled for this camera.
float m_maxExpTime
The maximum exposure time, used for INDI attributes.
float m_stepFPS
The FPS step size, used for INDI attributes.
int newCallBack_mode(const pcf::IndiProperty &ipRecv)
Callback to process a NEW mode request.
int m_default_h
Power-on ROI height.
pcf::IndiProperty m_indiP_roi_last
Property used to trigger setting the last ROI.
pcf::IndiProperty m_indiP_temp
float m_expTimeSet
The exposure time, in seconds, as set by user.
pcf::IndiProperty m_indiP_roi_h
Property used to set the ROI height.
int m_full_h
The full ROI height.
pcf::IndiProperty m_indiP_synchro
int m_full_bin_y
The y-binning in the full ROI.
int checkNextROI(const mx::meta::trueFalseT< true > &t)
Interface to checkNextROI when the derivedT uses ROIs.
int setVShiftSpeed(const mx::meta::trueFalseT< true > &t)
Interface to setVShiftSpeed when the derivedT has vshift speed control.
float m_full_y
The full ROI center y coordinate.
static int st_newCallBack_stdCamera(void *app, const pcf::IndiProperty &ipRecv)
The static callback function to be registered for stdCamera properties.
float m_maxFPS
The maximum FPS, used for INDI attributes.
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.
pcf::IndiProperty m_indiP_fps
int newCallBack_roi_last(const pcf::IndiProperty &ipRecv)
Callback to process a NEW roi_last request.
int setupConfig(mx::app::appConfigurator &config)
Setup the configuration system.
pcf::IndiProperty m_indiP_roi_loadlast
Property used to trigger loading the last ROI as the target.
pcf::IndiProperty m_indiP_roi_w
Property used to set the ROI width.
int newCallBack_roi_set(const pcf::IndiProperty &ipRecv)
Callback to process a NEW roi_set request.
pcf::IndiProperty m_indiP_roi_default
Property used to trigger setting the default and startup ROI.
int newCallBack_stdCamera(const pcf::IndiProperty &ipRecv)
The callback function for stdCamera properties.
std::string m_modeName
The current mode name.
pcf::IndiProperty m_indiP_reconfig
Request switch which forces the framegrabber to go through the reconfigure process.
int setSynchro(const mx::meta::trueFalseT< true > &t)
Interface to setSynchro when the derivedT has synchronization.
pcf::IndiProperty m_indiP_roi_bin_x
Property used to set the ROI x binning.
int newCallBack_roi_loadlast(const pcf::IndiProperty &ipRecv)
Callback to process a NEW roi_loadlast request.
pcf::IndiProperty m_indiP_readoutSpeed
int setEMGain(const mx::meta::trueFalseT< true > &t)
Interface to setEMGain when the derivedT has EM Gain.
int newCallBack_vShiftSpeed(const pcf::IndiProperty &ipRecv)
Callback to process a NEW vshift speed request.
std::string m_startupMode
The camera mode to load during first init after a power-on.
pcf::IndiProperty m_indiP_roi_check
Property used to trigger checking the target ROI.
pcf::IndiProperty m_indiP_vShiftSpeed
int newCallBack_roi_check(const pcf::IndiProperty &ipRecv)
Callback to process a NEW roi_check request.
int newCallBack_roi_x(const pcf::IndiProperty &ipRecv)
Callback to process a NEW roi_x request.
float m_full_x
The full ROI center x coordinate.
float m_startupTemp
The temperature to set after a power-on. Set to <= -999 to not use [default].
int appLogic()
Application logic.
int m_full_currbin_h
The current-binning full ROI height.
int loadConfig(mx::app::appConfigurator &config)
load the configuration system results
pcf::IndiProperty m_indiP_exptime
int newCallBack_shutter(const pcf::IndiProperty &ipRecv)
Callback to process a NEW shutter request.
bool m_synchro
Status of synchronization, true is on, false is off.
std::string m_vShiftSpeedName
The current vshift speed name.
int newCallBack_roi_full(const pcf::IndiProperty &ipRecv)
Callback to process a NEW roi_full request.
int setCropMode(const mx::meta::trueFalseT< true > &t)
Interface to setCropMode when the derivedT has crop mode.
pcf::IndiProperty m_indiP_emGain
float m_ccdTemp
The current temperature, in C.
int updateINDI()
Update the INDI properties for this device controller.
std::vector< std::string > m_vShiftSpeedNames
int newCallBack_fps(const pcf::IndiProperty &ipRecv)
Callback to process a NEW fps request.
std::string m_defaultVShiftSpeed
The default readout speed of the camera.
int newCallBack_roi_y(const pcf::IndiProperty &ipRecv)
Callback to process a NEW roi_y request.
pcf::IndiProperty m_indiP_tempcont
bool m_tempControlOnTarget
Whether or not the temperature control system is on its target temperature.
int createVShiftSpeed(const mx::meta::trueFalseT< true > &t)
cameraConfigMap m_cameraModes
Map holding the possible camera mode configurations.
float m_stepExpTime
The maximum exposure time stepsize, used for INDI attributes.
int whilePowerOff()
Actions while powered off.
int setFPS(const mx::meta::trueFalseT< true > &t)
Interface to setFPS when the derivedT uses FPS controls.
int newCallBack_reconfigure(const pcf::IndiProperty &ipRecv)
Callback to process a NEW reconfigure request.
int newCallBack_emgain(const pcf::IndiProperty &ipRecv)
Callback to process a NEW EM gain request.
float m_full_currbin_y
The current-binning full ROI center y coordinate.
int appStartup()
Startup function.
int newCallBack_roi_bin_x(const pcf::IndiProperty &ipRecv)
Callback to process a NEW bin_x request.
std::vector< std::string > m_vShiftSpeedNameLabels
int setTempSetPt(const mx::meta::trueFalseT< true > &t)
Interface to setTempSetPt when the derivedT has temperature control.
pcf::IndiProperty m_indiP_roi_full
Property used to trigger setting the full ROI.
int newCallBack_synchro(const pcf::IndiProperty &ipRecv)
Callback to process a NEW synchro request.
int newCallBack_roi_default(const pcf::IndiProperty &ipRecv)
Callback to process a NEW roi_default request.
pcf::IndiProperty m_indiP_roi_set
Property used to trigger setting the ROI.
int appShutdown()
Application shutdown.
pcf::IndiProperty m_indiP_shutter
Property used to control the shutter, a switch.
int newCallBack_roi_w(const pcf::IndiProperty &ipRecv)
Callback to process a NEW roi_w request.
pcf::IndiProperty m_indiP_stateString
pcf::IndiProperty m_indiP_mode
Property used to report the current mode.
pcf::IndiProperty m_indiP_cropMode
Property used to toggle crop mode on and off.
float m_default_y
Power-on ROI center y coordinate.
int setNextROI(const mx::meta::trueFalseT< true > &t)
Interface to setNextROI when the derivedT uses ROIs.
pcf::IndiProperty m_indiP_fullROI
Property used to preset the full ROI dimensions.
bool stateStringValid(const mx::meta::trueFalseT< true > &t)
Interface to stateStringValid when the derivedT provides it.
float m_fps
The current FPS.
std::string stateString(const mx::meta::trueFalseT< true > &t)
Interface to stateString when the derivedT provides it.
pcf::IndiProperty m_indiP_roi_bin_y
Property used to set the ROI y binning.
pcf::IndiProperty m_indiP_roi_x
Property used to set the ROI x center coordinate.
int newCallBack_roi_fullbin(const pcf::IndiProperty &ipRecv)
Callback to process a NEW roi_fullbin request.
int newCallBack_temp_controller(const pcf::IndiProperty &ipRecv)
Callback to process a NEW CCD temp control request.
int newCallBack_exptime(const pcf::IndiProperty &ipRecv)
Callback to process a NEW exposure time request.
int setReadoutSpeed(const mx::meta::trueFalseT< true > &t)
Interface to setReadoutSpeed when the derivedT has readout speed control.
int m_default_bin_y
Power-on ROI y binning.
bool m_tempControlStatusSet
Desired state of temperature control.
float m_minFPS
The minimum FPS, used for INDI attributes.
int onPowerOff()
Actions on power off.
std::string m_readoutSpeedNameSet
The user requested readout speed name, to be set by derived()
~stdCamera() noexcept
Destructor.
@ OPERATING
The device is operating, other than homing.
@ READY
The device is ready for operation, but is not operating.
@ NOTCONNECTED
The application is not connected to the device or service.
@ POWERON
The device power is on.
#define INDI_IDLE
Definition indiUtils.hpp:27
#define INDI_BUSY
Definition indiUtils.hpp:29
#define INDI_OK
Definition indiUtils.hpp:28
std::unordered_map< std::string, cameraConfig > cameraConfigMap
Definition stdCamera.hpp:48
int loadCameraConfig(cameraConfigMap &ccmap, mx::app::appConfigurator &config)
Load the camera configurations contained in the app configuration into a map.
Definition stdCamera.cpp:18
std::string m_serialCommand
The command to send to the camera to place it in this mode.
Definition stdCamera.hpp:34
std::string m_configFile
The file to use for this mode, e.g. an EDT configuration file.
Definition stdCamera.hpp:33
A camera configuration.
Definition stdCamera.hpp:32
void updateIfChanged(pcf::IndiProperty &p, const std::string &el, const T &newVal, indiDriverT *indiDriver, pcf::IndiProperty::PropertyStateType newState=pcf::IndiProperty::Ok)
Update the value of the INDI element, but only if it has changed.
Definition indiUtils.hpp:92
void updateSelectionSwitchIfChanged(pcf::IndiProperty &p, const std::string &el, indiDriverT *indiDriver, pcf::IndiProperty::PropertyStateType newState=pcf::IndiProperty::Ok)
Update the values of a one-of-many INDI switch vector, but only if it has changed.
void updateSwitchIfChanged(pcf::IndiProperty &p, const std::string &el, const pcf::IndiElement::SwitchStateType &newVal, indiDriverT *indiDriver, pcf::IndiProperty::PropertyStateType newState=pcf::IndiProperty::Ok)
Update the value of the INDI element, but only if it has changed.
const pcf::IndiProperty & ipRecv
std::unique_lock< std::mutex > lock(m_indiMutex)
Definition dm.hpp:24
static constexpr logPrioT LOG_CRITICAL
The process can not continue and will shut down (fatal)
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.
#define CAMCTRL_E_NOCONFIGS
Definition stdCamera.hpp:26
Software ERR log entry.