API
 
Loading...
Searching...
No Matches
modalGainOpt.hpp
Go to the documentation of this file.
1/** \file modalGainOpt.hpp
2 * \brief The MagAO-X PSD-based gain optimizer header file
3 *
4 * \ingroup modalGainOpt_files
5 */
6
7#ifndef modalGainOpt_hpp
8#define modalGainOpt_hpp
9
10#include <mx/ao/analysis/clGainOpt.hpp>
11#include <mx/ao/analysis/clAOLinearPredictor.hpp>
12
13#include "../../libMagAOX/libMagAOX.hpp" //Note this is included on command line to trigger pch
14#include "../../magaox_git_version.h"
15
16/** \defgroup modalGainOpt
17 * \brief The MagAO-X application to perform PSD-based gain optimization
18 *
19 * <a href="../handbook/operating/software/apps/modalGainOpt.html">Application Documentation</a>
20 *
21 * \ingroup apps
22 *
23 */
24
25/** \defgroup modalGainOpt_files
26 * \ingroup modalGainOpt
27 */
28
29namespace MagAOX
30{
31namespace app
32{
33
35{
36 static std::string configSection()
37 {
38 return "psdShmim";
39 };
40
41 static std::string indiPrefix()
42 {
43 return "psd";
44 };
45};
46
48{
49 static std::string configSection()
50 {
51 return "freqShmim";
52 };
53
54 static std::string indiPrefix()
55 {
56 return "freq";
57 };
58};
59
61{
62 static std::string configSection()
63 {
64 return "gainFactShmim";
65 };
66
67 static std::string indiPrefix()
68 {
69 return "gainFact";
70 };
71};
72
74{
75 static std::string configSection()
76 {
77 return "multFactShmim";
78 };
79
80 static std::string indiPrefix()
81 {
82 return "multFact";
83 };
84};
85
87{
88 static std::string configSection()
89 {
90 return "pcGainFactShmim";
91 };
92
93 static std::string indiPrefix()
94 {
95 return "pcGainFact";
96 };
97};
98
100{
101 static std::string configSection()
102 {
103 return "pcMultFactShmim";
104 };
105
106 static std::string indiPrefix()
107 {
108 return "pcMultFact";
109 };
110};
111
113{
114 static std::string configSection()
115 {
116 return "numpccoeffShmim";
117 };
118
119 static std::string indiPrefix()
120 {
121 return "numpccoeff";
122 };
123};
124
126{
127 static std::string configSection()
128 {
129 return "acoeffShmim";
130 };
131
132 static std::string indiPrefix()
133 {
134 return "acoeff";
135 };
136};
137
139{
140 static std::string configSection()
141 {
142 return "bcoeffShmim";
143 };
144
145 static std::string indiPrefix()
146 {
147 return "bcoeff";
148 };
149};
150
152{
153 static std::string configSection()
154 {
155 return "gainCalShmim";
156 };
157
158 static std::string indiPrefix()
159 {
160 return "gainCal";
161 };
162};
163
165{
166 static std::string configSection()
167 {
168 return "gainCalFactShmim";
169 };
170
171 static std::string indiPrefix()
172 {
173 return "gainCalFact";
174 };
175};
176
178{
179 static std::string configSection()
180 {
181 return "tauShmim";
182 };
183
184 static std::string indiPrefix()
185 {
186 return "tau";
187 };
188};
189
190/// The MagAO-X PSD-based gain optimizer
191/**
192 * \ingroup modalGainOpt
193 */
194class modalGainOpt : public MagAOXApp<true>,
195 dev::shmimMonitor<modalGainOpt, psdShmimT>,
196 dev::shmimMonitor<modalGainOpt, freqShmimT>,
197 dev::shmimMonitor<modalGainOpt, gainFactShmimT>,
198 dev::shmimMonitor<modalGainOpt, multFactShmimT>,
199 dev::shmimMonitor<modalGainOpt, pcGainFactShmimT>,
200 dev::shmimMonitor<modalGainOpt, pcMultFactShmimT>,
201 dev::shmimMonitor<modalGainOpt, numpccoeffShmimT>,
202 dev::shmimMonitor<modalGainOpt, acoeffShmimT>,
203 dev::shmimMonitor<modalGainOpt, bcoeffShmimT>,
204 dev::shmimMonitor<modalGainOpt, gainCalShmimT>,
205 dev::shmimMonitor<modalGainOpt, gainCalFactShmimT>,
206 dev::shmimMonitor<modalGainOpt, tauShmimT>
207{
208
209 // Give the test harness access.
210 friend class modalGainOpt_test;
211
224
225 public:
238
239 typedef std::chrono::time_point<std::chrono::steady_clock> timePointT;
240 typedef std::chrono::duration<double> durationT;
241
242 protected:
243 /** \name Configurable Parameters
244 *@{
245 */
246
247 int m_loopNum{ 1 }; ///< The number of the loop. Used to set shmim names, as in aolN_mgainfact.
248
249 std::string m_loopName; ///< The name of the loop control INDI device name.
250
251 std::string m_psdDevice; /**< The INDI device name of the PSD calculator. Defaults to aolN_modevalPSDs
252 where N is m_loopNum.*/
253
254 bool m_autoUpdate{ false }; ///< Flag controlling whether gains are automatically updated
255
256 float m_gainGain{ 0.1 }; ///< The gain to use for closed-loop gain updates. Default is 0.1.
257
258 ///@}
259
260 bool m_updateOnce{ false }; ///< Flag to trigger a single update with gain.
261
262 bool m_dump{ false }; ///< Flag to trigger a single update with no gain.
263
264 float m_fps{ 0 };
265
266 /// Each mode gets its own gain optimizer
267 std::vector<mx::AO::analysis::clGainOpt<float>> m_goptCurrent;
268 std::vector<mx::AO::analysis::clGainOpt<float>> m_goptSI;
269 std::vector<mx::AO::analysis::clGainOpt<float>> m_goptLP;
270 std::vector<mx::AO::analysis::clAOLinearPredictor<float>> m_linPred;
271
272 bool m_goptUpdated{ true }; ///< Tracks if a parameter has updated requiring updates to the m_gopt entries.
273 bool m_pcgoptUpdated{ true }; ///< Tracks if a parameter has updated requiring updates to the m_gopt entries.
274
275 bool m_freqUpdated{ true }; /**< Tracks if the frequency scale has updated, which necessitates additional calcs.
276 If true, implies m_goptUpdate == true.*/
277 float m_psdTime{ 1 };
278 float m_psdAvgTime{ 10 };
280
281 std::vector<float> m_freq;
282
283 mx::improc::eigenImage<float> m_clPSDs;
284 mx::improc::eigenImage<float> m_clXferCurrent;
285 mx::improc::eigenImage<float> m_clXferSI;
286 mx::improc::eigenImage<float> m_clXferLP;
287
288 std::vector<std::vector<float>> m_olPSDs;
289 std::vector<std::vector<float>> m_nPSDs;
290
291 std::vector<float> m_optGainSI;
292 std::vector<float> m_modeVarSI;
293
294 std::vector<float> m_optGainLP;
295 std::vector<float> m_modeVarLP;
296
297 bool m_loop{ false };
298
299 float m_gain{ 0 };
300
301 float m_mult{ 1 };
302
303 float m_siGain{ 0 };
304
305 float m_siMult{ 1 };
306
307 float m_pcGain{ 0 };
308
309 float m_pcMult{ 0 };
310
311 bool m_pcOn{ false };
312
313 std::vector<float> m_gainFacts;
314
315 std::vector<float> m_multFacts;
316
317 std::vector<float> m_pcGainFacts;
318
319 std::vector<float> m_pcMultFacts;
320
321 std::vector<int> m_Na;
322
323 std::vector<int> m_NaCurrent;
324
325 std::vector<int> m_Nb;
326
327 std::vector<int> m_NbCurrent;
328
330
332
333 std::vector<float> m_gmaxLP; ///< The previously calculated maximum gains for LP
334
335 int m_nRegCycles{ 60 }; ///< How often to regularize each mode
336
337 std::vector<int> m_regCounter; ///< Counters to track when this mode was last regularized
338
339 std::vector<float> m_regScale; ///< The regularizatio scale factors for each mode
340
341 std::vector<float> m_gainCals;
342
343 std::vector<float> m_gainCalFacts;
344
345 std::vector<float> m_taus;
346
347 int m_sinceChange{ -1 };
348
349 std::string m_olPSDShmimName;
353
357
358 IMAGE *m_olPSDStream{ nullptr }; ///< The ImageStreamIO shared memory buffer to publish the open loop PSDs
359 IMAGE *m_clXferCurrentStream{ nullptr }; ///< The ImageStreamIO shared memory buffer to publish the SI ETF
360 IMAGE *m_clXferSIStream{ nullptr }; ///< The ImageStreamIO shared memory buffer to publish the SI ETF
361 IMAGE *m_clXferLPStream{ nullptr }; ///< The ImageStreamIO shared memory buffer to publish the LP ETF
362
363 IMAGE *m_optGainStream{ nullptr }; ///< The ImageStreamIO shared memory buffer to publish the optimal gains
364 IMAGE *m_optGainSIStream{ nullptr }; ///< The ImageStreamIO shared memory buffer to publish the SI optimal gains
365 IMAGE *m_optGainLPStream{ nullptr }; ///< The ImageStreamIO shared memory buffer to publish the LP optimal gains
366
367 public:
368 /// Default c'tor.
369 modalGainOpt();
370
371 /// D'tor, declared and defined for noexcept.
373 {
374 }
375
376 virtual void setupConfig();
377
378 /// Implementation of loadConfig logic, separated for testing.
379 /** This is called by loadConfig().
380 */
381 int loadConfigImpl( mx::app::appConfigurator &_config /**< [in] an application configuration
382 from which to load values*/ );
383
384 virtual void loadConfig();
385
386 /// Startup function
387 /**
388 *
389 */
390 virtual int appStartup();
391
392 /// Implementation of the FSM for modalGainOpt.
393 /**
394 * \returns 0 on no critical error
395 * \returns -1 on an error requiring shutdown
396 */
397 virtual int appLogic();
398
399 /// Shutdown the app.
400 /**
401 *
402 */
403 virtual int appShutdown();
404
405 int allocate( const psdShmimT & ///< [in] tag to differentiate shmimMonitor parents.
406 );
407
408 int processImage( void *curr_src, ///< [in] pointer to the start of the current frame
409 const psdShmimT & ///< [in] tag to differentiate shmimMonitor parents.
410 );
411
412 int allocate( const freqShmimT & ///< [in] tag to differentiate shmimMonitor parents.
413 );
414
415 int processImage( void *curr_src, ///< [in] pointer to the start of the current frame
416 const freqShmimT & ///< [in] tag to differentiate shmimMonitor parents.
417 );
418
419 int allocate( const gainFactShmimT & ///< [in] tag to differentiate shmimMonitor parents.
420 );
421
422 int processImage( void *curr_src, ///< [in] pointer to the start of the current frame
423 const gainFactShmimT & ///< [in] tag to differentiate shmimMonitor parents.
424 );
425
426 int allocate( const multFactShmimT & ///< [in] tag to differentiate shmimMonitor parents.
427 );
428
429 int processImage( void *curr_src, ///< [in] pointer to the start of the current frame
430 const multFactShmimT & ///< [in] tag to differentiate shmimMonitor parents.
431 );
432
433 int allocate( const pcGainFactShmimT & ///< [in] tag to differentiate shmimMonitor parents.
434 );
435
436 int processImage( void *curr_src, ///< [in] pointer to the start of the current frame
437 const pcGainFactShmimT & ///< [in] tag to differentiate shmimMonitor parents.
438 );
439
440 int allocate( const pcMultFactShmimT & ///< [in] tag to differentiate shmimMonitor parents.
441 );
442
443 int processImage( void *curr_src, ///< [in] pointer to the start of the current frame
444 const pcMultFactShmimT & ///< [in] tag to differentiate shmimMonitor parents.
445 );
446
447 int allocate( const numpccoeffShmimT & ///< [in] tag to differentiate shmimMonitor parents.
448 );
449
450 int processImage( void *curr_src, ///< [in] pointer to the start of the current frame
451 const numpccoeffShmimT & ///< [in] tag to differentiate shmimMonitor parents.
452 );
453
454 int allocate( const acoeffShmimT & ///< [in] tag to differentiate shmimMonitor parents.
455 );
456
457 int processImage( void *curr_src, ///< [in] pointer to the start of the current frame
458 const acoeffShmimT & ///< [in] tag to differentiate shmimMonitor parents.
459 );
460
461 int allocate( const bcoeffShmimT & ///< [in] tag to differentiate shmimMonitor parents.
462 );
463
464 int processImage( void *curr_src, ///< [in] pointer to the start of the current frame
465 const bcoeffShmimT & ///< [in] tag to differentiate shmimMonitor parents.
466 );
467
468 int allocate( const gainCalShmimT & ///< [in] tag to differentiate shmimMonitor parents.
469 );
470
471 int processImage( void *curr_src, ///< [in] pointer to the start of the current frame
472 const gainCalShmimT & ///< [in] tag to differentiate shmimMonitor parents.
473 );
474
475 int allocate( const gainCalFactShmimT & ///< [in] tag to differentiate shmimMonitor parents.
476 );
477
478 int processImage( void *curr_src, ///< [in] pointer to the start of the current frame
479 const gainCalFactShmimT & ///< [in] tag to differentiate shmimMonitor parents.
480 );
481
482 int allocate( const tauShmimT & ///< [in] tag to differentiate shmimMonitor parents.
483 );
484
485 int processImage( void *curr_src, ///< [in] pointer to the start of the current frame
486 const tauShmimT & ///< [in] tag to differentiate shmimMonitor parents.
487 );
488
489 protected:
490 /// Mutex for synchronizing updates.
491 std::mutex m_goptMutex;
492
493 /// Flag used to indicate to the goptThread that it should stop calculations ASAP
494 bool m_updating{ false };
495
496 /** \name Gain Optimization Thread
497 *
498 * @{
499 */
500 int m_goptThreadPrio{ 0 }; ///< Priority of the gain optimization thread.
501
502 std::string m_goptThreadCpuset; ///< The cpuset to use for the gain optimization thread.
503
504 std::thread m_goptThread; ///< The gain optimization thread.
505
506 bool m_goptThreadInit{ true }; ///< Initialization flag for the gain optimization thread.
507
508 pid_t m_goptThreadID{ 0 }; ///< gain optimization thread PID.
509
510 pcf::IndiProperty m_goptThreadProp; ///< The property to hold the gain optimization thread details.
511
512 sem_t m_goptSemaphore; ///< Semaphore used to synchronize the psdShmim thread and the gopt thread.
513
514 /// Gain Optimization thread starter function
515 static void goptThreadStart( modalGainOpt *p /**< [in] pointer to this */ );
516
517 /// Gain optimization thread function
518 /** Runs until m_shutdown is true.
519 */
520 void goptThreadExec();
521
522 ///@}
523
524 public:
525 /** \name INDI
526 * @{
527 */
528
529 pcf::IndiProperty m_indiP_autoUpdate;
530 pcf::IndiProperty m_indiP_updateOnce;
531 pcf::IndiProperty m_indiP_dump;
532
533 pcf::IndiProperty m_indiP_gainGain;
534
535 pcf::IndiProperty m_indiP_fps;
536 pcf::IndiProperty m_indiP_psdTime;
537 pcf::IndiProperty m_indiP_psdAvgTime;
538 pcf::IndiProperty m_indiP_loop;
539 pcf::IndiProperty m_indiP_siGain;
540 pcf::IndiProperty m_indiP_siMult;
541 pcf::IndiProperty m_indiP_pcGain;
542 pcf::IndiProperty m_indiP_pcMult;
543 pcf::IndiProperty m_indiP_pcOn;
544
558
559 ///@}
560};
561
579
581{
582 config.add( "loop.number",
583 "",
584 "loop.number",
585 argType::Required,
586 "loop",
587 "number",
588 false,
589 "int",
590 "The number of the loop. Used to set shmim names, as in aolN_mgainfact." );
591
592 config.add( "loop.name",
593 "",
594 "loop.name",
595 argType::Required,
596 "loop",
597 "name",
598 false,
599 "string",
600 "The name of the loop control INDI device name." );
601
602 config.add( "loop.psdDev",
603 "",
604 "loop.psdDev",
605 argType::Required,
606 "loop",
607 "psdDev",
608 false,
609 "string",
610 "The INDI device name of the PSD calculator. Defaults to aolN_modevalPSDs where N is loop.number." );
611
612 config.add( "loop.autoUpdate",
613 "",
614 "loop.autoUpdate",
615 argType::Required,
616 "loop",
617 "autoUpdate",
618 false,
619 "bool",
620 "Flag controlling whether the gains are auto updated. Also settable via INDI." );
621
622 config.add( "loop.gainGain",
623 "",
624 "loop.gainGain",
625 argType::Required,
626 "loop",
627 "gainGain",
628 false,
629 "float",
630 "The gain to use for closed-loop gain updates. Default is 0.1" );
631
644}
645
646int modalGainOpt::loadConfigImpl( mx::app::appConfigurator &_config )
647{
648 _config( m_loopNum, "loop.number" );
649 _config( m_loopName, "loop.name" );
650 _config( m_autoUpdate, "loop.audoUpdate" );
651 _config( m_autoUpdate, "loop.gainGain" );
652
653 char shmim[1024];
654
655 snprintf( shmim, sizeof( shmim ), "aol%d_modevalPSDs", m_loopNum );
656 m_psdDevice = shmim;
657 _config( m_psdDevice, "loop.psdDev" );
658
661
664
665 snprintf( shmim, sizeof( shmim ), "aol%d_mgainfact", m_loopNum );
668
669 snprintf( shmim, sizeof( shmim ), "aol%d_mmultfact", m_loopNum );
672
673 snprintf( shmim, sizeof( shmim ), "aol%d_mpcgainfact", m_loopNum );
676
677 snprintf( shmim, sizeof( shmim ), "aol%d_mpcmultfact", m_loopNum );
680
681 snprintf( shmim, sizeof( shmim ), "aol%d_numpccoeff", m_loopNum );
684
685 snprintf( shmim, sizeof( shmim ), "aol%d_acoeff", m_loopNum );
688
689 snprintf( shmim, sizeof( shmim ), "aol%d_bcoeff", m_loopNum );
692
693 snprintf( shmim, sizeof( shmim ), "aol%d_mgaincal", m_loopNum );
696
697 snprintf( shmim, sizeof( shmim ), "aol%d_mgaincalfact", m_loopNum );
700
701 snprintf( shmim, sizeof( shmim ), "aol%d_looptau", m_loopNum );
704
705 snprintf( shmim, sizeof( shmim ), "aol%d_olpsds", m_loopNum );
706 m_olPSDShmimName = shmim;
707
708 snprintf( shmim, sizeof( shmim ), "aol%d_clxferCurrent", m_loopNum );
710
711 snprintf( shmim, sizeof( shmim ), "aol%d_clxferSI", m_loopNum );
712 m_clXferSIShmimName = shmim;
713
714 snprintf( shmim, sizeof( shmim ), "aol%d_clxferLP", m_loopNum );
715 m_clXferLPShmimName = shmim;
716
717 snprintf( shmim, sizeof( shmim ), "aol%d_mgainoptimal", m_loopNum );
718 m_optGainShmimName = shmim;
719
720 snprintf( shmim, sizeof( shmim ), "aol%d_mgainoptimalSI", m_loopNum );
721 m_optGainSIShmimName = shmim;
722
723 snprintf( shmim, sizeof( shmim ), "aol%d_mgainoptimalLP", m_loopNum );
724 m_optGainLPShmimName = shmim;
725
726 return 0;
727}
728
730{
731 loadConfigImpl( config );
732}
733
735{
748
752
753 CREATE_REG_INDI_NEW_NUMBERF( m_indiP_gainGain, "gainGain", 0, 1, 0.01, "%0.01f", "Gain Gain", "Gain Opt." );
754
757 REG_INDI_SETPROP( m_indiP_loop, m_loopName, "loop_state" );
759 REG_INDI_SETPROP( m_indiP_siMult, m_loopName, "loop_multcoeff" );
760 REG_INDI_SETPROP( m_indiP_pcGain, m_loopName, "loop_pcgain" );
761 REG_INDI_SETPROP( m_indiP_pcMult, m_loopName, "loop_pcmultcoeff" );
763
764 if( sem_init( &m_goptSemaphore, 0, 0 ) < 0 )
765 {
766 return log<software_critical, -1>( { __FILE__, __LINE__, errno, 0, "Initializing gopt semaphore" } );
767 }
768
775 "gainopt",
777
779 return 0;
780}
781
783{
796
797 XWCAPP_THREAD_CHECK( m_goptThread, "gainopt" );
798
811
812 if( m_autoUpdate )
813 {
814 updateSwitchIfChanged( m_indiP_autoUpdate, "toggle", pcf::IndiElement::On, INDI_OK );
815 }
816 else
817 {
818 updateSwitchIfChanged( m_indiP_autoUpdate, "toggle", pcf::IndiElement::Off, INDI_IDLE );
819 }
820
821 if( m_updateOnce )
822 {
823 updateSwitchIfChanged( m_indiP_updateOnce, "request", pcf::IndiElement::On, INDI_OK );
824 }
825 else
826 {
827 updateSwitchIfChanged( m_indiP_updateOnce, "request", pcf::IndiElement::Off, INDI_IDLE );
828 }
829
830 if( m_dump )
831 {
832 updateSwitchIfChanged( m_indiP_dump, "request", pcf::IndiElement::On, INDI_OK );
833 }
834 else
835 {
836 updateSwitchIfChanged( m_indiP_dump, "request", pcf::IndiElement::Off, INDI_IDLE );
837 }
838
839 updatesIfChanged<float>( m_indiP_gainGain, { "current", "target" }, { m_gainGain, m_gainGain } );
840
841 return 0;
842}
843
845{
847
860
861 if( m_olPSDStream != nullptr )
862 {
865 m_olPSDStream = nullptr;
866
869 m_clXferCurrentStream = nullptr;
870
873 m_clXferSIStream = nullptr;
874
877 m_clXferLPStream = nullptr;
878 }
879
880 if( m_optGainStream != nullptr )
881 {
884 m_optGainStream = nullptr;
885
888 m_optGainSIStream = nullptr;
889
892 m_optGainLPStream = nullptr;
893 }
894
895 return 0;
896}
897
899{
900 static_cast<void>( dummy );
901
902 m_updating = true;
903 std::lock_guard<std::mutex> lock( m_goptMutex );
904 m_updating = true;
905
910
913
914 for( size_t n = 0; n < m_olPSDs.size(); ++n )
915 {
918 }
919
922
925
926 if( m_olPSDStream != nullptr && ( m_olPSDStream->md->size[0] != psdShmimMonitorT::m_width ||
927 m_olPSDStream->md->size[1] != psdShmimMonitorT::m_height ) )
928 {
931 m_olPSDStream = nullptr;
932
935 m_clXferCurrentStream = nullptr;
936
939 m_clXferSIStream = nullptr;
940
943 m_clXferLPStream = nullptr;
944 }
945
946 if( m_olPSDStream == nullptr )
947 {
948 m_olPSDStream = static_cast<IMAGE *>( malloc( sizeof( IMAGE ) ) );
949 uint32_t imsize[3];
950
953 imsize[2] = 1;
955 m_olPSDShmimName.c_str(),
956 3,
957 imsize,
959 -1,
960 1,
962 0,
964 0 );
965
966 m_olPSDStream->md->cnt0 = 0;
967 m_olPSDStream->md->cnt1 = 0;
968
969 m_clXferCurrentStream = static_cast<IMAGE *>( malloc( sizeof( IMAGE ) ) );
970
973 3,
974 imsize,
976 -1,
977 1,
979 0,
981 0 );
982
983 m_clXferCurrentStream->md->cnt0 = 0;
984 m_clXferCurrentStream->md->cnt1 = 0;
985
986 m_clXferSIStream = static_cast<IMAGE *>( malloc( sizeof( IMAGE ) ) );
987
989 m_clXferSIShmimName.c_str(),
990 3,
991 imsize,
993 -1,
994 1,
996 0,
998 0 );
999
1000 m_clXferSIStream->md->cnt0 = 0;
1001 m_clXferSIStream->md->cnt1 = 0;
1002
1003 m_clXferLPStream = static_cast<IMAGE *>( malloc( sizeof( IMAGE ) ) );
1004
1006 m_clXferLPShmimName.c_str(),
1007 3,
1008 imsize,
1010 -1,
1011 1,
1013 0,
1015 0 );
1016
1017 m_clXferLPStream->md->cnt0 = 0;
1018 m_clXferLPStream->md->cnt1 = 0;
1019 }
1020
1021 if( m_optGainStream != nullptr &&
1022 ( m_optGainStream->md->size[0] != psdShmimMonitorT::m_height || m_optGainStream->md->size[1] != 1 ) )
1023 {
1026 m_optGainStream = nullptr;
1027
1030 m_optGainSIStream = nullptr;
1031
1034 m_optGainSIStream = nullptr;
1035 }
1036
1037 if( m_optGainStream == nullptr )
1038 {
1039 m_optGainStream = static_cast<IMAGE *>( malloc( sizeof( IMAGE ) ) );
1040 uint32_t imsize[3];
1041
1043 imsize[1] = 1;
1044 imsize[2] = 1;
1046 m_optGainShmimName.c_str(),
1047 3,
1048 imsize,
1050 -1,
1051 1,
1053 0,
1055 0 );
1056
1057 m_optGainStream->md->cnt0 = 0;
1058 m_optGainStream->md->cnt1 = 0;
1059
1060 m_optGainSIStream = static_cast<IMAGE *>( malloc( sizeof( IMAGE ) ) );
1061
1063 m_optGainSIShmimName.c_str(),
1064 3,
1065 imsize,
1067 -1,
1068 1,
1070 0,
1072 0 );
1073
1074 m_optGainSIStream->md->cnt0 = 0;
1075 m_optGainSIStream->md->cnt1 = 0;
1076
1077 m_optGainLPStream = static_cast<IMAGE *>( malloc( sizeof( IMAGE ) ) );
1078
1080 m_optGainLPShmimName.c_str(),
1081 3,
1082 imsize,
1084 -1,
1085 1,
1087 0,
1089 0 );
1090
1091 m_optGainLPStream->md->cnt0 = 0;
1092 m_optGainLPStream->md->cnt1 = 0;
1093 }
1094
1095 m_sinceChange = -1;
1096
1097 m_updating = false;
1098 return 0;
1099}
1100
1101int modalGainOpt::processImage( void *curr_src, const psdShmimT &dummy )
1102{
1103 static_cast<void>( dummy );
1104
1105 ++m_sinceChange;
1106
1107 if( m_psdAvgTime <= 0 || m_psdTime <= 0 ) // Safety check, shouldn't happen but means we need to wait.
1108 {
1109 return 0;
1110 }
1111
1113
1114 if( m_sinceChange < deadTime )
1115 {
1116 return 0;
1117 }
1118
1119 // Here we would update psds, but don't do that if we're in the middle of calculating
1120 std::unique_lock<std::mutex> lock( m_goptMutex, std::try_to_lock );
1121 if( !lock.owns_lock() )
1122 {
1123 ///\todo update a frame-missed counter
1124 return 0;
1125 }
1126
1127 m_updating = true;
1128
1129 m_clPSDs = Eigen::Map<Eigen::Array<float, -1, -1>>(
1131
1132 m_updating = false;
1133
1134 lock.unlock();
1135
1136 if( sem_post( &m_goptSemaphore ) < 0 )
1137 {
1138 return log<software_critical, -1>( { __FILE__, __LINE__, errno, 0, "Error posting to semaphore" } );
1139 }
1140
1141 return 0;
1142}
1143
1145{
1146 static_cast<void>( dummy );
1147
1148 return 0;
1149}
1150
1151int modalGainOpt::processImage( void *curr_src, const freqShmimT &dummy )
1152{
1153 static_cast<void>( dummy );
1154
1156 {
1157 return log<software_error, -1>( { __FILE__, __LINE__, "got freq with width not 1" } );
1158 }
1159
1160 bool change = false;
1161
1162 float *f = static_cast<float *>( curr_src );
1163
1165
1166 if( sz != m_freq.size() )
1167 {
1168 change = true;
1169 }
1170
1171 if( !change ) // f is same size
1172 {
1173 for( size_t n = 0; n < sz; ++n )
1174 {
1175 if( f[n] != m_freq[n] )
1176 {
1177 change = true;
1178 break;
1179 }
1180 }
1181 }
1182
1183 if( change )
1184 {
1185 m_updating = true;
1186 std::lock_guard<std::mutex> lock( m_goptMutex );
1187 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1188
1189 m_freq.resize( sz );
1190
1191 for( size_t n = 0; n < sz; ++n )
1192 {
1193 m_freq[n] = f[n];
1194 }
1195
1196 m_fps = 2 * m_freq.back();
1197
1198 m_sinceChange = -1;
1199 m_goptUpdated = true;
1200 m_freqUpdated = true;
1201
1202 m_updating = false;
1203 std::cerr << "got freq: " << sz << '\n';
1204 std::cerr << " fps: " << m_fps << '\n';
1205 }
1206
1207 return 0;
1208}
1209
1211{
1212 static_cast<void>( dummy );
1213
1215 {
1216 return log<software_error, -1>( { __FILE__, __LINE__, "got gains with height not 1" } );
1217 }
1218
1219 return 0;
1220}
1221
1222int modalGainOpt::processImage( void *curr_src, const gainFactShmimT &dummy )
1223{
1224 static_cast<void>( dummy );
1225
1226 bool change = false;
1227
1229
1230 std::unique_lock<std::mutex> lock( m_goptMutex, std::defer_lock );
1231
1232 if( w != m_gainFacts.size() )
1233 {
1234 m_updating = true;
1235 lock.lock();
1236 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1237
1238 change = true;
1239 m_gainFacts.resize( w );
1240 }
1241
1242 float *g = static_cast<float *>( curr_src );
1243
1244 for( uint32_t n = 0; n < w; ++n )
1245 {
1246 if( change || m_gainFacts[n] != g[n] )
1247 {
1248 if( !change )
1249 {
1250 m_updating = true;
1251 lock.lock();
1252 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1253 change = true;
1254 }
1255
1256 m_gainFacts[n] = g[n];
1257 }
1258 }
1259
1260 if( change )
1261 {
1262 if( m_loop )
1263 {
1264 m_sinceChange = -1;
1265 }
1266
1267 m_updating = false;
1268 lock.unlock();
1269 std::cerr << "got gains: " << m_gainFacts.size() << "\n";
1270 }
1271
1272 return 0;
1273}
1274
1276{
1277 static_cast<void>( dummy );
1278
1280 {
1281 return log<software_error, -1>( { __FILE__, __LINE__, "got multcoeffs with height not 1" } );
1282 }
1283
1284 return 0;
1285}
1286
1287int modalGainOpt::processImage( void *curr_src, const multFactShmimT &dummy )
1288{
1289 static_cast<void>( dummy );
1290
1291 bool change = false;
1292
1294
1295 std::unique_lock<std::mutex> lock( m_goptMutex, std::defer_lock );
1296
1297 if( w != m_multFacts.size() )
1298 {
1299 m_updating = true;
1300 lock.lock();
1301 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1302
1303 change = true;
1304 m_multFacts.resize( w );
1305 }
1306
1307 float *m = static_cast<float *>( curr_src );
1308
1309 for( uint32_t n = 0; n < w; ++n )
1310 {
1311 if( change || m_multFacts[n] != m[n] )
1312 {
1313 if( !change )
1314 {
1315 m_updating = true;
1316 lock.lock();
1317 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1318
1319 change = true;
1320 }
1321
1322 m_multFacts[n] = m[n];
1323 }
1324 }
1325
1326 if( change )
1327 {
1328 if( m_loop )
1329 {
1330 m_sinceChange = -1;
1331 }
1332
1333 m_updating = false;
1334 m_goptUpdated = true;
1335
1336 lock.unlock();
1337 std::cerr << "got mcs: " << m_multFacts.size() << "\n";
1338 }
1339
1340 return 0;
1341}
1342
1344{
1345 static_cast<void>( dummy );
1346
1348 {
1349 return log<software_error, -1>( { __FILE__, __LINE__, "got pc gains with height not 1" } );
1350 }
1351
1352 return 0;
1353}
1354
1355int modalGainOpt::processImage( void *curr_src, const pcGainFactShmimT &dummy )
1356{
1357 static_cast<void>( dummy );
1358
1359 bool change = false;
1360
1362
1363 std::unique_lock<std::mutex> lock( m_goptMutex, std::defer_lock );
1364
1365 if( w != m_pcGainFacts.size() )
1366 {
1367 m_updating = true;
1368 lock.lock();
1369 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1370
1371 change = true;
1372 m_pcGainFacts.resize( w );
1373 }
1374
1375 float *g = static_cast<float *>( curr_src );
1376
1377 for( uint32_t n = 0; n < w; ++n )
1378 {
1379 if( change || m_pcGainFacts[n] != g[n] )
1380 {
1381 if( !change )
1382 {
1383 m_updating = true;
1384 lock.lock();
1385 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1386 change = true;
1387 }
1388
1389 m_pcGainFacts[n] = g[n];
1390 }
1391 }
1392
1393 if( change )
1394 {
1395 if( m_loop )
1396 {
1397 m_sinceChange = -1;
1398 }
1399
1400 m_updating = false;
1401 lock.unlock();
1402 std::cerr << "got pc gains: " << m_pcGainFacts.size() << "\n";
1403 }
1404
1405 return 0;
1406}
1407
1409{
1410 static_cast<void>( dummy );
1411
1413 {
1414 return log<software_error, -1>( { __FILE__, __LINE__, "got pcMultcoeffs with height not 1" } );
1415 }
1416
1417 return 0;
1418}
1419
1420int modalGainOpt::processImage( void *curr_src, const pcMultFactShmimT &dummy )
1421{
1422 static_cast<void>( dummy );
1423
1424 bool change = false;
1425
1427
1428 std::unique_lock<std::mutex> lock( m_goptMutex, std::defer_lock );
1429
1430 if( w != m_pcMultFacts.size() )
1431 {
1432 m_updating = true;
1433 lock.lock();
1434 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1435
1436 change = true;
1437 m_pcMultFacts.resize( w );
1438 }
1439
1440 float *m = static_cast<float *>( curr_src );
1441
1442 for( uint32_t n = 0; n < w; ++n )
1443 {
1444 if( change || m_pcMultFacts[n] != m[n] )
1445 {
1446 if( !change )
1447 {
1448 m_updating = true;
1449 lock.lock();
1450 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1451
1452 change = true;
1453 }
1454
1455 m_pcMultFacts[n] = m[n];
1456 }
1457 }
1458
1459 if( change )
1460 {
1461 if( m_loop )
1462 {
1463 m_sinceChange = -1;
1464 }
1465
1466 m_updating = false;
1467 m_pcgoptUpdated = true;
1468
1469 lock.unlock();
1470 std::cerr << "got mcs: " << m_multFacts.size() << "\n";
1471 }
1472
1473 return 0;
1474}
1475
1477{
1478 static_cast<void>( dummy );
1479
1481 {
1482 return log<software_error, -1>( { __FILE__, __LINE__, "got numpccoeff's with height not 2" } );
1483 }
1484
1485 return 0;
1486}
1487
1488int modalGainOpt::processImage( void *curr_src, const numpccoeffShmimT &dummy )
1489{
1490 static_cast<void>( dummy );
1491
1492 bool change = false;
1493
1495
1496 std::unique_lock<std::mutex> lock( m_goptMutex, std::defer_lock );
1497
1498 if( w != m_Na.size() || w != m_Nb.size() || w != m_regCounter.size() || w != m_regScale.size() )
1499 {
1500 m_updating = true;
1501 lock.lock();
1502 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1503
1504 change = true;
1505 m_Na.resize( w );
1506 m_Nb.resize( w );
1507
1508 m_regCounter.resize( w );
1509
1510 // Initialize the regCounter
1511 int nc = 0;
1512
1513 for( size_t n = 0; n < m_regCounter.size(); ++n )
1514 {
1515 m_regCounter[n] = nc;
1516 ++nc;
1517
1518 if( nc >= m_nRegCycles )
1519 {
1520 nc = 0;
1521 }
1522 }
1523
1524 m_regScale.resize( w, -999 );
1525 m_gmaxLP.resize( w, 0 );
1526 }
1527
1528 mx::improc::eigenMap<int> Npc( reinterpret_cast<int *>( curr_src ), w, 2 );
1529
1530 for( uint32_t n = 0; n < w; ++n )
1531 {
1532 if( change || m_Na[n] != Npc( n, 0 ) || m_Nb[n] != Npc( n, 1 ) )
1533 {
1534 if( !change )
1535 {
1536 m_updating = true;
1537 lock.lock();
1538 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1539
1540 change = true;
1541 }
1542
1543 m_Na[n] = Npc( n, 0 );
1544 m_Nb[n] = Npc( n, 1 );
1545 }
1546 }
1547
1548 if( change )
1549 {
1550 if( m_loop )
1551 {
1552 m_sinceChange = -1;
1553 }
1554
1555 m_updating = false;
1556 m_goptUpdated = true;
1557
1558 lock.unlock();
1559 std::cerr << "got num pc coeffs: " << m_Na.size() << "\n";
1560 }
1561
1562 return 0;
1563 return 0;
1564}
1565
1567{
1568 static_cast<void>( dummy );
1569
1570 return 0;
1571}
1572
1573int modalGainOpt::processImage( void *curr_src, const acoeffShmimT &dummy )
1574{
1575 static_cast<void>( dummy );
1576
1577 bool change = false;
1578
1579 std::unique_lock<std::mutex> lock( m_goptMutex, std::defer_lock );
1580
1583
1584 // If there's a size change we lock
1585 if( w - 1 != m_as.rows() || h != m_as.cols() || h != m_NaCurrent.size() )
1586 {
1587 m_updating = true;
1588 lock.lock();
1589 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1590
1591 change = true;
1592 m_NaCurrent.resize( h );
1593 m_as.resize( w - 1, h );
1594 }
1595
1596 eigenMap<float> ac( reinterpret_cast<float *>( curr_src ), w, h );
1597
1598 for( uint32_t cc = 0; cc < h; ++cc )
1599 {
1600 if( change || m_NaCurrent[cc] != ac( 0, cc ) )
1601 {
1602 if( !change )
1603 {
1604 m_updating = true;
1605 lock.lock();
1606 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1607
1608 change = true;
1609 }
1610
1611 m_NaCurrent[cc] = ac( 0, cc );
1612 }
1613
1614 for( uint32_t rr = 1; rr < w; ++rr )
1615 {
1616 if( change || m_as( rr - 1, cc ) != ac( rr, cc ) )
1617 {
1618 if( !change )
1619 {
1620 m_updating = true;
1621 lock.lock();
1622 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1623
1624 change = true;
1625 }
1626
1627 m_as( rr - 1, cc ) = ac( rr, cc );
1628 }
1629 }
1630 }
1631
1632 if( change )
1633 {
1634 if( m_loop && m_pcOn )
1635 {
1636 m_sinceChange = -1;
1637 }
1638
1639 m_updating = false;
1640 m_pcgoptUpdated = true;
1641
1642 lock.unlock();
1643 std::cerr << "got a coeffs: " << w << ' ' << h << ' ' << m_NaCurrent.size() << "\n";
1644 }
1645
1646 return 0;
1647}
1648
1650{
1651 static_cast<void>( dummy );
1652
1653 return 0;
1654}
1655
1656int modalGainOpt::processImage( void *curr_src, const bcoeffShmimT &dummy )
1657{
1658 static_cast<void>( dummy );
1659
1660 bool change = false;
1661
1662 std::unique_lock<std::mutex> lock( m_goptMutex, std::defer_lock );
1663
1666
1667 // If there's a size change we lock
1668 if( w - 1 != m_bs.rows() || h != m_bs.cols() || h != m_NbCurrent.size() )
1669 {
1670 m_updating = true;
1671 lock.lock();
1672 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1673
1674 change = true;
1675
1676 m_NbCurrent.resize( h );
1677 m_bs.resize( w - 1, h );
1678 }
1679
1680 eigenMap<float> bc( reinterpret_cast<float *>( curr_src ), w, h );
1681
1682 for( uint32_t cc = 0; cc < h; ++cc )
1683 {
1684 if( change || m_NbCurrent[cc] != bc( 0, cc ) )
1685 {
1686 if( !change )
1687 {
1688 m_updating = true;
1689 lock.lock();
1690 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1691
1692 change = true;
1693 }
1694
1695 m_NbCurrent[cc] = bc( 0, cc );
1696 }
1697
1698 for( uint32_t rr = 1; rr < w; ++rr )
1699 {
1700 if( change || m_bs( rr - 1, cc ) != bc( rr, cc ) )
1701 {
1702 if( !change )
1703 {
1704 m_updating = true;
1705 lock.lock();
1706 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1707
1708 change = true;
1709 }
1710
1711 m_bs( rr - 1, cc ) = bc( rr, cc );
1712 }
1713 }
1714 }
1715
1716 if( change )
1717 {
1718 if( m_loop && m_pcOn )
1719 {
1720 m_sinceChange = -1;
1721 }
1722
1723 m_updating = false;
1724 m_pcgoptUpdated = true;
1725
1726 lock.unlock();
1727 std::cerr << "got b coeffs: " << m_NbCurrent.size() << "\n";
1728 }
1729
1730 return 0;
1731}
1732
1734{
1735 static_cast<void>( dummy );
1736
1737 return 0;
1738}
1739
1740int modalGainOpt::processImage( void *curr_src, const gainCalShmimT &dummy )
1741{
1742 static_cast<void>( dummy );
1743
1745 {
1746 return log<software_error, -1>( { __FILE__, __LINE__, "got gainCals with height not 1" } );
1747 }
1748
1749 bool change = false;
1750
1752
1753 std::unique_lock<std::mutex> lock( m_goptMutex, std::defer_lock );
1754
1755 if( w != m_gainCals.size() )
1756 {
1757 m_updating = true;
1758 lock.lock();
1759 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1760
1761 change = true;
1762 m_gainCals.resize( w );
1763 }
1764
1765 float *g = static_cast<float *>( curr_src );
1766
1767 for( uint32_t n = 0; n < w; ++n )
1768 {
1769 if( change || m_gainCals[n] != g[n] )
1770 {
1771 if( !change )
1772 {
1773 m_updating = true;
1774 lock.lock();
1775 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1776
1777 change = true;
1778 }
1779
1780 m_gainCals[n] = g[n];
1781 }
1782 }
1783
1784 if( change )
1785 {
1786 m_sinceChange = -1;
1787 m_updating = false;
1788 lock.unlock();
1789 std::cerr << "got gainCals: " << m_gainCals.size() << "\n";
1790 }
1791
1792 return 0;
1793}
1794
1796{
1797 static_cast<void>( dummy );
1798
1799 return 0;
1800}
1801
1802int modalGainOpt::processImage( void *curr_src, const gainCalFactShmimT &dummy )
1803{
1804 static_cast<void>( dummy );
1805
1807 {
1808 return log<software_error, -1>( { __FILE__, __LINE__, "got gainCalFacts with height not 1" } );
1809 }
1810
1811 bool change = false;
1812
1814
1815 std::unique_lock<std::mutex> lock( m_goptMutex, std::defer_lock );
1816
1817 if( w != m_gainCalFacts.size() )
1818 {
1819 m_updating = true;
1820 lock.lock();
1821 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1822
1823 change = true;
1824 m_gainCalFacts.resize( w );
1825 }
1826
1827 float *g = static_cast<float *>( curr_src );
1828
1829 for( uint32_t n = 0; n < w; ++n )
1830 {
1831 if( change || m_gainCalFacts[n] != g[n] )
1832 {
1833 if( !change )
1834 {
1835 m_updating = true;
1836 lock.lock();
1837 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1838
1839 change = true;
1840 }
1841
1842 m_gainCalFacts[n] = g[n];
1843 }
1844 }
1845
1846 if( change )
1847 {
1848 m_sinceChange = -1;
1849 m_updating = false;
1850 lock.unlock();
1851 std::cerr << "got gainCalsFacts: " << m_gainCalFacts.size() << "\n";
1852 }
1853
1854 return 0;
1855}
1856
1858{
1859 static_cast<void>( dummy );
1860
1861 return 0;
1862}
1863
1864int modalGainOpt::processImage( void *curr_src, const tauShmimT &dummy )
1865{
1866 static_cast<void>( dummy );
1867
1869 {
1870 return log<software_error, -1>( { __FILE__, __LINE__, "got tau with height not 1" } );
1871 }
1872
1873 bool change = false;
1874
1876
1877 std::unique_lock<std::mutex> lock( m_goptMutex, std::defer_lock );
1878
1879 if( w != m_taus.size() )
1880 {
1881 m_updating = true;
1882 lock.lock();
1883 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1884
1885 change = true;
1886 m_taus.resize( w );
1887 }
1888
1889 float *t = static_cast<float *>( curr_src );
1890
1891 for( uint32_t n = 0; n < w; ++n )
1892 {
1893 if( change || m_taus[n] != t[n] )
1894 {
1895 if( !change )
1896 {
1897 m_updating = true;
1898 lock.lock();
1899 m_updating = true; // Make sure it didn't get set to false by thread that had the lock
1900
1901 change = true;
1902 }
1903
1904 m_taus[n] = t[n];
1905 }
1906 }
1907
1908 if( change )
1909 {
1910 m_sinceChange = -1;
1911 m_updating = false;
1912 m_goptUpdated = true;
1913 lock.unlock();
1914 std::cerr << "got taus: " << m_taus.size() << "\n";
1915 }
1916 return 0;
1917}
1918
1923
1925{
1927
1928 while( m_goptThreadInit == true && shutdown() == 0 )
1929 {
1930 sleep( 1 );
1931 }
1932
1933 std::vector<bool> logged( 50, false );
1934
1935 while( shutdown() == 0 )
1936 {
1937 timespec ts;
1938 XWC_SEM_WAIT_TS_RETVOID( ts, 1, 0 );
1939
1940 if( sem_timedwait( &m_goptSemaphore, &ts ) == 0 )
1941 {
1942 std::lock_guard<std::mutex> lock( m_goptMutex );
1943
1944 int L = 0;
1945 if( (size_t)m_clPSDs.rows() == 0 || m_clPSDs.cols() == 0 ) // somehow here without any data
1946 {
1947 if( !logged[L] )
1948 {
1949 log<software_error>( { __FILE__, __LINE__, "PSDs have not been updated" } );
1950 }
1951 logged[L] = true;
1952 continue;
1953 }
1954 logged[L++] = false;
1955
1956 if( (size_t)m_clPSDs.rows() != m_freq.size() )
1957 {
1958 if( !logged[L] )
1959 {
1960 log<software_error>( { __FILE__, __LINE__, "PSDs and freq size mismatch" } );
1961 }
1962 logged[L] = true;
1963 continue;
1964 }
1965 logged[L++] = false;
1966
1967 if( (size_t)m_clPSDs.cols() != m_gainFacts.size() )
1968 {
1969 if( !logged[L] )
1970 {
1971 log<software_error>( { __FILE__, __LINE__, "PSDs and gains number of modes mismatch" } );
1972 }
1973 logged[L] = true;
1974 continue;
1975 }
1976 logged[L++] = false;
1977
1978 if( (size_t)m_clPSDs.cols() != m_multFacts.size() )
1979 {
1980 if( !logged[L] )
1981 {
1982 log<software_error>( { __FILE__, __LINE__, "PSDs and mult coeffs number of modes mismatch" } );
1983 }
1984 logged[L] = true;
1985 continue;
1986 }
1987 logged[L++] = false;
1988
1989 if( (size_t)m_clPSDs.cols() != m_gainCals.size() )
1990 {
1991 if( !logged[L] )
1992 {
1993 log<software_error>( { __FILE__, __LINE__, "PSDs and gain cals number of modes mismatch" } );
1994 }
1995 logged[L] = true;
1996 continue;
1997 }
1998 logged[L++] = false;
1999
2000 if( (size_t)m_clPSDs.cols() != m_gainCalFacts.size() )
2001 {
2002 if( !logged[L] )
2003 {
2004 log<software_error>( { __FILE__, __LINE__, "PSDs and gain cal facts number of modes mismatch" } );
2005 }
2006 logged[L] = true;
2007 continue;
2008 }
2009 logged[L++] = false;
2010
2011 if( (size_t)m_clPSDs.cols() != m_taus.size() )
2012 {
2013 if( !logged[L] )
2014 {
2015 log<software_error>( { __FILE__, __LINE__, "Loop taus have not been set" } );
2016 }
2017 logged[L] = true;
2018 continue;
2019 }
2020 logged[L++] = false;
2021
2022 if( m_fps <= 0 )
2023 {
2024 if( !logged[L] )
2025 {
2026 log<software_error>( { __FILE__, __LINE__, "Loop fps has not been set" } );
2027 }
2028 logged[L] = true;
2029 continue;
2030 }
2031 logged[L++] = false;
2032
2033 if( m_olPSDStream == nullptr )
2034 {
2035 if( !logged[L] )
2036 {
2037 log<software_error>( { __FILE__, __LINE__, "m_olPSDStream is not allocated" } );
2038 }
2039 logged[L] = true;
2040 continue;
2041 }
2042 logged[L++] = false;
2043
2044 if( m_clXferCurrentStream == nullptr )
2045 {
2046 if( !logged[L] )
2047 {
2048 log<software_error>( { __FILE__, __LINE__, "m_clXferCurrentStream is not allocated" } );
2049 }
2050 logged[L] = true;
2051 continue;
2052 }
2053 logged[L++] = false;
2054
2055 if( m_clXferSIStream == nullptr )
2056 {
2057 if( !logged[L] )
2058 {
2059 log<software_error>( { __FILE__, __LINE__, "m_clXferSIStream is not allocated" } );
2060 }
2061 logged[L] = true;
2062 continue;
2063 }
2064 logged[L++] = false;
2065
2066 if( m_optGainStream == nullptr )
2067 {
2068 if( !logged[L] )
2069 {
2070 log<software_error>( { __FILE__, __LINE__, "optGainsStream is not allocated" } );
2071 }
2072 logged[L] = true;
2073 continue;
2074 }
2075 logged[L++] = false;
2076
2077 if( m_optGainSIStream == nullptr )
2078 {
2079 if( !logged[L] )
2080 {
2081 log<software_error>( { __FILE__, __LINE__, "optGainsStream SI is not allocated" } );
2082 }
2083 logged[L] = true;
2084 continue;
2085 }
2086 logged[L++] = false;
2087
2088 bool doPCCalcs = true;
2089
2090 if( (size_t)m_clPSDs.cols() != m_Na.size() )
2091 {
2092 if( !logged[L] )
2093 {
2094 log<software_error>( { __FILE__, __LINE__, "Na is not allocated" } );
2095 }
2096 logged[L] = true;
2097 doPCCalcs = false;
2098 }
2099 else
2100 {
2101 logged[L] = false;
2102 }
2103 ++L;
2104
2105 if( (size_t)m_clPSDs.cols() != m_NaCurrent.size() )
2106 {
2107 if( !logged[L] )
2108 {
2109 log<software_error>( { __FILE__, __LINE__, "NaCurrent is not allocated" } );
2110 }
2111 logged[L] = true;
2112 doPCCalcs = false;
2113 }
2114 else
2115 {
2116 logged[L] = false;
2117 }
2118 ++L;
2119
2120 if( m_clPSDs.cols() != m_as.cols() )
2121 {
2122 if( !logged[L] )
2123 {
2124 log<software_error>( { __FILE__, __LINE__, "a coefficients are not allocated" } );
2125 }
2126 logged[L] = true;
2127 doPCCalcs = false;
2128 }
2129 else
2130 {
2131 logged[L] = false;
2132 }
2133 ++L;
2134
2135 if( (size_t)m_clPSDs.cols() != m_Nb.size() )
2136 {
2137 if( !logged[L] )
2138 {
2139 log<software_error>( { __FILE__, __LINE__, "Nb is not allocated" } );
2140 }
2141 logged[L] = true;
2142 doPCCalcs = false;
2143 }
2144 else
2145 {
2146 logged[L] = false;
2147 }
2148 ++L;
2149
2150 if( (size_t)m_clPSDs.cols() != m_NbCurrent.size() )
2151 {
2152 if( !logged[L] )
2153 {
2154 log<software_error>( { __FILE__, __LINE__, "NbCurrent is not allocated" } );
2155 }
2156 logged[L] = true;
2157 doPCCalcs = false;
2158 }
2159 else
2160 {
2161 logged[L] = false;
2162 }
2163 ++L;
2164
2165 if( m_clPSDs.cols() != m_bs.cols() )
2166 {
2167 if( !logged[L] )
2168 {
2169 log<software_error>( { __FILE__, __LINE__, "b coefficients are not allocated" } );
2170 }
2171 logged[L] = true;
2172 doPCCalcs = false;
2173 }
2174 else
2175 {
2176 logged[L] = false;
2177 }
2178 ++L;
2179
2180 if( m_clXferLPStream == nullptr || m_optGainLPStream == nullptr ||
2181 (size_t)m_clPSDs.cols() != m_optGainLP.size() || (size_t)m_clPSDs.cols() != m_goptLP.size() )
2182 {
2183 if( !logged[L] )
2184 {
2185 log<software_error>( { __FILE__, __LINE__, "LP is not allocated" } );
2186 }
2187 logged[L] = true;
2188 doPCCalcs = false;
2189 }
2190 logged[L++] = false;
2191
2193 {
2194 if( m_goptCurrent.size() != m_gainFacts.size() )
2195 {
2196 m_freqUpdated = true; // force freq update in this case
2197 }
2198
2199 std::cerr << "updating gopt structures\n";
2200
2201 m_goptCurrent.resize( m_gainFacts.size() );
2202 m_goptSI.resize( m_gainFacts.size() );
2203 m_goptLP.resize( m_gainFacts.size() );
2204 m_linPred.resize( m_gainFacts.size() );
2205
2206 for( size_t n = 0; n < m_goptCurrent.size(); ++n )
2207 {
2208 m_goptCurrent[n].Ti( 1.0 / m_fps );
2209 m_goptCurrent[n].tau( m_taus[n] );
2210
2211 m_goptSI[n].Ti( 1.0 / m_fps );
2212 m_goptSI[n].tau( m_taus[n] );
2213
2214 m_goptLP[n].Ti( 1.0 / m_fps );
2215 m_goptLP[n].tau( m_taus[n] );
2216
2217 if( !m_pcOn )
2218 {
2219 m_goptCurrent[n].setLeakyIntegrator( m_mult * m_multFacts[n] );
2220 }
2221 else
2222 {
2223 std::vector<float> ta( m_as.rows() );
2224 for( size_t m = 0; m < ta.size(); ++m )
2225 {
2226 ta[m] = m_as( m, n );
2227 }
2228 m_goptCurrent[n].a( ta );
2229
2230 std::vector<float> tb( m_bs.rows() );
2231 for( size_t m = 0; m < tb.size(); ++m )
2232 {
2233 tb[m] = m_bs( m, n );
2234 }
2235 m_goptCurrent[n].b( tb );
2236
2237 m_goptCurrent[n].remember( m_pcMult * m_pcMultFacts[n] );
2238 }
2239
2240 m_goptSI[n].setLeakyIntegrator( m_mult * m_multFacts[n] );
2241
2242 if( m_freqUpdated )
2243 {
2244 m_goptCurrent[n].f( m_freq );
2245 m_goptSI[n].f( m_freq );
2246 m_goptLP[n].f( m_freq );
2247 }
2248 }
2249
2250 m_goptUpdated = false;
2251 m_pcgoptUpdated = false;
2252 m_freqUpdated = false;
2253
2254 std::cerr << "done\n";
2255 }
2256
2257 if( m_updating )
2258 {
2259 continue;
2260 }
2261
2262 timePointT t0 = std::chrono::steady_clock::now();
2263
2264#pragma omp parallel for num_threads( 15 )
2265 for( size_t n = 0; n < m_goptCurrent.size(); ++n )
2266 {
2267 if( m_updating || m_shutdown )
2268 {
2269 continue; // don't break b/c of omp
2270 }
2271
2272 if( !m_pcOn )
2273 {
2274 for( size_t f = 0; f < m_goptCurrent[n].f_size(); ++f )
2275 {
2276 m_clXferCurrent( f, n ) = m_goptCurrent[n].clETF2( f, m_gain * m_gainFacts[n] * m_gainCals[n] );
2277 }
2278 }
2279 else
2280 {
2281 for( size_t f = 0; f < m_goptCurrent[n].f_size(); ++f )
2282 {
2283 m_clXferCurrent( f, n ) =
2284 m_goptCurrent[n].clETF2( f, m_pcGain * m_pcGainFacts[n] * m_gainCals[n] );
2285 }
2286 }
2287
2288 // Calculate the OL PSD with the current gopt (PC or SI)
2289 if( !m_loop )
2290 {
2291 for( size_t f = 1; f < m_goptCurrent[n].f_size(); ++f )
2292 {
2293 m_olPSDs[n][f] = m_clPSDs( f, n );
2294 m_nPSDs[n][f] = 1e-20;
2295 }
2296 }
2297 else
2298 {
2299 for( size_t f = 1; f < m_goptCurrent[n].f_size(); ++f )
2300 {
2301 m_olPSDs[n][f] = m_clPSDs( f, n ) / m_clXferCurrent( f, n );
2302 m_nPSDs[n][f] = 1e-20;
2303 }
2304 }
2305
2306 m_olPSDs[n][0] = m_olPSDs[n][1];
2307 m_nPSDs[n][0] = m_nPSDs[n][1];
2308
2309 m_optGainSI[n] = m_goptSI[n].optGainOpenLoop( m_modeVarSI[n], m_olPSDs[n], m_nPSDs[n], false );
2310
2311 for( size_t f = 0; f < m_goptCurrent[n].f_size(); ++f )
2312 {
2313 m_clXferSI( f, n ) = m_goptSI[n].clETF2( f, m_optGainSI[n] );
2314 }
2315
2316 if( doPCCalcs )
2317 {
2318
2319 if( m_regScale[n] == -999 || m_regCounter[n] >= m_nRegCycles )
2320 {
2321 float min_sc;
2322 float gmax_lp = 0;
2323
2324 m_linPred[n].regularizeCoefficients( gmax_lp,
2325 m_optGainLP[n],
2326 m_modeVarLP[n],
2327 min_sc,
2328 m_goptLP[n],
2329 m_olPSDs[n],
2330 m_nPSDs[n],
2331 m_Na[n] );
2332
2333 if( m_regScale[n] == -999 )
2334 {
2336 }
2337 else
2338 {
2339 m_regCounter[n] = 0;
2340 }
2341
2342 m_regScale[n] = min_sc;
2343 m_gmaxLP[n] = gmax_lp;
2344 // save max-stable gain m_maxLPGain[n]
2345 }
2346 else
2347 {
2348 // use new pre-regularized version
2349 float psdReg = m_olPSDs[n][0];
2351 m_olPSDs[n], m_nPSDs[n], psdReg * pow( 10, -m_regScale[n] / 10 ), m_Na[n] ) < 0 )
2352 {
2353 log<software_error>( { __FILE__, __LINE__, "error calculating coefficients" } );
2354 }
2355
2356 m_goptLP[n].a( m_linPred[n].m_lp.m_c );
2357 m_goptLP[n].b( m_linPred[n].m_lp.m_c );
2358
2359 m_optGainLP[n] =
2360 m_goptLP[n].optGainOpenLoop( m_modeVarLP[n], m_olPSDs[n], m_nPSDs[n], m_gmaxLP[n], false );
2361
2362 ++m_regCounter[n];
2363 }
2364
2365 for( size_t f = 0; f < m_goptCurrent[n].f_size(); ++f )
2366 {
2367 m_clXferLP( f, n ) = m_goptLP[n].clETF2( f, m_optGainLP[n] );
2368 }
2369 }
2370 else
2371 {
2372 for( size_t f = 0; f < m_goptCurrent[n].f_size(); ++f )
2373 {
2374 m_clXferLP( f, n ) = 1;
2375 }
2376 }
2377 }
2378
2379 timePointT t1 = std::chrono::steady_clock::now();
2380 durationT dt = t1 - t0;
2381 if( m_updating )
2382 {
2383 continue; // don't break b/c of omp
2384 }
2385
2386 std::cerr << "Optimization took " << dt.count() << " seconds\n";
2387
2388 float *f = m_optGainStream->array.F;
2389 float *fLP = m_optGainLPStream->array.F;
2390 float *fSI = m_optGainSIStream->array.F;
2391
2392 m_optGainStream->md->write = 1;
2393 m_optGainSIStream->md->write = 1;
2394 m_optGainLPStream->md->write = 1;
2395
2396 for( size_t n = 0; n < m_optGainSI.size(); ++n )
2397 {
2399 fSI[n] = f[n];
2401 }
2402
2403 clock_gettime( CLOCK_ISIO, &m_optGainStream->md->writetime );
2404 m_optGainStream->md->atime = m_optGainStream->md->writetime;
2405
2406 m_optGainSIStream->md->writetime = m_optGainStream->md->writetime;
2407 m_optGainSIStream->md->atime = m_optGainStream->md->writetime;
2408
2409 m_optGainLPStream->md->writetime = m_optGainStream->md->writetime;
2410 m_optGainLPStream->md->atime = m_optGainStream->md->writetime;
2411
2412 ++m_optGainStream->md->cnt0;
2413 m_optGainStream->md->write = 0;
2415
2416 ++m_optGainSIStream->md->cnt0;
2417 m_optGainSIStream->md->write = 0;
2419
2420 ++m_optGainLPStream->md->cnt0;
2421 m_optGainLPStream->md->write = 0;
2423
2424 if( m_autoUpdate || m_updateOnce || m_dump )
2425 {
2426 float *f = gainFactShmimMonitorT::m_imageStream.array.F;
2427
2429
2430 if( !m_loop || m_dump )
2431 {
2432 for( size_t n = 0; n < m_optGainSI.size(); ++n )
2433 {
2435 }
2436 }
2437 else
2438 {
2439 for( size_t n = 0; n < m_optGainSI.size(); ++n )
2440 {
2441 f[n] = f[n] + m_gainGain * ( m_gainCalFacts[n] * m_optGainSI[n] / m_gainCals[n] - f[n] );
2442 }
2443 }
2444
2450
2451 if( m_dump )
2452 {
2453 log<text_log>( "gains updated by dump", logPrio::LOG_NOTICE );
2454 m_dump = false;
2455 }
2456 else if( m_updateOnce && !m_autoUpdate )
2457 {
2458 log<text_log>( "gains updated once", logPrio::LOG_NOTICE );
2459 }
2460
2461 m_updateOnce = false;
2462 }
2463
2464 if( m_loop & m_autoUpdate )
2465 {
2466 m_sinceChange = -1;
2467 }
2468
2469 // Update Transfer Functions
2470 m_clXferCurrentStream->md->write = 1;
2471 m_clXferSIStream->md->write = 1;
2472 m_clXferLPStream->md->write = 1;
2473
2474 memcpy( m_clXferCurrentStream->array.F,
2475 m_clXferCurrent.data(),
2476 m_clXferCurrent.rows() * m_clXferCurrent.cols() );
2477 memcpy( m_clXferSIStream->array.F, m_clXferSI.data(), m_clXferSI.rows() * m_clXferSI.cols() );
2478 memcpy( m_clXferLPStream->array.F, m_clXferLP.data(), m_clXferLP.rows() * m_clXferLP.cols() );
2479
2481 m_clXferCurrentStream->md->atime = m_clXferCurrentStream->md->writetime;
2482
2483 clock_gettime( CLOCK_ISIO, &m_clXferSIStream->md->writetime );
2484 m_clXferSIStream->md->atime = m_clXferSIStream->md->writetime;
2485
2486 clock_gettime( CLOCK_ISIO, &m_clXferLPStream->md->writetime );
2487 m_clXferLPStream->md->atime = m_clXferLPStream->md->writetime;
2488
2489 ++m_clXferCurrentStream->md->cnt0;
2490 m_clXferCurrentStream->md->write = 0;
2492
2493 ++m_clXferSIStream->md->cnt0;
2494 m_clXferSIStream->md->write = 0;
2496
2497 ++m_clXferLPStream->md->cnt0;
2498 m_clXferLPStream->md->write = 0;
2500 }
2501 else
2502 {
2503 /* Check for why we timed out */
2504 /* ETIMEDOUT just means keep waiting */
2505 if( errno == ETIMEDOUT )
2506 {
2507 // Could Update gopts if needed (requires size checks and requires mutex lock)
2508 // Probably not worth it for pred. control anyway.
2509 continue;
2510 }
2511
2512 /* EINTER probably indicates time to shutdown, loop wil exit if m_shutdown is set */
2513 if( errno == EINTR )
2514 {
2515 continue;
2516 }
2517
2518 /*Otherwise, report an error.*/
2519 log<software_error>( { __FILE__, __LINE__, errno, "sem_timedwait" } );
2520 break;
2521 }
2522 }
2523}
2524
2525INDI_NEWCALLBACK_DEFN( modalGainOpt, m_indiP_autoUpdate )( const pcf::IndiProperty &ipRecv )
2526{
2527 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_autoUpdate, ipRecv );
2528
2529 if( ipRecv.find( "toggle" ) )
2530 {
2531 if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::On )
2532 {
2533 if( !m_autoUpdate )
2534 {
2535 log<text_log>( "updating gains", logPrio::LOG_NOTICE );
2536 }
2537 m_autoUpdate = true;
2538 }
2539 else
2540 {
2541 if( m_autoUpdate )
2542 {
2543 log<text_log>( "stopped updating gains", logPrio::LOG_NOTICE );
2544 }
2545 m_autoUpdate = false;
2546 }
2547 }
2548
2549 return 0;
2550}
2551
2552INDI_NEWCALLBACK_DEFN( modalGainOpt, m_indiP_updateOnce )( const pcf::IndiProperty &ipRecv )
2553{
2554 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_updateOnce, ipRecv );
2555
2556 if( ipRecv.find( "request" ) )
2557 {
2558 if( ipRecv["request"].getSwitchState() == pcf::IndiElement::On )
2559 {
2560 m_updateOnce = true;
2561 }
2562 else
2563 {
2564 m_updateOnce = false;
2565 }
2566 }
2567
2568 return 0;
2569}
2570
2571INDI_NEWCALLBACK_DEFN( modalGainOpt, m_indiP_dump )( const pcf::IndiProperty &ipRecv )
2572{
2573 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_dump, ipRecv );
2574
2575 if( ipRecv.find( "request" ) )
2576 {
2577 if( ipRecv["request"].getSwitchState() == pcf::IndiElement::On )
2578 {
2579 m_dump = true;
2580 }
2581 else
2582 {
2583 m_dump = false;
2584 }
2585 }
2586
2587 return 0;
2588}
2589
2590INDI_NEWCALLBACK_DEFN( modalGainOpt, m_indiP_gainGain )( const pcf::IndiProperty &ipRecv )
2591{
2592 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_gainGain, ipRecv );
2593
2594 float target;
2595 if( indiTargetUpdate( m_indiP_gainGain, target, ipRecv, true ) < 0 )
2596 {
2597 log<software_error>( { __FILE__, __LINE__ } );
2598 return -1;
2599 }
2600
2601 m_gainGain = target;
2602
2603 return 0;
2604}
2605
2606INDI_SETCALLBACK_DEFN( modalGainOpt, m_indiP_psdTime )( const pcf::IndiProperty &ipRecv )
2607{
2608 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_psdTime, ipRecv );
2609
2610 if( ipRecv.find( "current" ) )
2611 {
2612 float psdTime = ipRecv["current"].get<float>();
2613
2614 if( psdTime != m_psdTime )
2615 {
2616 m_updating = true;
2617 std::lock_guard<std::mutex> lock( m_goptMutex );
2618 m_updating = true;
2619
2620 m_psdTime = psdTime;
2621
2622 m_sinceChange = -1;
2623 m_updating = false;
2624
2625 std::cerr << "Got psdTime: " << m_psdTime << '\n';
2626 }
2627 }
2628 return 0;
2629}
2630
2631INDI_SETCALLBACK_DEFN( modalGainOpt, m_indiP_psdAvgTime )( const pcf::IndiProperty &ipRecv )
2632{
2633 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_psdAvgTime, ipRecv );
2634
2635 if( ipRecv.find( "current" ) )
2636 {
2637 float psdAvgTime = ipRecv["current"].get<float>();
2638
2639 if( psdAvgTime != m_psdAvgTime )
2640 {
2641 m_updating = true;
2642 std::lock_guard<std::mutex> lock( m_goptMutex );
2643 m_updating = true;
2644
2645 m_psdAvgTime = psdAvgTime;
2646
2647 m_sinceChange = -1;
2648 m_updating = false;
2649
2650 std::cerr << "Got psdAvgTime: " << m_psdAvgTime << '\n';
2651 }
2652 }
2653
2654 return 0;
2655}
2656
2657INDI_SETCALLBACK_DEFN( modalGainOpt, m_indiP_loop )( const pcf::IndiProperty &ipRecv )
2658{
2659 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_loop, ipRecv );
2660
2661 if( ipRecv.find( "toggle" ) )
2662 {
2663 bool state;
2664
2665 if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::On )
2666 {
2667 state = true;
2668 }
2669 else
2670 {
2671 state = false;
2672 }
2673
2674 if( state != m_loop )
2675 {
2676 m_updating = true;
2677 std::lock_guard<std::mutex> lock( m_goptMutex );
2678 m_updating = true;
2679
2680 m_loop = state;
2681
2682 m_sinceChange = -1;
2683 m_updating = false;
2684 std::cerr << "Got loop: " << m_loop << '\n';
2685 }
2686 }
2687
2688 return 0;
2689}
2690
2691INDI_SETCALLBACK_DEFN( modalGainOpt, m_indiP_siGain )( const pcf::IndiProperty &ipRecv )
2692{
2693 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_siGain, ipRecv );
2694
2695 if( ipRecv.find( "current" ) )
2696 {
2697 float gain = ipRecv["current"].get<float>();
2698
2699 if( gain != m_gain && !m_pcOn )
2700 {
2701 m_updating = true;
2702 std::lock_guard<std::mutex> lock( m_goptMutex );
2703 m_updating = true;
2704
2705 m_gain = gain;
2706
2707 if( m_loop )
2708 {
2709 m_sinceChange = -1;
2710 }
2711
2712 m_updating = false;
2713
2714 std::cerr << "Got gain: " << m_gain << '\n';
2715 }
2716 else
2717 {
2718 m_gain = gain; // for the m_pcOn case
2719 }
2720 }
2721
2722 return 0;
2723}
2724
2725INDI_SETCALLBACK_DEFN( modalGainOpt, m_indiP_siMult )( const pcf::IndiProperty &ipRecv )
2726{
2727 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_siMult, ipRecv );
2728
2729 if( ipRecv.find( "current" ) )
2730 {
2731 float mc = ipRecv["current"].get<float>();
2732
2733 if( mc != m_mult && !m_pcOn )
2734 {
2735 m_updating = true;
2736 std::lock_guard<std::mutex> lock( m_goptMutex );
2737 m_updating = true;
2738
2739 m_mult = mc;
2740
2741 if( m_loop )
2742 {
2743 m_sinceChange = -1;
2744 }
2745
2746 m_goptUpdated = true;
2747 m_updating = false;
2748 std::cerr << "Got mc: " << m_mult << '\n';
2749 }
2750 else
2751 {
2752 m_mult = mc; // for the m_pcOn case
2753 }
2754 }
2755
2756 return 0;
2757}
2758
2759INDI_SETCALLBACK_DEFN( modalGainOpt, m_indiP_pcGain )( const pcf::IndiProperty &ipRecv )
2760{
2761 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_pcGain, ipRecv );
2762
2763 if( ipRecv.find( "current" ) )
2764 {
2765 float gain = ipRecv["current"].get<float>();
2766
2767 if( gain != m_pcGain && m_pcOn )
2768 {
2769 m_updating = true;
2770 std::lock_guard<std::mutex> lock( m_goptMutex );
2771 m_updating = true;
2772
2773 m_pcGain = gain;
2774
2775 if( m_loop )
2776 {
2777 m_sinceChange = -1;
2778 }
2779
2780 m_updating = false;
2781
2782 std::cerr << "Got pc gain: " << m_pcGain << '\n';
2783 }
2784 else
2785 {
2786 m_pcGain = gain; // for the !m_pcOn case
2787 }
2788 }
2789
2790 return 0;
2791}
2792
2793INDI_SETCALLBACK_DEFN( modalGainOpt, m_indiP_pcMult )( const pcf::IndiProperty &ipRecv )
2794{
2795 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_pcMult, ipRecv );
2796
2797 if( ipRecv.find( "current" ) )
2798 {
2799 float mc = ipRecv["current"].get<float>();
2800
2801 if( mc != m_pcMult && m_pcOn )
2802 {
2803 m_updating = true;
2804 std::lock_guard<std::mutex> lock( m_goptMutex );
2805 m_updating = true;
2806
2807 m_pcMult = mc;
2808
2809 if( m_loop )
2810 {
2811 m_sinceChange = -1;
2812 }
2813
2814 m_pcgoptUpdated = true;
2815 m_updating = false;
2816 std::cerr << "Got pc mc: " << m_pcMult << '\n';
2817 }
2818 else
2819 {
2820 m_pcMult = mc; // for the !m_pcOn case
2821 }
2822 }
2823
2824 return 0;
2825}
2826
2827INDI_SETCALLBACK_DEFN( modalGainOpt, m_indiP_pcOn )( const pcf::IndiProperty &ipRecv )
2828{
2829 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_pcOn, ipRecv );
2830
2831 if( ipRecv.find( "toggle" ) )
2832 {
2833 bool state;
2834
2835 if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::On )
2836 {
2837 state = true;
2838 }
2839 else
2840 {
2841 state = false;
2842 }
2843
2844 if( state != m_pcOn )
2845 {
2846 m_updating = true;
2847 std::lock_guard<std::mutex> lock( m_goptMutex );
2848 m_updating = true;
2849
2850 m_pcOn = state;
2851
2852 m_sinceChange = -1;
2853 m_updating = false;
2854 std::cerr << "Got pcOn: " << std::boolalpha << m_pcOn << '\n';
2855 }
2856 }
2857
2858 return 0;
2859}
2860
2861} // namespace app
2862} // namespace MagAOX
2863
2864#endif // modalGainOpt_hpp
#define XWCAPP_THREAD_CHECK(thrdSt, thrdName)
Error handling wrapper for checking on thread status with tryjoin.
#define XWCAPP_THREAD_START(thrdSt, thrdInit, thrdId, thrdProp, thrdPrio, thrdCpuset, thrdName, thrdStart)
Error handling wrapper for the threadStart function of the XWCApp.
#define XWCAPP_THREAD_STOP(thrdSt)
Error handlng wrapper for stopping a thread.
The base-class for MagAO-X applications.
Definition MagAOXApp.hpp:73
stateCodes::stateCodeT state()
Get the current state code.
int m_shutdown
Flag to signal it's time to shutdown. When not 0, the main loop exits.
int shutdown()
Get the value of the shutdown flag.
void updateSwitchIfChanged(pcf::IndiProperty &p, const std::string &el, const pcf::IndiElement::SwitchStateType &newVal, pcf::IndiProperty::PropertyStateType ipState=pcf::IndiProperty::Ok)
Update an INDI switch element value if it has changed.
static int log(const typename logT::messageT &msg, logPrioT level=logPrio::LOG_DEFAULT)
Make a log entry.
uint32_t m_width
The width of the images in the stream.
uint32_t m_height
The height of the images in the stream.
IMAGE m_imageStream
The ImageStreamIO shared memory buffer.
std::string m_shmimName
The name of the shared memory image, is used in /tmp/<shmimName>.im.shm. Derived classes should set a...
bool m_getExistingFirst
If set to true by derivedT, any existing image will be grabbed and sent to processImage before waitin...
The MagAO-X PSD-based gain optimizer.
std::vector< mx::AO::analysis::clGainOpt< float > > m_goptCurrent
Each mode gets its own gain optimizer.
pcf::IndiProperty m_indiP_gainGain
INDI_NEWCALLBACK_DECL(modalGainOpt, m_indiP_updateOnce)
dev::shmimMonitor< modalGainOpt, numpccoeffShmimT > numpccoeffShmimMonitorT
mx::improc::eigenImage< float > m_clXferCurrent
~modalGainOpt() noexcept
D'tor, declared and defined for noexcept.
IMAGE * m_clXferLPStream
The ImageStreamIO shared memory buffer to publish the LP ETF.
std::vector< int > m_regCounter
Counters to track when this mode was last regularized.
std::vector< float > m_gmaxLP
The previously calculated maximum gains for LP.
std::vector< float > m_pcGainFacts
std::vector< mx::AO::analysis::clGainOpt< float > > m_goptLP
std::chrono::time_point< std::chrono::steady_clock > timePointT
eigenImage< float > m_as
dev::shmimMonitor< modalGainOpt, pcGainFactShmimT > pcGainFactShmimMonitorT
pid_t m_goptThreadID
gain optimization thread PID.
INDI_NEWCALLBACK_DECL(modalGainOpt, m_indiP_autoUpdate)
dev::shmimMonitor< modalGainOpt, pcMultFactShmimT > pcMultFactShmimMonitorT
pcf::IndiProperty m_indiP_fps
std::vector< float > m_taus
dev::shmimMonitor< modalGainOpt, gainCalShmimT > gainCalShmimMonitorT
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_fps)
int allocate(const psdShmimT &)
INDI_NEWCALLBACK_DECL(modalGainOpt, m_indiP_dump)
std::vector< float > m_freq
pcf::IndiProperty m_indiP_siGain
virtual int appLogic()
Implementation of the FSM for modalGainOpt.
std::vector< float > m_modeVarSI
pcf::IndiProperty m_indiP_autoUpdate
int m_loopNum
The number of the loop. Used to set shmim names, as in aolN_mgainfact.
std::vector< float > m_gainCalFacts
pcf::IndiProperty m_indiP_dump
std::vector< float > m_gainFacts
dev::shmimMonitor< modalGainOpt, bcoeffShmimT > bcoeffShmimMonitorT
int m_goptThreadPrio
Priority of the gain optimization thread.
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_pcOn)
std::vector< float > m_optGainLP
sem_t m_goptSemaphore
Semaphore used to synchronize the psdShmim thread and the gopt thread.
IMAGE * m_optGainLPStream
The ImageStreamIO shared memory buffer to publish the LP optimal gains.
std::string m_goptThreadCpuset
The cpuset to use for the gain optimization thread.
pcf::IndiProperty m_indiP_loop
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_siGain)
mx::improc::eigenImage< float > m_clPSDs
pcf::IndiProperty m_indiP_siMult
pcf::IndiProperty m_indiP_pcOn
IMAGE * m_olPSDStream
The ImageStreamIO shared memory buffer to publish the open loop PSDs.
IMAGE * m_clXferSIStream
The ImageStreamIO shared memory buffer to publish the SI ETF.
std::vector< std::vector< float > > m_olPSDs
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_loop)
bool m_goptUpdated
Tracks if a parameter has updated requiring updates to the m_gopt entries.
mx::improc::eigenImage< float > m_clXferLP
dev::shmimMonitor< modalGainOpt, freqShmimT > freqShmimMonitorT
std::string m_loopName
The name of the loop control INDI device name.
std::chrono::duration< double > durationT
dev::shmimMonitor< modalGainOpt, gainFactShmimT > gainFactShmimMonitorT
std::vector< int > m_NbCurrent
bool m_updateOnce
Flag to trigger a single update with gain.
std::vector< int > m_Nb
pcf::IndiProperty m_indiP_updateOnce
dev::shmimMonitor< modalGainOpt, tauShmimT > tauShmimMonitorT
std::vector< mx::AO::analysis::clAOLinearPredictor< float > > m_linPred
pcf::IndiProperty m_goptThreadProp
The property to hold the gain optimization thread details.
static void goptThreadStart(modalGainOpt *p)
Gain Optimization thread starter function.
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_psdAvgTime)
eigenImage< float > m_bs
bool m_dump
Flag to trigger a single update with no gain.
mx::improc::eigenImage< float > m_clXferSI
pcf::IndiProperty m_indiP_psdAvgTime
std::vector< float > m_modeVarLP
std::vector< int > m_Na
IMAGE * m_optGainStream
The ImageStreamIO shared memory buffer to publish the optimal gains.
std::vector< float > m_optGainSI
IMAGE * m_optGainSIStream
The ImageStreamIO shared memory buffer to publish the SI optimal gains.
IMAGE * m_clXferCurrentStream
The ImageStreamIO shared memory buffer to publish the SI ETF.
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_psdTime)
int loadConfigImpl(mx::app::appConfigurator &_config)
Implementation of loadConfig logic, separated for testing.
std::vector< float > m_gainCals
virtual int appStartup()
Startup function.
std::thread m_goptThread
The gain optimization thread.
void goptThreadExec()
Gain optimization thread function.
dev::shmimMonitor< modalGainOpt, psdShmimT > psdShmimMonitorT
bool m_updating
Flag used to indicate to the goptThread that it should stop calculations ASAP.
virtual int appShutdown()
Shutdown the app.
bool m_goptThreadInit
Initialization flag for the gain optimization thread.
int processImage(void *curr_src, const psdShmimT &)
std::vector< float > m_pcMultFacts
std::vector< int > m_NaCurrent
std::mutex m_goptMutex
Mutex for synchronizing updates.
dev::shmimMonitor< modalGainOpt, multFactShmimT > multFactShmimMonitorT
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_siMult)
bool m_autoUpdate
Flag controlling whether gains are automatically updated.
dev::shmimMonitor< modalGainOpt, gainCalFactShmimT > gainCalFactShmimMonitorT
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_pcMult)
std::vector< float > m_multFacts
dev::shmimMonitor< modalGainOpt, acoeffShmimT > acoeffShmimMonitorT
std::vector< float > m_regScale
The regularizatio scale factors for each mode.
bool m_pcgoptUpdated
Tracks if a parameter has updated requiring updates to the m_gopt entries.
std::vector< std::vector< float > > m_nPSDs
float m_gainGain
The gain to use for closed-loop gain updates. Default is 0.1.
pcf::IndiProperty m_indiP_pcGain
std::vector< mx::AO::analysis::clGainOpt< float > > m_goptSI
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_pcGain)
INDI_NEWCALLBACK_DECL(modalGainOpt, m_indiP_gainGain)
pcf::IndiProperty m_indiP_psdTime
int m_nRegCycles
How often to regularize each mode.
pcf::IndiProperty m_indiP_pcMult
#define INDI_NEWCALLBACK_DEFN(class, prop)
Define the callback for a new property request.
#define CREATE_REG_INDI_NEW_TOGGLESWITCH(prop, name)
Create and register a NEW INDI property as a standard toggle switch, using the standard callback name...
#define CREATE_REG_INDI_NEW_REQUESTSWITCH(prop, name)
Create and register a NEW INDI property as a standard request switch, using the standard callback nam...
#define CREATE_REG_INDI_NEW_NUMBERF(prop, name, min, max, step, format, label, group)
Create and register a NEW INDI property as a standard number as float, using the standard callback na...
#define INDI_SETCALLBACK_DEFN(class, prop)
Define the callback for a set property request.
#define REG_INDI_SETPROP(prop, devName, propName)
Register a SET INDI property with the class, using the standard callback name.
@ OPERATING
The device is operating, other than homing.
#define INDI_VALIDATE_CALLBACK_PROPS(prop1, prop2)
Standard check for matching INDI properties in a callback.
#define INDI_IDLE
Definition indiUtils.hpp:27
#define INDI_OK
Definition indiUtils.hpp:28
const pcf::IndiProperty & ipRecv
std::unique_lock< std::mutex > lock(m_indiMutex)
Definition dm.hpp:24
static constexpr logPrioT LOG_NOTICE
A normal but significant condition.
#define XWC_SEM_WAIT_TS_RETVOID(ts, sec, nsec)
Add the wait time to a timespec for a sem_timedwait call, with no value returned on error.
Definition semUtils.hpp:20
#define SHMIMMONITORT_APP_STARTUP(SHMIMMONITORT)
Call shmimMonitorT::appStartup with error checking for a typedef-ed shmimMonitor.
#define SHMIMMONITORT_UPDATE_INDI(SHMIMMONITORT)
Call shmimMonitorT::updateINDI with error checking for a typedef-ed shmimMonitor.
#define SHMIMMONITORT_SETUP_CONFIG(SHMIMMONITORT, cfig)
Call shmimMonitorT::setupConfig with error checking for a typedef-ed shmimMonitor.
#define SHMIMMONITORT_APP_LOGIC(SHMIMMONITORT)
Call shmimMonitorT::appLogic with error checking for a typedef-ed shmimMonitor.
#define SHMIMMONITORT_APP_SHUTDOWN(SHMIMMONITORT)
Call shmimMonitorT::appShutodwn with error checking for a typedef-ed shmimMonitor.
#define SHMIMMONITORT_LOAD_CONFIG(SHMIMMONITORT, cfig)
Call shmimMonitorT::loadConfig with error checking for a typedef-ed shmimMonitor.
static std::string configSection()
static std::string indiPrefix()
static std::string indiPrefix()
static std::string configSection()
static std::string indiPrefix()
static std::string configSection()
static std::string configSection()
static std::string indiPrefix()
static std::string indiPrefix()
static std::string configSection()
static std::string configSection()
static std::string indiPrefix()
static std::string configSection()
static std::string indiPrefix()
static std::string indiPrefix()
static std::string configSection()
static std::string configSection()
static std::string indiPrefix()
static std::string indiPrefix()
static std::string configSection()
static std::string configSection()
static std::string indiPrefix()
static std::string indiPrefix()
static std::string configSection()
Software CRITICAL log entry.
Software ERR log entry.