API
 
Loading...
Searching...
No Matches
baslerCtrl.hpp
Go to the documentation of this file.
1/** \file baslerCtrl.hpp
2 * \brief The MagAO-X basler camera controller.
3 *
4 * \author Jared R. Males (jaredmales@gmail.com)
5 *
6 * \ingroup baslerCtrl_files
7 */
8
9#ifndef baslerCtrl_hpp
10#define baslerCtrl_hpp
11
12#include <pylon/PylonIncludes.h>
13#include <pylon/PixelData.h>
14#include <pylon/GrabResultData.h>
15#include <pylon/usb/BaslerUsbInstantCamera.h>
16#include <pylon/usb/_BaslerUsbCameraParams.h>
17#include <GenApi/IFloat.h>
18
19typedef Pylon::CBaslerUsbInstantCamera Camera_t;
20typedef int16_t pixelT;
21
22using namespace Basler_UsbCameraParams;
23
24using namespace Pylon;
25
26// #include <ImageStruct.h>
27#include <ImageStreamIO/ImageStreamIO.h>
28
29#include "../../libMagAOX/libMagAOX.hpp" //Note this is included on command line to trigger pch
30#include "../../magaox_git_version.h"
31
32namespace MagAOX
33{
34namespace app
35{
36
37/** \defgroup baslerCtrl Basler USB3 Camera
38 * \brief Control of a Basler USB3 Camera
39 *
40 * <a href="../handbook/operating/software/apps/baslerCtrl.html">Application Documentation</a>
41 *
42 * \ingroup apps
43 *
44 */
45
46/** \defgroup baslerCtrl_files Basler USB3 Camera Files
47 * \ingroup baslerCtrl
48 */
49
50/** MagAO-X application to control a Basler USB3 Camera
51 *
52 * \ingroup baslerCtrl
53 *
54 */
55class baslerCtrl : public MagAOXApp<>,
56 public dev::stdCamera<baslerCtrl>,
57 public dev::frameGrabber<baslerCtrl>,
58 public dev::telemeter<baslerCtrl>
59{
60
61 friend class dev::stdCamera<baslerCtrl>;
62 friend class dev::frameGrabber<baslerCtrl>;
63 friend class dev::telemeter<baslerCtrl>;
64
65 public:
66 /** \name app::dev Configurations
67 *@{
68 */
69 static constexpr bool c_stdCamera_tempControl =
70 false; ///< app::dev config to tell stdCamera to not expose temperature controls
71
72 static constexpr bool c_stdCamera_temp = true; ///< app::dev config to tell stdCamera to expose temperature
73
74 static constexpr bool c_stdCamera_readoutSpeed =
75 false; ///< app::dev config to tell stdCamera not to expose readout speed controls
76
77 static constexpr bool c_stdCamera_vShiftSpeed =
78 false; ///< app:dev config to tell stdCamera not to expose vertical shift speed control
79
80 static constexpr bool c_stdCamera_emGain =
81 false; ///< app::dev config to tell stdCamera to not expose EM gain controls
82
83 static constexpr bool c_stdCamera_exptimeCtrl =
84 true; ///< app::dev config to tell stdCamera to expose exposure time controls
85
86 static constexpr bool c_stdCamera_fpsCtrl = true; ///< app::dev config to tell stdCamera to expose FPS controls
87
88 static constexpr bool c_stdCamera_fps =
89 true; ///< app::dev config to tell stdCamera not to expose FPS status (ignored since fpsCtrl=true)
90
91 static constexpr bool c_stdCamera_synchro =
92 false; ///< app::dev config to tell stdCamera to not expose synchro mode controls
93
94 static constexpr bool c_stdCamera_usesModes =
95 false; ///< app:dev config to tell stdCamera not to expose mode controls
96
97 static constexpr bool c_stdCamera_usesROI = true; ///< app:dev config to tell stdCamera to expose ROI controls
98
99 static constexpr bool c_stdCamera_cropMode =
100 false; ///< app:dev config to tell stdCamera not to expose Crop Mode controls
101
102 static constexpr bool c_stdCamera_hasShutter =
103 false; ///< app:dev config to tell stdCamera to expose shutter controls
104
105 static constexpr bool c_stdCamera_usesStateString =
106 false; ///< app::dev confg to tell stdCamera to expose the state string property
107
108 static constexpr bool c_frameGrabber_flippable =
109 true; ///< app:dev config to tell framegrabber that this camera can be flipped
110
111 ///@}
112
113 protected:
114 /** \name configurable parameters
115 *@{
116 */
117 std::string m_serialNumber; ///< The camera's identifying serial number
118
119 int m_bits{ 10 }; ///< The number of bits used by the camera.
120
121 ///@}
122
123 /** \name binning allowed values
124 * @{
125 */
126 std::vector<int> m_binXs; ///< The allowed values of binning in X (horizontal)
127 std::vector<int> m_binYs; ///< The allowed values of binning in Y (vertical)
128
129 std::vector<int> m_incXs; ///< The allowed increment in X for each X-binning
130
131 std::vector<int> m_minWs; ///< The minimum value of the width for each X-binning
132 std::vector<int> m_incWs; ///< The minimum value of the width for each X-binning
133 std::vector<int> m_maxWs; ///< The minimum value of the width for each X-binning
134
135 std::vector<int> m_incYs; ///< The allowed increment in Y for each Y-binning
136
137 std::vector<int> m_minHs; ///< The minimum value of the height for each Y-binning
138 std::vector<int> m_incHs; ///< The minimum value of the height for each Y-binning
139 std::vector<int> m_maxHs; ///< The minimum value of the height for each Y-binning
140
141 ///@}
142
143 CBaslerUsbInstantCamera *m_camera{ nullptr }; ///< The library camera handle
144 CGrabResultPtr ptrGrabResult; ///< The result of an attempt to grab an image
145
146 public:
147 /// Default c'tor
148 baslerCtrl();
149
150 /// Destructor
152
153 /// Setup the configuration system (called by MagAOXApp::setup())
154 virtual void setupConfig();
155
156 /// load the configuration system results (called by MagAOXApp::setup())
157 virtual void loadConfig();
158
159 /// Startup functions
160 /** Sets up the INDI vars.
161 *
162 */
163 virtual int appStartup();
164
165 /// Implementation of the FSM for the Siglent SDG
166 virtual int appLogic();
167
168 /// Do any needed shutdown tasks. Currently nothing in this app.
169 virtual int appShutdown();
170
171 int connect();
172
174 int startAcquisition();
176 int loadImageIntoStream( void *dest );
177 int reconfig();
178
179 protected:
180 /// Get the current detector temperature
181 /**
182 * \returns 0 on success
183 * \returns -1 on an error.
184 */
185 int getTemp();
186
187 /// Get the current exposure time
188 /**
189 * \returns 0 on success
190 * \returns -1 on an error.
191 */
192 int getExpTime();
193
194 /// Get the current framerate
195 /**
196 * \returns 0 on success
197 * \returns -1 on an error.
198 */
199 int getFPS();
200
201 float fps();
202
203 /** \name stdCamera Interface
204 *
205 * @{
206 */
207
208 /// Set defaults for a power on state.
209 /**
210 * \returns 0 on success
211 * \returns -1 on error
212 */
213 int powerOnDefaults();
214
215 /// Set the framerate.
216 /** This uses the acquistion framerate feature. If m_fpsSet is 0, acuisition framerate is disabled
217 * and the resultant framerate is based solely on exposure time and ROI. If non-zero, then the
218 * framerate will be set to m_fpsSet and the camera will maintain this (as long as exposure time
219 * and ROI allow).
220 *
221 * \returns 0 always
222 */
223 int setFPS();
224
225 /// Set the Exposure Time. [stdCamera interface]
226 /** Sets the frame rate to m_expTimeSet.
227 *
228 * \returns 0 on success
229 * \returns -1 on error
230 */
231 int setExpTime();
232
233 /// Check the next ROI
234 /** Checks if the target values are valid and adjusts them to the closest valid values if needed.
235 *
236 * \returns 0 always
237 */
238 int checkNextROI();
239
240 /// Set the next ROI
241 /**
242 * \returns 0 always
243 */
244 int setNextROI();
245
246 ///@}
247
248 /** \name Telemeter Interface
249 *
250 * @{
251 */
252
253 int checkRecordTimes();
254
256
257 ///@}
258};
259
266
268{
269 return;
270}
271
273{
275
277
278 config.add( "camera.serialNumber",
279 "",
280 "camera.serialNumber",
281 argType::Required,
282 "camera",
283 "serialNumber",
284 false,
285 "int",
286 "The identifying serial number of the camera." );
287 config.add( "camera.bits",
288 "",
289 "camera.bits",
290 argType::Required,
291 "camera",
292 "bits",
293 false,
294 "int",
295 "The number of bits used by the camera. Default is 10." );
296
298}
299
301{
303
304 config( m_serialNumber, "camera.serialNumber" );
305 config( m_bits, "camera.bits" );
306
308
310}
311
313{
314
315 //=================================
316 // Do camera configuration here
317
318 PylonInitialize(); // Initializes pylon runtime before using any pylon methods
319
321 {
322 return log<software_critical, -1>( { __FILE__, __LINE__ } );
323 }
324
326 {
327 return log<software_critical, -1>( { __FILE__, __LINE__ } );
328 }
329
331 {
332 return log<software_error, -1>( { __FILE__, __LINE__ } );
333 }
334
336
337 return 0;
338}
339
341{
342 // and run stdCamera's appLogic
344 {
345 return log<software_error, -1>( { __FILE__, __LINE__ } );
346 }
347
348 // and run frameGrabber's appLogic to see if the f.g. thread has exited.
350 {
351 return log<software_error, -1>( { __FILE__, __LINE__ } );
352 }
353
355 {
356 std::unique_lock<std::mutex> lock( m_indiMutex );
357 if( connect() < 0 )
358 {
360 }
361
363 return 0;
364 }
365
367 {
368 // Get a lock
369 std::unique_lock<std::mutex> lock( m_indiMutex );
370
372 }
373
375 {
376 // Get a lock if we can
377 std::unique_lock<std::mutex> lock( m_indiMutex, std::try_to_lock );
378
379 // but don't wait for it, just go back around.
380 if( !lock.owns_lock() )
381 return 0;
382
383 if( getTemp() < 0 )
384 {
387 return 0;
388 }
389
390 if( getExpTime() < 0 )
391 {
394 return 0;
395 }
396
397 if( getFPS() < 0 )
398 {
401 return 0;
402 }
403
405 {
408 return 0;
409 }
410
412 {
415 return 0;
416 }
417
419 {
421 return 0;
422 }
423 }
424
425 ///\todo Fall through check?
426
427 return 0;
428}
429
445
447{
449 // info.SetDeviceClass(Camera_t::DeviceClass());
450 info.SetSerialNumber( m_serialNumber.c_str() );
451
452 try
453 {
454 if( m_camera )
455 {
456 m_camera->Close();
457 delete m_camera;
458 }
459 m_camera = nullptr;
460
461 m_camera = new CBaslerUsbInstantCamera( CTlFactory::GetInstance().CreateFirstDevice( info ) );
462 }
463 catch( ... )
464 {
465 if( m_camera )
466 {
467 m_camera->Close();
468 delete m_camera;
469 }
470 m_camera = nullptr;
471
473 if( !stateLogged() )
474 {
475 log<text_log>( "no camera with serial number " + m_serialNumber + " found." );
476 }
477 return 0;
478 }
479
480 try
481 {
482 if( m_shmimName == "" )
483 {
485 (std::string)m_camera->GetDeviceInfo().GetModelName() + "_" +
486 (std::string)m_camera->GetDeviceInfo().GetSerialNumber(); // Gets m_camera model name and serial number
487 }
488
489 m_camera->RegisterConfiguration(
491
492 m_camera->Open(); // Opens camera parameters to grab images and set exposure time
493 }
494 catch( ... )
495 {
496 if( m_camera )
497 {
498 m_camera->Close();
499 delete m_camera;
500 }
501 m_camera = nullptr;
502
504 if( !stateLogged() )
505 {
506 log<text_log>( "error opening camera " + m_serialNumber + "." );
507 }
508 return -1;
509 }
510
511 try
512 {
513 m_camera->ExposureAuto.SetValue( ExposureAuto_Off );
514 }
515 catch( ... )
516 {
517 if( m_camera )
518 {
519 m_camera->Close();
520 delete m_camera;
521 }
522 m_camera = nullptr;
523
525 if( !stateLogged() )
526 {
527 log<text_log>( "failed to set exposure auto off for camera " + m_serialNumber );
528 }
529 return -1;
530 }
531
532 try
533 {
534 if( m_bits == 8 )
535 {
536 m_camera->PixelFormat.SetValue( PixelFormat_Mono8 );
537 }
538 else if( m_bits == 10 )
539 {
540 m_camera->PixelFormat.SetValue( PixelFormat_Mono10 ); // Set to 10 bits
541 }
542 else if( m_bits == 12 )
543 {
544 m_camera->PixelFormat.SetValue( PixelFormat_Mono12 );
545 }
546 else
547 {
548 log<text_log>( "unsupported bit depth for camera" + m_serialNumber + "" );
549 }
550 }
551 catch( ... )
552 {
553 if( m_camera )
554 {
555 m_camera->Close();
556 delete m_camera;
557 }
558 m_camera = nullptr;
559
561 if( !stateLogged() )
562 {
563 log<text_log>( "failed to set bit depth for camera" + m_serialNumber + "" );
564 }
565 return -1;
566 }
567
569 if( !stateLogged() )
570 {
571 log<text_log>( "Found camera of type " + (std::string)m_camera->GetDeviceInfo().GetModelName() +
572 " with serial number " + m_serialNumber + "." );
573 log<text_log>( "Using shared memory name " + m_shmimName + "." );
574 }
575
576 m_camera->BinningHorizontalMode.SetValue( BinningHorizontalMode_Sum );
577 m_camera->BinningVerticalMode.SetValue( BinningVerticalMode_Sum );
578
579 // -- Here we interrogate the camera to find valid ROI settings -- //
580
581 // Stop the camera and cycle through settings to get limits for each binning
582 m_camera->StopGrabbing();
583 m_camera->OffsetX.SetValue( 0 ); // ensure that all values are valid
584 m_camera->OffsetY.SetValue( 0 );
585
586 int minb = m_camera->BinningHorizontal.GetMin();
587 int incb = m_camera->BinningHorizontal.GetInc();
588 int maxb = m_camera->BinningHorizontal.GetMax();
589
590 m_binXs.clear();
591 for( int b = minb; b <= maxb; b += incb )
592 m_binXs.push_back( b );
593
594 minb = m_camera->BinningVertical.GetMin();
595 incb = m_camera->BinningVertical.GetInc();
596 maxb = m_camera->BinningVertical.GetMax();
597
598 m_binYs.clear();
599 for( int b = minb; b <= maxb; b += incb )
600 m_binYs.push_back( b );
601
602 m_incXs.clear();
603 m_minWs.clear();
604 m_incWs.clear();
605 m_maxWs.clear();
606 for( size_t b = 0; b < m_binXs.size(); ++b )
607 {
608 m_camera->BinningHorizontal.SetValue( m_binXs[b] );
609 m_camera->BinningVertical.SetValue( m_binYs[0] );
610
611 m_incXs.push_back( m_camera->OffsetX.GetInc() );
612 m_minWs.push_back( m_camera->Width.GetMin() );
613 m_incWs.push_back( m_camera->Width.GetInc() );
614 m_maxWs.push_back( m_camera->Width.GetMax() );
615
616 /*//Leave for troubleshooting:
617 std::cerr << "--------------------\nH-binning: " << m_binXs[b] << "\n";
618 std::cerr << "OffsetX: " << 1 << " " << m_camera->OffsetX.GetInc() << " " << m_camera->Width.GetMax() -
619 m_camera->Width.GetMin() << "\n"; std::cerr << "Width: " << m_camera->Width.GetMin() << " " <<
620 m_camera->Width.GetInc() << " " << m_camera->Width.GetMax() << "\n"; std::cerr << "OffsetY: " << 1 << " " <<
621 m_camera->OffsetY.GetInc() << " " << m_camera->Height.GetMax() - m_camera->Height.GetMin() << "\n"; std::cerr <<
622 "Height: " << m_camera->Height.GetMin() << " " << m_camera->Height.GetInc() << " " << m_camera->Height.GetMax()
623 << "\n";
624 */
625 }
626
627 m_incYs.clear();
628 m_minHs.clear();
629 m_incHs.clear();
630 m_maxHs.clear();
631 for( size_t b = 0; b < m_binYs.size(); ++b )
632 {
633 m_camera->BinningHorizontal.SetValue( m_binXs[0] );
634 m_camera->BinningVertical.SetValue( m_binYs[b] );
635
636 m_incYs.push_back( m_camera->OffsetX.GetInc() );
637 m_minHs.push_back( m_camera->Height.GetMin() );
638 m_incHs.push_back( m_camera->Height.GetInc() );
639 m_maxHs.push_back( m_camera->Height.GetMax() );
640
641 /*//Leave for troubleshooting:
642 std::cerr << "--------------------\nV-binning: " << m_binYs[b] << "\n";
643 std::cerr << "OffsetX: " << 1 << " " << m_camera->OffsetX.GetInc() << " " << m_camera->Width.GetMax() -
644 m_camera->Width.GetMin() << "\n"; std::cerr << "Width: " << m_camera->Width.GetMin() << " " <<
645 m_camera->Width.GetInc() << " " << m_camera->Width.GetMax() << "\n"; std::cerr << "OffsetY: " << 1 << " " <<
646 m_camera->OffsetY.GetInc() << " " << m_camera->Height.GetMax() - m_camera->Height.GetMin() << "\n"; std::cerr <<
647 "Height: " << m_camera->Height.GetMin() << " " << m_camera->Height.GetInc() << " " << m_camera->Height.GetMax()
648 << "\n";
649 */
650 }
651
652 m_full_w = m_camera->SensorWidth.GetValue();
653 m_full_h = m_camera->SensorHeight.GetValue();
654 m_full_x = 0.5 * ( (float)m_full_w - 1.0 );
655 m_full_y = 0.5 * ( (float)m_full_h - 1.0 );
656
657 if( m_default_w == 0 )
659 if( m_default_h == 0 )
661 if( m_default_x == 0 )
663 if( m_default_y == 0 )
665 if( m_default_bin_x == 0 )
667 if( m_default_bin_y == 0 )
669
676
677 return 0;
678}
679
681{
682 if( !m_camera )
683 return -1;
684
685 try
686 {
687 recordCamera( true );
688 m_camera->StopGrabbing();
689 /*
690 The CenterX/Y has to be set to false otherwise the software tries to auto-center the frames.
691 See: https://docs.baslerweb.com/image-roi
692 */
693 m_camera->CenterX.SetValue( false );
694 m_camera->CenterY.SetValue( false );
695
696 // set offsets to 0 so any valid w/h will work.
697 m_camera->OffsetX.SetValue( 0 );
698 m_camera->OffsetY.SetValue( 0 );
699
700 if( checkNextROI() < 0 )
701 {
702 log<software_error>( { __FILE__, __LINE__, "error from checkNextROI()" } );
703 return -1;
704 }
705
706 // Note: assuming checkNextROI has adjusted m_nextROI to valid values, so not doing any checks
707 // First find binning indices
708 size_t bx = 0;
709 for( size_t b = 0; b < m_binXs.size(); ++b )
710 {
711 if( m_nextROI.bin_x == m_binXs[b] )
712 {
713 bx = b;
714 break;
715 }
716 }
717
718 size_t by = 0;
719 for( size_t b = 0; b < m_binYs.size(); ++b )
720 {
721 if( m_nextROI.bin_y == m_binYs[b] )
722 {
723 by = b;
724 break;
725 }
726 }
727
728 // Set ROI.
729 int xoff;
730 int yoff;
732 {
733 xoff = ( m_maxWs[bx] - 1 - m_nextROI.x ) - 0.5 * ( (float)m_nextROI.w - 1 );
734 }
735 else
736 {
737 xoff = m_nextROI.x - 0.5 * ( (float)m_nextROI.w - 1 );
738 }
739
741 {
742 yoff = ( m_maxHs[by] - 1 - m_nextROI.y ) - 0.5 * ( (float)m_nextROI.h - 1 );
743 }
744 else
745 {
746 yoff = m_nextROI.y - 0.5 * ( (float)m_nextROI.h - 1 );
747 }
748
749 m_camera->BinningHorizontal.SetValue( m_nextROI.bin_x );
750 m_camera->BinningVertical.SetValue( m_nextROI.bin_y );
751 // Probably not necessary to do it every time, but just in case:
752 m_camera->BinningHorizontalMode.SetValue( BinningHorizontalMode_Sum );
753 m_camera->BinningVerticalMode.SetValue( BinningVerticalMode_Sum );
754
755 m_camera->Width.SetValue( m_nextROI.w );
756 m_camera->Height.SetValue( m_nextROI.h );
757
758 m_camera->OffsetX.SetValue( xoff );
759 m_camera->OffsetY.SetValue( yoff );
760
761 // Read the parameter from the camera to check if parameter change is successful
762 m_currentROI.bin_x = m_camera->BinningHorizontal.GetValue();
763 m_currentROI.bin_y = m_camera->BinningVertical.GetValue();
764
765 bx = 0;
766 for( size_t b = 0; b < m_binXs.size(); ++b )
767 {
768 if( m_nextROI.bin_x == m_binXs[b] )
769 {
770 bx = b;
771 break;
772 }
773 }
774
775 by = 0;
776 for( size_t b = 0; b < m_binYs.size(); ++b )
777 {
778 if( m_nextROI.bin_y == m_binYs[b] )
779 {
780 by = b;
781 break;
782 }
783 }
784
785 m_currentROI.w = m_camera->Width.GetValue();
786 m_currentROI.h = m_camera->Height.GetValue();
787
789 {
790 m_currentROI.x = m_maxWs[bx] - 1 - ( m_camera->OffsetX.GetValue() + 0.5 * ( (float)m_currentROI.w - 1 ) );
791 }
792 else
793 {
794 m_currentROI.x = m_camera->OffsetX.GetValue() + 0.5 * ( (float)m_currentROI.w - 1 );
795 }
796
798 {
799 m_currentROI.y = m_maxHs[by] - 1 - ( m_camera->OffsetY.GetValue() + 0.5 * ( (float)m_currentROI.h - 1 ) );
800 }
801 else
802 {
803 m_currentROI.y = m_camera->OffsetY.GetValue() + 0.5 * ( (float)m_currentROI.h - 1 );
804 }
805
806 // Set the full window for this binning
808 m_full_currbin_x = 0.5 * ( (float)m_full_currbin_w - 1.0 );
810 m_full_currbin_y = 0.5 * ( (float)m_full_currbin_h - 1.0 );
811
812 // Update binning
819
820 // We also update target to the settable values
825 m_nextROI.bin_x = m_currentROI.bin_x;
826 m_nextROI.bin_y = m_currentROI.bin_y;
827
834
838
839 getFPS();
840
841 recordCamera( true );
842 }
843 catch( ... )
844 {
845 log<software_error>( { __FILE__, __LINE__, "invalid ROI specifications" } );
847 return -1;
848 }
849
850 return 0;
851}
852
854{
855 try
856 {
857 m_camera->StartGrabbing( GrabStrategy_LatestImageOnly ); // Start grabbing, and always grab just the last image.
858 }
859 catch( ... )
860 {
862 return -1;
863 }
864
866
867 return 0;
868}
869
871{
872 try
873 {
874 m_camera->RetrieveResult( 1000, ptrGrabResult, TimeoutHandling_ThrowException );
875 }
876 catch( ... )
877 {
879 return -1;
880 }
881
882 if( ptrGrabResult->GrabSucceeded() ) // If image is grabbed successfully
883 {
885 return 0;
886 }
887 else
888 {
890 return -1;
891 }
892}
893
895{
896 pixelT *src = nullptr;
897 try
898 {
899 src = reinterpret_cast<pixelT *>( ptrGrabResult->GetBuffer() );
900 }
901 catch( ... )
902 {
904 return -1;
905 }
906
907 if( src == nullptr )
908 return -1;
909
911 return -1;
912
913 return 0;
914}
915
917{
918
919 return 0;
920}
921
923{
924 if( m_camera == nullptr )
925 return 0;
926
927 try
928 {
929 m_ccdTemp = (float)m_camera->DeviceTemperature.GetValue();
930 recordCamera();
931 }
932 catch( ... )
933 {
934 m_ccdTemp = -999;
935 recordCamera();
937 return -1;
938 }
939
940 return 0;
941}
942
944{
945 if( m_camera == nullptr )
946 return 0;
947
948 try
949 {
950 m_expTime = (float)m_camera->ExposureTime.GetValue() / 1e6;
951 recordCamera();
952 }
953 catch( ... )
954 {
955 m_expTime = -999;
956 recordCamera();
958 return -1;
959 }
960
961 return 0;
962}
963
965{
966 if( m_camera == nullptr )
967 return 0;
968
969 try
970 {
971 m_fps = m_camera->ResultingFrameRate.GetValue();
972 recordCamera();
973 }
974 catch( ... )
975 {
976 m_fps = -999;
977 recordCamera();
979 return -1;
980 }
981
982 return 0;
983}
984
985inline float baslerCtrl::fps()
986{
987 return m_fps;
988}
989
991{
998
999 return 0;
1000}
1001
1003{
1004 if( m_camera == nullptr )
1005 return 0;
1006
1007 recordCamera( true );
1008
1009 if( m_fpsSet == 0 )
1010 {
1011 try
1012 {
1013 m_camera->AcquisitionFrameRateEnable.SetValue( false );
1014 }
1015 catch( ... )
1016 {
1017 return log<software_error, -1>( { __FILE__, __LINE__, "Error disabling frame rate limit." } );
1018 }
1019 }
1020 else
1021 {
1022 try
1023 {
1024 m_camera->AcquisitionFrameRateEnable.SetValue( true );
1025 m_camera->AcquisitionFrameRate.SetValue( m_fpsSet );
1026 }
1027 catch( ... )
1028 {
1029 return log<software_error, -1>( { __FILE__, __LINE__, "Error setting frame rate limit." } );
1030 }
1031 }
1032
1033 return 0;
1034}
1035
1037{
1038 if( m_camera == nullptr )
1039 return 0;
1040
1041 try
1042 {
1043 recordCamera( true );
1044 m_camera->ExposureTime.SetValue( m_expTimeSet * 1e6 );
1045 }
1046 catch( ... )
1047 {
1048 log<software_error>( { __FILE__, __LINE__, "Error setting exposure time" } );
1049 return -1;
1050 }
1051
1052 log<text_log>( "Set exposure time: " + std::to_string( m_expTimeSet ) + " sec" );
1053
1054 return 0;
1055}
1056
1058{
1059 std::cerr << "checkNextROI!\n";
1060
1061 // First find binning indices
1062 size_t bx = 0;
1063 for( size_t b = 0; b < m_binXs.size(); ++b )
1064 {
1065 if( m_nextROI.bin_x == m_binXs[b] )
1066 {
1067 bx = b;
1068 break;
1069 }
1070 }
1071 std::cerr << "req bin_x: " << m_nextROI.bin_x << " " << "adj bin_x: " << m_binXs[bx] << "\n";
1072 m_nextROI.bin_x = m_binXs[bx]; // In case no valid value was found.
1073
1074 size_t by = 0;
1075 for( size_t b = 0; b < m_binYs.size(); ++b )
1076 {
1077 if( m_nextROI.bin_y == m_binYs[b] )
1078 {
1079 by = b;
1080 break;
1081 }
1082 }
1083 std::cerr << "req bin_y: " << m_nextROI.bin_y << " " << "adj bin_y: " << m_binYs[by] << "\n";
1084 m_nextROI.bin_y = m_binYs[by]; // In case no valid value was found.
1085
1086 // Next check width
1087 //-- round to nearest increment
1088 //-- check limits
1089 int w = m_nextROI.w;
1090 int rw = w % m_incWs[bx];
1091 if( rw < 0.5 * m_incWs[bx] )
1092 w -= rw;
1093 else
1094 w += m_incWs[bx] - rw;
1095
1096 if( w < m_minWs[bx] )
1097 w = m_minWs[bx];
1098 else if( w > m_maxWs[bx] )
1099 w = m_maxWs[bx];
1100
1101 std::cerr << "req w: " << m_nextROI.w << " " << "adj w: " << w << "\n";
1102 m_nextROI.w = w;
1103
1104 // Now check x
1105 //-- calculate offset from center
1106 //-- round to nearest increment
1107 //-- recalculate center
1108 int x;
1110 {
1111 x = ( m_maxWs[bx] - 1 - m_nextROI.x ) - 0.5 * ( (float)w - 1 );
1112 }
1113 else
1114 {
1115 x = m_nextROI.x - 0.5 * ( (float)w - 1 );
1116 }
1117
1118 int rx = x % m_incXs[bx];
1119 if( rx < 0.5 * m_incXs[bx] )
1120 x -= rx;
1121 else
1122 x += m_incXs[bx] - rx;
1123
1124 if( x < 0 )
1125 x = 0;
1126 else if( x > m_maxWs[bx] - w )
1127 x = m_maxWs[bx] - w;
1128
1129 std::cerr << "req x: " << m_nextROI.x;
1131 {
1132 m_nextROI.x = m_maxWs[bx] - 1 - ( x + 0.5 * ( (float)w - 1.0 ) );
1133 }
1134 else
1135 {
1136 m_nextROI.x = x + 0.5 * ( (float)w - 1.0 );
1137 }
1138 std::cerr << " adj x: " << m_nextROI.x << "\n";
1139
1140 // Next check height
1141 //-- round to nearest increment
1142 //-- check limits
1143 int h = m_nextROI.h;
1144 int rh = h % m_incHs[by];
1145 if( rh < 0.5 * m_incHs[by] )
1146 h -= rh;
1147 else
1148 h += m_incHs[by] - rh;
1149
1150 if( h < m_minHs[by] )
1151 h = m_minHs[by];
1152 else if( h > m_maxHs[by] )
1153 h = m_maxHs[by];
1154
1155 std::cerr << "req h: " << m_nextROI.h << " " << "adj h: " << h << "\n";
1156 m_nextROI.h = h;
1157
1158 // Now check y
1159 //-- calculate offset from center
1160 //-- round to nearest increment
1161 //-- recalculate center
1162 int y;
1164 {
1165 y = ( m_maxHs[by] - 1 - m_nextROI.y ) - 0.5 * ( (float)h - 1 );
1166 }
1167 else
1168 {
1169 y = m_nextROI.y - 0.5 * ( (float)h - 1 );
1170 }
1171
1172 int ry = y % m_incYs[by];
1173 if( ry < 0.5 * m_incYs[by] )
1174 y -= ry;
1175 else
1176 y += m_incYs[by] - ry;
1177
1178 if( y < 0 )
1179 y = 0;
1180 else if( y > m_maxHs[by] - h )
1181 y = m_maxHs[by] - h;
1182
1183 std::cerr << "req y: " << m_nextROI.y;
1185 {
1186 m_nextROI.y = m_maxHs[by] - 1 - ( y + 0.5 * ( (float)h - 1 ) );
1187 }
1188 else
1189 {
1190 m_nextROI.y = y + 0.5 * ( (float)h - 1 );
1191 }
1192 std::cerr << " adj y: " << m_nextROI.y << "\n";
1193
1194 try
1195 {
1202
1203 return 0;
1204 }
1205 catch( const std::exception &e )
1206 {
1207 return log<software_error, -1>( { __FILE__, __LINE__, std::string( "Exception caught: " ) + e.what() } );
1208 }
1209}
1210
1212{
1213 std::cerr << "setNextROI:\n";
1214 std::cerr << " m_nextROI.x = " << m_nextROI.x << "\n";
1215 std::cerr << " m_nextROI.y = " << m_nextROI.y << "\n";
1216 std::cerr << " m_nextROI.w = " << m_nextROI.w << "\n";
1217 std::cerr << " m_nextROI.h = " << m_nextROI.h << "\n";
1218 std::cerr << " m_nextROI.bin_x = " << m_nextROI.bin_x << "\n";
1219 std::cerr << " m_nextROI.bin_y = " << m_nextROI.bin_y << "\n";
1220
1221 m_reconfig = true;
1222
1223 updateSwitchIfChanged( m_indiP_roi_set, "request", pcf::IndiElement::Off, INDI_IDLE );
1224 updateSwitchIfChanged( m_indiP_roi_full, "request", pcf::IndiElement::Off, INDI_IDLE );
1225 updateSwitchIfChanged( m_indiP_roi_last, "request", pcf::IndiElement::Off, INDI_IDLE );
1226 updateSwitchIfChanged( m_indiP_roi_default, "request", pcf::IndiElement::Off, INDI_IDLE );
1227 return 0;
1228}
1229
1234
1236{
1237 return recordCamera( true );
1238}
1239
1240} // namespace app
1241} // namespace MagAOX
1242#endif
Pylon::CBaslerUsbInstantCamera Camera_t
int16_t pixelT
The base-class for MagAO-X applications.
Definition MagAOXApp.hpp:73
void updateIfChanged(pcf::IndiProperty &p, const std::string &el, const T &newVal, pcf::IndiProperty::PropertyStateType ipState=pcf::IndiProperty::Ok)
Update an INDI property element value if it has changed.
stateCodes::stateCodeT state()
Get the current state code.
void updateSwitchIfChanged(pcf::IndiProperty &p, const std::string &el, const pcf::IndiElement::SwitchStateType &newVal, pcf::IndiProperty::PropertyStateType ipState=pcf::IndiProperty::Ok)
Update an INDI switch element value if it has changed.
int stateLogged()
Updates and returns the value of m_stateLogged. Will be 0 on first call after a state change,...
static int log(const typename logT::messageT &msg, logPrioT level=logPrio::LOG_DEFAULT)
Make a log entry.
std::mutex m_indiMutex
Mutex for locking INDI communications.
int setFPS()
Set the framerate.
~baslerCtrl() noexcept
Destructor.
virtual int appStartup()
Startup functions.
static constexpr bool c_stdCamera_exptimeCtrl
app::dev config to tell stdCamera to expose exposure time controls
virtual int appShutdown()
Do any needed shutdown tasks. Currently nothing in this app.
baslerCtrl()
Default c'tor.
static constexpr bool c_stdCamera_cropMode
app:dev config to tell stdCamera not to expose Crop Mode controls
static constexpr bool c_stdCamera_emGain
app::dev config to tell stdCamera to not expose EM gain controls
static constexpr bool c_stdCamera_fpsCtrl
app::dev config to tell stdCamera to expose FPS controls
int m_bits
The number of bits used by the camera.
static constexpr bool c_stdCamera_readoutSpeed
app::dev config to tell stdCamera not to expose readout speed controls
std::vector< int > m_minWs
The minimum value of the width for each X-binning.
static constexpr bool c_stdCamera_fps
app::dev config to tell stdCamera not to expose FPS status (ignored since fpsCtrl=true)
static constexpr bool c_stdCamera_hasShutter
app:dev config to tell stdCamera to expose shutter controls
int setExpTime()
Set the Exposure Time. [stdCamera interface].
static constexpr bool c_stdCamera_usesStateString
app::dev confg to tell stdCamera to expose the state string property
static constexpr bool c_stdCamera_vShiftSpeed
app:dev config to tell stdCamera not to expose vertical shift speed control
std::vector< int > m_maxWs
The minimum value of the width for each X-binning.
int getTemp()
Get the current detector temperature.
static constexpr bool c_stdCamera_synchro
app::dev config to tell stdCamera to not expose synchro mode controls
static constexpr bool c_stdCamera_temp
app::dev config to tell stdCamera to expose temperature
int recordTelem(const telem_stdcam *)
static constexpr bool c_stdCamera_usesModes
app:dev config to tell stdCamera not to expose mode controls
static constexpr bool c_stdCamera_usesROI
app:dev config to tell stdCamera to expose ROI controls
int checkNextROI()
Check the next ROI.
CBaslerUsbInstantCamera * m_camera
The library camera handle.
CGrabResultPtr ptrGrabResult
The result of an attempt to grab an image.
std::vector< int > m_maxHs
The minimum value of the height for each Y-binning.
std::vector< int > m_incYs
The allowed increment in Y for each Y-binning.
std::vector< int > m_binXs
The allowed values of binning in X (horizontal)
int setNextROI()
Set the next ROI.
static constexpr bool c_frameGrabber_flippable
app:dev config to tell framegrabber that this camera can be flipped
static constexpr bool c_stdCamera_tempControl
app::dev config to tell stdCamera to not expose temperature controls
int getExpTime()
Get the current exposure time.
std::vector< int > m_incHs
The minimum value of the height for each Y-binning.
virtual void loadConfig()
load the configuration system results (called by MagAOXApp::setup())
int getFPS()
Get the current framerate.
virtual int appLogic()
Implementation of the FSM for the Siglent SDG.
int loadImageIntoStream(void *dest)
std::vector< int > m_incWs
The minimum value of the width for each X-binning.
std::string m_serialNumber
The camera's identifying serial number.
std::vector< int > m_incXs
The allowed increment in X for each X-binning.
int powerOnDefaults()
Set defaults for a power on state.
std::vector< int > m_binYs
The allowed values of binning in Y (vertical)
virtual void setupConfig()
Setup the configuration system (called by MagAOXApp::setup())
std::vector< int > m_minHs
The minimum value of the height for each Y-binning.
timespec m_currImageTimestamp
The timestamp of the current image.
uint32_t m_width
The width of the image, once deinterlaced etc.
int appShutdown()
Shuts down the framegrabber thread.
int loadConfig(mx::app::appConfigurator &config)
load the configuration system results
std::string m_shmimName
The name of the shared memory image, is used in /tmp/<shmimName>.im.shm. Derived classes should set a...
uint8_t m_dataType
The ImageStreamIO type code.
bool m_reconfig
Flag to set if a camera reconfiguration requires a framegrabber reset.
uint32_t m_height
The height of the image, once deinterlaced etc.
int setupConfig(mx::app::appConfigurator &config)
Setup the configuration system.
MagAO-X standard camera interface.
float m_fpsSet
The commanded fps, as set by user.
pcf::IndiProperty m_indiP_roi_y
Property used to set the ROI x center coordinate.
int m_full_currbin_w
The current-binning full ROI width.
float m_default_x
Power-on ROI center x coordinate.
float m_full_currbin_x
The current-binning full ROI center x coordinate.
float m_expTime
The current exposure time, in seconds.
int m_default_bin_x
Power-on ROI x binning.
pcf::IndiProperty m_indiP_roi_last
Property used to trigger setting the last ROI.
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.
float m_full_y
The full ROI center y coordinate.
int setupConfig(mx::app::appConfigurator &config)
Setup the configuration system.
pcf::IndiProperty m_indiP_roi_w
Property used to set the ROI width.
pcf::IndiProperty m_indiP_roi_default
Property used to trigger setting the default and startup ROI.
pcf::IndiProperty m_indiP_roi_bin_x
Property used to set the ROI x binning.
float m_full_x
The full ROI center x coordinate.
int m_full_currbin_h
The current-binning full ROI height.
int loadConfig(mx::app::appConfigurator &config)
load the configuration system results
float m_ccdTemp
The current temperature, in C.
float m_full_currbin_y
The current-binning full ROI center y coordinate.
pcf::IndiProperty m_indiP_roi_full
Property used to trigger setting the full ROI.
pcf::IndiProperty m_indiP_roi_set
Property used to trigger setting the ROI.
int appShutdown()
Application shutdown.
float m_default_y
Power-on ROI center y coordinate.
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 m_default_bin_y
Power-on ROI y binning.
@ OPERATING
The device is operating, other than homing.
@ NODEVICE
No device exists for the application to control.
@ ERROR
The application has encountered an error, from which it is recovering (with or without intervention)
@ READY
The device is ready for operation, but is not operating.
@ CONNECTED
The application has connected to the device or service.
@ NOTCONNECTED
The application is not connected to the device or service.
@ info
For information only.
#define INDI_IDLE
Definition indiUtils.hpp:27
#define INDI_OK
Definition indiUtils.hpp:28
std::unique_lock< std::mutex > lock(m_indiMutex)
Definition dm.hpp:24
A device base class which saves telemetry.
Definition telemeter.hpp:69
int appShutdown()
Perform telemeter application shutdown.
int loadConfig(appConfigurator &config)
Load the device section from an application configurator.
int setupConfig(appConfigurator &config)
Setup an application configurator for the device section.
Software CRITICAL log entry.
Software ERR log entry.
Log entry recording stdcam stage specific status.