API
 
Loading...
Searching...
No Matches
cacaoInterface.hpp
Go to the documentation of this file.
1/** \file cacaoInterface.hpp
2 * \brief The MagAO-X CACAO Interface header file
3 *
4 * \ingroup cacaoInterface_files
5 */
6
7#ifndef cacaoInterface_hpp
8#define cacaoInterface_hpp
9
10#include "../../libMagAOX/libMagAOX.hpp" //Note this is included on command line to trigger pch
11#include "../../magaox_git_version.h"
12
13/** \defgroup cacaoInterface
14 * \brief The CACAO Interface to provide loop status
15 *
16 * <a href="../handbook/apps/cacaoInterface.html">Application Documentation</a>
17 *
18 * \ingroup apps
19 *
20 */
21
22/** \defgroup cacaoInterface_files
23 * \ingroup cacaoInterface
24 */
25
26namespace MagAOX
27{
28namespace app
29{
30
31/// The MagAO-X CACAO Interface
32/**
33 * \ingroup cacaoInterface
34 */
35class cacaoInterface : public MagAOXApp<true>, public dev::telemeter<cacaoInterface>
36{
37
38 //Give the test harness access.
39 friend class cacaoInterface_test;
40
42
43 friend class dev::telemeter<cacaoInterface>;
44
45protected:
46
47 /** \name Configurable Parameters
48 *@{
49 */
50 std::string m_loopNumber; ///< The loop number, X in aolX. We keep it a string because that's how it gets used.
51
52 ///@}
53
54 std::string m_aoCalDir;
55 std::string m_aoCalArchiveTime;
56 std::string m_aoCalLoadTime;
57
58 std::string m_loopName; ///< the loop name
59
60
61 std::string m_fpsName;
62 std::string m_fpsFifo;
63
64
65 int m_loopState {0}; ///< The loop state. 0 = off, 1 = paused (on, 0 gain), 2 = on
66
67 bool m_loopProcesses {false}; ///< Status of the loop processes.
68 bool m_loopProcesses_stat {false}; ///< What the cacao status file says the state of loop processes is.
69
70 float m_gain {0.0}; ///< The current loop gain.
71 float m_gain_target {0.0}; ///< The target loop gain.
72
73 float m_multCoeff {0.0}; ///< The current multiplicative coefficient (1-leak)
74 float m_multCoeff_target {0.0}; ///< The target multiplicative coefficient (1-leak)
75
76 std::vector<int> m_modeBlockStart;
77 std::vector<int> m_modeBlockN;
78
79 std::vector<float> m_modeBlockGains;
80 std::vector<float> m_modeBlockMCs;
81 std::vector<float> m_modeBlockLims;
82
83 std::mutex m_modeBlockMutex;
84
85
86 float m_maxLim {0.0}; ///< The current max limit
87 float m_maxLim_target {0.0}; ///< The target max limit
88
89public:
90 /// Default c'tor.
92
93 /// D'tor, declared and defined for noexcept.
96
97 virtual void setupConfig();
98
99 /// Implementation of loadConfig logic, separated for testing.
100 /** This is called by loadConfig().
101 */
102 int loadConfigImpl( mx::app::appConfigurator & _config /**< [in] an application configuration from which to load values*/);
103
104 virtual void loadConfig();
105
106 /// Startup function
107 /**
108 *
109 */
110 virtual int appStartup();
111
112 /// Implementation of the FSM for cacaoInterface.
113 /**
114 * \returns 0 on no critical error
115 * \returns -1 on an error requiring shutdown
116 */
117 virtual int appLogic();
118
119 /// Shutdown the app.
120 /**
121 *
122 */
123 virtual int appShutdown();
124
125
126 /** \name CACAO Interface Functions
127 * @{
128 */
129
130 int setFPSVal( const std::string & fps,
131 const std::string & param,
132 const std::string & val
133 );
134
135
136 template<typename T>
137 int setFPSVal( const std::string & fps,
138 const std::string & param,
139 const T & val
140 );
141
142 std::string getFPSValStr(const std::string & fps,
143 const std::string & param );
144
145 std::string getFPSValNum(const std::string & fps,
146 const std::string & param );
147
148 /// Get the calibration details
149 /** This is done each loop
150 *
151 * \returns 0 on success
152 * \returns -1 on an error
153 */
154 int getAOCalib();
155
157
158 /// Check if the loop processes are running
159 /** sets m_loopProcesses to true or false depending on what it finds out.
160 *
161 * \returns 0 on success
162 * \returns -1 on an error
163 */
164 int checkLoopProcesses();
165
166 /// Set loop gain to the value of m_gain_target;
167 /**
168 *
169 * \returns 0 on success
170 * \returns -1 on an error
171 */
172 int setGain();
173
174 /// Set loop multiplication coefficient to the value of m_multCoeff_target;
175 /**
176 *
177 * \returns 0 on success
178 * \returns -1 on an error
179 */
180 int setMultCoeff();
181
182 /// Set loop max lim to the value of m_maxLim_target;
183 /**
184 *
185 * \returns 0 on success
186 * \returns -1 on an error
187 */
188 int setMaxLim();
189
190 /// Turn the loop on
191 /**
192 * \returns 0 on success
193 * \returns -1 on an error
194 */
195 int loopOn();
196
197 /// Turn the loop off
198 /**
199 * \returns 0 on success
200 * \returns -1 on an error
201 */
202 int loopOff();
203
204 /// Zero the loop control channel
205 /**
206 * \returns 0 on success
207 * \returns -1 on an error
208 */
209 int loopZero();
210
211 /// @}
212
213 /** \name File Monitoring Thread
214 * Handling of offloads from the average woofer shape
215 * @{
216 */
217 int m_fmThreadPrio {0}; ///< Priority of the filemonitoring thread.
218
219 std::thread m_fmThread; ///< The file monitoring thread.
220
221 bool m_fmThreadInit {true}; ///< Initialization flag for the file monitoring thread.
222
223 pid_t m_fmThreadID {0}; ///< File monitor thread PID.
224
225 pcf::IndiProperty m_fmThreadProp; ///< The property to hold the f.m. thread details.
226
227 /// File monitoring thread starter function
228 static void fmThreadStart( cacaoInterface * c /**< [in] pointer to this */);
229
230
231 /// File monitoring thread function
232 /** Runs until m_shutdown is true.
233 */
234 void fmThreadExec();
235
236
237 ///@}
238
239 pcf::IndiProperty m_indiP_loop;
240 pcf::IndiProperty m_indiP_loopProcesses;
241
242 pcf::IndiProperty m_indiP_loopState;
243 pcf::IndiProperty m_indiP_loopZero;
244 pcf::IndiProperty m_indiP_loopGain;
245 pcf::IndiProperty m_indiP_multCoeff;
246 pcf::IndiProperty m_indiP_maxLim;
247
253
254 /** \name Telemeter Interface
255 *
256 * @{
257 */
258 int checkRecordTimes();
259
260 int recordTelem( const telem_loopgain * );
261
262 int recordLoopGain( bool force = false );
263
264 ///@}
265
266
267};
268
269cacaoInterface::cacaoInterface() : MagAOXApp(MAGAOX_CURRENT_SHA1, MAGAOX_REPO_MODIFIED)
270{
271
272 return;
273}
274
276{
277 config.add("loop.number", "", "loop.number", argType::Required, "loop", "number", false, "string", "the loop number");
278
280}
281
282int cacaoInterface::loadConfigImpl( mx::app::appConfigurator & _config )
283{
284 _config(m_loopNumber, "loop.number");
285
287 {
288 log<text_log>("Error during telemeter config", logPrio::LOG_CRITICAL);
289 m_shutdown = true;
290 }
291
292 return 0;
293}
294
296{
297 loadConfigImpl(config);
298}
299
301{
302
303 if(m_loopNumber == "")
304 {
305 return log<software_critical, -1>({__FILE__, __LINE__, "loop number not set"});
306 }
307
308 createROIndiText( m_indiP_loop, "loop", "name", "Loop Description", "Loop Controls", "Name");
309 indi::addTextElement(m_indiP_loop, "number", "Number");
310 m_indiP_loop["number"] = m_loopNumber;
311 m_indiP_loop["name"] = m_loopName;
313
314 createStandardIndiToggleSw( m_indiP_loopProcesses, "loop_processes", "Loop Processes", "Loop Controls");
316
317 createStandardIndiToggleSw( m_indiP_loopState, "loop_state", "Loop State", "Loop Controls");
319
320 createStandardIndiRequestSw( m_indiP_loopZero, "loop_zero", "Loop Zero", "Loop Controls");
322
323 createStandardIndiNumber<float>( m_indiP_loopGain, "loop_gain", 0.0, 10.0, 0.01, "%0.3f", "Loop Gain", "Loop Controls");
325
326 createStandardIndiNumber<float>( m_indiP_multCoeff, "loop_multcoeff", 0.0, 1.0, 0.001, "%0.3f", "Mult. Coefficient", "Loop Controls");
328
329 createStandardIndiNumber<float>( m_indiP_maxLim, "loop_max_limit", 0.0, 10.0, 0.001, "%0.3f", "Max. Limit", "Loop Controls");
331
333 {
335 return -1;
336 }
337
338 if(telemeterT::appStartup() < 0)
339 {
340 return log<software_error,-1>({__FILE__,__LINE__});
341 }
342
343 return 0;
344}
345
347{
348 //do a join check to see if other threads have exited.
349 if(pthread_tryjoin_np(m_fmThread.native_handle(),0) == 0)
350 {
351 log<software_critical>({__FILE__, __LINE__, "cacao file monitoring thread has exited"});
352
353 return -1;
354 }
355
356 //These could change if a new calibration is loaded
357 if(getAOCalib() < 0 )
358 {
360 if(!stateLogged()) log<text_log>("Could not get AO calib", logPrio::LOG_ERROR);
361 return 0;
362 }
363
366
367 if(telemeterT::appLogic() < 0)
368 {
370 return 0;
371 }
372
373 std::unique_lock<std::mutex> lock(m_indiMutex);
374
375 updateIfChanged(m_indiP_loop, std::vector<std::string>({"name", "number"}), std::vector<std::string>({m_loopName, m_loopNumber}));
376
378 {
379 updateSwitchIfChanged(m_indiP_loopProcesses, "toggle", pcf::IndiElement::On, INDI_OK);
380 }
381 else
382 {
383 updateSwitchIfChanged(m_indiP_loopProcesses, "toggle", pcf::IndiElement::Off, INDI_IDLE);
384 }
385
386 if(m_loopState == 0)
387 {
388 updateSwitchIfChanged(m_indiP_loopState, "toggle", pcf::IndiElement::Off, INDI_IDLE);
389 }
390 else if(m_loopState == 1)
391 {
392 updateSwitchIfChanged(m_indiP_loopState, "toggle", pcf::IndiElement::On, INDI_IDLE);
393 }
394 else if(m_loopState == 2)
395 {
396 updateSwitchIfChanged(m_indiP_loopState, "toggle", pcf::IndiElement::On, INDI_OK);
397 }
398
401
405
406 return 0;
407}
408
410{
411
412 if(m_fmThread.joinable())
413 {
414 try
415 {
416 m_fmThread.join(); //this will throw if it was already joined
417 }
418 catch(...)
419 {
420 }
421 }
422
424
425 return 0;
426}
427
428int cacaoInterface::setFPSVal( const std::string & fps,
429 const std::string & param,
430 const std::string & val
431 )
432{
433 std::string comout = "setval " + fps + "-" + m_loopNumber + "." + param + " " + val + "\n";
434
435 int wfd = open( m_fpsFifo.c_str(), O_WRONLY);
436 if(wfd < 0)
437 {
438 log<software_error>({__FILE__, __LINE__, errno, "error opening " + m_fpsFifo});
439 return -1;
440 }
441
442 int w = write(wfd, comout.c_str(), comout.size());
443
444 if(w != (int) comout.size())
445 {
446 log<software_error>({__FILE__, __LINE__, errno, "error on write to " + m_fpsFifo});
447 return -1;
448 }
449
450 close(wfd);
451
452 return 0;
453}
454
455
456template<typename T>
457int cacaoInterface::setFPSVal( const std::string & fps,
458 const std::string & param,
459 const T & val
460 )
461{
462 return setFPSVal( fps, param, std::to_string(val));
463}
464
465std::string cacaoInterface::getFPSValStr( const std::string & fps,
466 const std::string & param
467 )
468{
469 std::string outfile = "/dev/shm/" + m_loopName + "_out_" + fps + "-" + m_loopNumber + "." + param;
470
471 std::string comout = "fwrval " + fps + "-" + m_loopNumber + "." + param + " " + outfile + "\n";
472
473 int wfd = open( m_fpsFifo.c_str(), O_WRONLY);
474 if(wfd < 0)
475 {
476 log<software_error>({__FILE__, __LINE__, errno, "error opening " + m_fpsFifo});
477 return "";
478 }
479
480 int w = write(wfd, comout.c_str(), comout.size());
481
482 if(w != (int) comout.size())
483 {
484 log<software_error>({__FILE__, __LINE__, errno, "error on write to " + m_fpsFifo});
485 return "";
486 }
487
488 close(wfd);
489
490 char inbuff [4096];
491
492 int rfd = -1;
493 int nr =0;
494 while(rfd < 0 && nr < 500)
495 {
496 rfd = open(outfile.c_str(), O_RDONLY);
497 ++nr;
498 mx::sys::milliSleep(10);
499 }
500
501 if(rfd < 0)
502 {
503 log<software_error>({__FILE__, __LINE__, "could not get an open fd on " + m_fpsFifo});
504 return "";
505 }
506
507 int r = read(rfd, inbuff, sizeof(inbuff));
508
509 if(r < 0)
510 {
511 log<software_error>({__FILE__, __LINE__, errno, "error on read from " + m_fpsFifo});
512 return "";
513 }
514
515 close(rfd);
516
517 remove(outfile.c_str());
518
519 inbuff[r] = '\0';
520
521 std::string instr = inbuff;
522
523 size_t ned = instr.find_last_not_of(" \t\r\n");
524
525 if(ned == std::string::npos || ned == 0)
526 {
527 //log<software_error>({__FILE__, __LINE__, "got empty result from " + m_fpsFifo});
528 return "";
529 }
530
531 size_t nst = instr.rfind(' ', ned);
532
533 if(nst == 0 || nst == std::string::npos || ned <= nst)
534 {
535 log<software_error>({__FILE__, __LINE__, "bad format in result from " + m_fpsFifo});
536 return "";
537 }
538
539 return instr.substr(nst+1);
540}
541
542std::string cacaoInterface::getFPSValNum( const std::string & fps,
543 const std::string & param
544 )
545{
546 std::string outfile = "/dev/shm/" + m_loopName + "_out_" + fps + "-" + m_loopNumber + "." + param;
547
548 std::string comout = "fwrval " + fps + "-" + m_loopNumber + "." + param + " " + outfile + "\n";
549
550 int wfd = open( m_fpsFifo.c_str(), O_WRONLY);
551 if(wfd < 0)
552 {
553 log<software_error>({__FILE__, __LINE__, errno, "error opening " + m_fpsFifo});
554 return "";
555 }
556
557 int w = write(wfd, comout.c_str(), comout.size());
558
559 if(w != (int) comout.size())
560 {
561 log<software_error>({__FILE__, __LINE__, errno, "error on write to " + m_fpsFifo});
562 return "";
563 }
564 close(wfd);
565
566 char inbuff [4096];
567
568 int rfd = -1;
569 int nr =0;
570 while(rfd < 0 && nr < 500)
571 {
572 rfd = open(outfile.c_str(), O_RDONLY);
573 ++nr;
574 mx::sys::milliSleep(10);
575 }
576
577 if(rfd < 0)
578 {
579 log<software_error>({__FILE__, __LINE__, "could not get an open fd on " + m_fpsFifo});
580 return "";
581 }
582
583 int r = read(rfd, inbuff, sizeof(inbuff));
584
585 close(rfd);
586
587 if(r < 0)
588 {
589 log<software_error>({__FILE__, __LINE__, errno, "error on read from " + m_fpsFifo});
590 return "";
591 }
592
593 remove(outfile.c_str());
594
595 inbuff[r] = '\0';
596
597 std::string instr = inbuff;
598
599 size_t ned = instr.find_last_not_of(" \t\r\n");
600
601 if(ned == std::string::npos || ned == 0)
602 {
603 log<software_error>({__FILE__, __LINE__, "got empty result from " + m_fpsFifo});
604 return "";
605 }
606
607 size_t nst = instr.rfind(' ', ned);
608
609 if(nst == 0 || nst == std::string::npos || ned <= nst)
610 {
611 log<software_error>({__FILE__, __LINE__, "bad format in result from " + m_fpsFifo});
612 return "";
613 }
614
615 return instr.substr(nst);
616}
617
618inline
620{
621 std::string calsrc = "/milk/shm/aol" + m_loopNumber + "_calib_source.txt";
622
623 std::ifstream fin;
624
625 //First read in the milk/shm directory name, which could be to a symlinked directory
626 fin.open(calsrc);
627 if(!fin)
628 {
629 return 0;
630 }
631 fin >> calsrc;
632 fin.close();
633
634 //Now read in the actual directory
635 calsrc += "/calib_dir.txt";
636 fin.open(calsrc);
637 if(!fin)
638 {
639 return log<software_error, -1>({__FILE__, __LINE__, errno, "cacaoInterface::getAOCalib failed to open: " + calsrc});
640 }
641 std::string aoCalDir;
642
643 fin >> aoCalDir;
644
645 fin.close();
646
647 bool newcal = false;
648 if(aoCalDir != m_aoCalDir)
649 {
651 newcal = true;
652 }
653
654 std::string nameFile = m_aoCalDir + "/LOOPNAME";
655 fin.open(nameFile);
656 if(!fin)
657 {
658 return log<software_error, -1>({__FILE__, __LINE__, errno, "cacaoInterface::getAOCalib failed to open: " + nameFile});
659 }
660 fin >> m_loopName;
661 fin.close();
662
663 m_fpsFifo = "/milk/shm/" + m_loopName + "_fpsCTRL.fifo";
664
665 calsrc = "/milk/shm/aol" + m_loopNumber + "_calib_loaded.txt";
666 fin.open(calsrc);
667 if(!fin)
668 {
669 return log<software_error, -1>({__FILE__, __LINE__, errno, "cacaoInterface::getAOCalib failed to open: " + calsrc});
670 }
671 std::string aoCalLoadTime;
673
675 {
676 newcal = true;
678 }
679
680 if(newcal)
681 {
682 log<text_log>("new calibration " + m_aoCalDir + " loaded at: " + m_aoCalLoadTime, logPrio::LOG_INFO);
683 }
684
685 fin.close();
686
687 calsrc = m_aoCalDir + "/calib_archived.txt";
688
689 fin.open(calsrc);
690 if(!fin)
691 {
692 return log<software_error, -1>({__FILE__, __LINE__, errno, "cacaoInterface::getAOCalib failed to open: " + calsrc});
693 }
694
695 while(!fin.eof())
696 {
698 }
699 fin.close();
700
701
702 return 0;
703}
704
706{
707 ///\todo look for actual evidence of processes, such as interrogating ps.
708
710
711 return 0;
712}
713
715{
716 recordLoopGain(true);
717 return setFPSVal("mfilt", "loopgain", m_gain_target);
718}
719
721{
722 recordLoopGain(true);
723 return setFPSVal("mfilt", "loopmult", m_multCoeff_target);
724}
725
727{
728 recordLoopGain(true);
729 return setFPSVal("mfilt", "looplimit", m_maxLim_target);
730}
731
733{
734 recordLoopGain(true);
735 if( setFPSVal("mfilt", "loopON", std::string("ON")) != 0)
736 {
737 return log<software_error,-1>({__FILE__, __LINE__, "error setting FPS val"});
738 }
739
741
742 return 0;
743
744}
745
747{
748 recordLoopGain(true);
749 if( setFPSVal("mfilt", "loopON", std::string("OFF")) != 0)
750 {
751 return log<software_error,-1>({__FILE__, __LINE__, "error setting FPS val"});
752 }
753
754 if(m_gain == 0)
755 {
757 }
758 else
759 {
761 }
762
763 return 0;
764
765}
766
768{
769 if( setFPSVal("mfilt", "loopZERO", std::string("ON")) != 0)
770 {
771 return log<software_error,-1>({__FILE__, __LINE__, "error setting FPS val"});
772 }
773
774 log<text_log>("loop zeroed", logPrio::LOG_NOTICE);
775
776 return 0;
777
778}
779
781{
782 c->fmThreadExec();
783}
784
785
787{
789
790 while( m_fmThreadInit == true && shutdown() == 0)
791 {
792 sleep(1);
793 }
794
795 while(shutdown() == 0)
796 {
797 if(m_fpsFifo == "")
798 {
799 sleep(1);
800 continue;
801 }
802
803 std::string ans = getFPSValStr("mfilt", "loopON");
804
805 if(ans[1] == 'F')
806 {
807 m_loopState = 0;
808 }
809 else m_loopState = 2; //closed
810
811 ans = getFPSValNum("mfilt", "loopgain");
812
813 try
814 {
815 m_gain = std::stof(ans);
816 }
817 catch(const std::exception& e)
818 {
819 m_gain = 0;
820 }
821
822 ans = getFPSValNum("mfilt", "loopmult");
823 try
824 {
825 m_multCoeff = std::stof(ans);
826 }
827 catch(...)
828 {
829 m_multCoeff = 0;
830 }
831 ans = getFPSValNum("mfilt", "looplimit");
832 try
833 {
834 m_maxLim = std::stof(ans);
835 }
836 catch(...)
837 {
838 m_maxLim = 0;
839 }
840
842
843 mx::sys::milliSleep(500);
844
845 }
846
847 return;
848}
849
850INDI_NEWCALLBACK_DEFN(cacaoInterface, m_indiP_loopState )(const pcf::IndiProperty &ipRecv)
851{
852 INDI_VALIDATE_CALLBACK_PROPS(m_indiP_loopState, ipRecv);
853
854 if(!ipRecv.find("toggle")) return 0;
855
856 std::unique_lock<std::mutex> lock(m_indiMutex);
857
858 if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::On)
859 {
860 return loopOn();
861 }
862 else if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::Off)
863 {
864 return loopOff();
865 }
866
867 log<software_error>({__FILE__,__LINE__, "switch state fall through."});
868 return -1;
869}
870
871INDI_NEWCALLBACK_DEFN(cacaoInterface, m_indiP_loopGain )(const pcf::IndiProperty &ipRecv)
872{
873 INDI_VALIDATE_CALLBACK_PROPS(m_indiP_loopGain, ipRecv);
874
875 float current = -1;
876 float target = -1;
877
878 if(ipRecv.find("current"))
879 {
880 current = ipRecv["current"].get<double>();
881 }
882
883 if(ipRecv.find("target"))
884 {
885 target = ipRecv["target"].get<double>();
886 }
887
888 if(target == -1) target = current;
889
890 if(target == -1)
891 {
892 return 0;
893 }
894
895 std::lock_guard<std::mutex> guard(m_indiMutex);
896
897 m_gain_target = target;
898
899 updateIfChanged(m_indiP_loopGain, "target", m_gain_target);
900
901 return setGain();
902}
903
904INDI_NEWCALLBACK_DEFN(cacaoInterface, m_indiP_loopZero )(const pcf::IndiProperty &ipRecv)
905{
906 INDI_VALIDATE_CALLBACK_PROPS(m_indiP_loopZero, ipRecv);
907
908 if(!ipRecv.find("request")) return 0;
909
910 std::unique_lock<std::mutex> lock(m_indiMutex);
911
912 if( ipRecv["request"].getSwitchState() == pcf::IndiElement::On)
913 {
914 return loopZero();
915 }
916
917 return 0;
918}
919
920INDI_NEWCALLBACK_DEFN(cacaoInterface, m_indiP_multCoeff )(const pcf::IndiProperty &ipRecv)
921{
922 INDI_VALIDATE_CALLBACK_PROPS(m_indiP_multCoeff, ipRecv);
923
924 float current = -1;
925 float target = -1;
926
927 if(ipRecv.find("current"))
928 {
929 current = ipRecv["current"].get<double>();
930 }
931
932 if(ipRecv.find("target"))
933 {
934 target = ipRecv["target"].get<double>();
935 }
936
937 if(target == -1) target = current;
938
939 if(target == -1)
940 {
941 return 0;
942 }
943
944 std::lock_guard<std::mutex> guard(m_indiMutex);
945
946 m_multCoeff_target = target;
947 updateIfChanged(m_indiP_multCoeff, "target", target);
948
949 return setMultCoeff();
950}
951
952INDI_NEWCALLBACK_DEFN(cacaoInterface, m_indiP_maxLim )(const pcf::IndiProperty &ipRecv)
953{
954 INDI_VALIDATE_CALLBACK_PROPS(m_indiP_maxLim, ipRecv);
955
956 float current = -1;
957 float target = -1;
958
959 if(ipRecv.find("current"))
960 {
961 current = ipRecv["current"].get<double>();
962 }
963
964 if(ipRecv.find("target"))
965 {
966 target = ipRecv["target"].get<double>();
967 }
968
969 if(target == -1) target = current;
970
971 if(target == -1)
972 {
973 return 0;
974 }
975
976 std::lock_guard<std::mutex> guard(m_indiMutex);
977 m_maxLim_target = target;
978 updateIfChanged(m_indiP_maxLim, "target", target);
979
980 return setMaxLim();
981}
982
983inline
988
989inline
991{
992 return recordLoopGain(true);
993}
994
995inline
997{
998 static uint8_t state {0};
999 static float gain {-1000};
1000 static float multcoef {0};
1001 static float limit {0};
1002
1003 if(state != m_loopState || gain != m_gain || multcoef != m_multCoeff || limit != m_maxLim || force)
1004 {
1006 gain = m_gain;
1007 multcoef = m_multCoeff;
1008 limit = m_maxLim;
1009
1011 }
1012
1013 return 0;
1014}
1015
1016} //namespace app
1017} //namespace MagAOX
1018
1019#endif //cacaoInterface_hpp
The base-class for MagAO-X applications.
Definition MagAOXApp.hpp:73
void updateIfChanged(pcf::IndiProperty &p, const std::string &el, const T &newVal, pcf::IndiProperty::PropertyStateType ipState=pcf::IndiProperty::Ok)
Update an INDI property element value if it has changed.
int createStandardIndiRequestSw(pcf::IndiProperty &prop, const std::string &name, const std::string &label="", const std::string &group="")
Create a standard R/W INDI switch with a single request element.
stateCodes::stateCodeT state()
Get the current state code.
int registerIndiPropertyNew(pcf::IndiProperty &prop, int(*)(void *, const pcf::IndiProperty &))
Register an INDI property which is exposed for others to request a New Property for.
int createStandardIndiToggleSw(pcf::IndiProperty &prop, const std::string &name, const std::string &label="", const std::string &group="")
Create a standard R/W INDI switch with a single toggle element.
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.
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.
int registerIndiPropertyReadOnly(pcf::IndiProperty &prop)
Register an INDI property which is read only.
int createROIndiText(pcf::IndiProperty &prop, const std::string &propName, const std::string &elName, const std::string &propLabel="", const std::string &propGroup="", const std::string &elLabel="")
Create a standard ReadOnly INDI Text property, with at least one element.
std::mutex m_indiMutex
Mutex for locking INDI communications.
int threadStart(std::thread &thrd, bool &thrdInit, pid_t &tpid, pcf::IndiProperty &thProp, int thrdPrio, const std::string &cpuset, const std::string &thrdName, thisPtr *thrdThis, Function &&thrdStart)
Start a thread, using this class's privileges to set priority, etc.
The MagAO-X CACAO Interface.
int setMultCoeff()
Set loop multiplication coefficient to the value of m_multCoeff_target;.
pcf::IndiProperty m_indiP_loopZero
int m_loopState
The loop state. 0 = off, 1 = paused (on, 0 gain), 2 = on.
int checkLoopProcesses()
Check if the loop processes are running.
INDI_NEWCALLBACK_DECL(cacaoInterface, m_indiP_maxLim)
bool m_fmThreadInit
Initialization flag for the file monitoring thread.
virtual int appLogic()
Implementation of the FSM for cacaoInterface.
std::string getFPSValNum(const std::string &fps, const std::string &param)
void fmThreadExec()
File monitoring thread function.
float m_gain_target
The target loop gain.
static void fmThreadStart(cacaoInterface *c)
File monitoring thread starter function.
INDI_NEWCALLBACK_DECL(cacaoInterface, m_indiP_loopZero)
int loopOn()
Turn the loop on.
int setFPSVal(const std::string &fps, const std::string &param, const std::string &val)
std::vector< float > m_modeBlockLims
pcf::IndiProperty m_indiP_loopState
bool m_loopProcesses
Status of the loop processes.
float m_maxLim
The current max limit.
INDI_NEWCALLBACK_DECL(cacaoInterface, m_indiP_multCoeff)
pcf::IndiProperty m_fmThreadProp
The property to hold the f.m. thread details.
std::vector< float > m_modeBlockGains
int m_fmThreadPrio
Priority of the filemonitoring thread.
std::vector< float > m_modeBlockMCs
int setGain()
Set loop gain to the value of m_gain_target;.
float m_gain
The current loop gain.
std::string m_loopName
the loop name
~cacaoInterface() noexcept
D'tor, declared and defined for noexcept.
std::string getFPSValStr(const std::string &fps, const std::string &param)
int recordLoopGain(bool force=false)
std::vector< int > m_modeBlockStart
std::thread m_fmThread
The file monitoring thread.
virtual int appStartup()
Startup function.
int getAOCalib()
Get the calibration details.
int setMaxLim()
Set loop max lim to the value of m_maxLim_target;.
virtual int appShutdown()
Shutdown the app.
float m_maxLim_target
The target max limit.
float m_multCoeff
The current multiplicative coefficient (1-leak)
dev::telemeter< cacaoInterface > telemeterT
int recordTelem(const telem_loopgain *)
INDI_NEWCALLBACK_DECL(cacaoInterface, m_indiP_loopState)
std::string m_loopNumber
The loop number, X in aolX. We keep it a string because that's how it gets used.
int loadConfigImpl(mx::app::appConfigurator &_config)
Implementation of loadConfig logic, separated for testing.
pid_t m_fmThreadID
File monitor thread PID.
std::vector< int > m_modeBlockN
int loopOff()
Turn the loop off.
pcf::IndiProperty m_indiP_loopProcesses
pcf::IndiProperty m_indiP_multCoeff
float m_multCoeff_target
The target multiplicative coefficient (1-leak)
pcf::IndiProperty m_indiP_loopGain
int loopZero()
Zero the loop control channel.
bool m_loopProcesses_stat
What the cacao status file says the state of loop processes is.
INDI_NEWCALLBACK_DECL(cacaoInterface, m_indiP_loopGain)
#define INDI_NEWCALLBACK_DEFN(class, prop)
Define the callback for a new property request.
#define INDI_NEWCALLBACK(prop)
Get the name of the static callback wrapper for a new property.
@ OPERATING
The device is operating, other than homing.
@ ERROR
The application has encountered an error, from which it is recovering (with or without intervention)
@ READY
The device is ready for operation, but is not operating.
#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
int addTextElement(pcf::IndiProperty &prop, const std::string &name, const std::string &label="")
Add a standard INDI Text element.
Definition indiUtils.hpp:37
const pcf::IndiProperty & ipRecv
updateIfChanged(m_indiP_angle, "target", m_angle)
std::unique_lock< std::mutex > lock(m_indiMutex)
Definition dm.hpp:24
static constexpr logPrioT LOG_NOTICE
A normal but significant condition.
static constexpr logPrioT LOG_INFO
Informational. The info log level is the lowest level recorded during normal operations.
static constexpr logPrioT LOG_CRITICAL
The process can not continue and will shut down (fatal)
static constexpr logPrioT LOG_ERROR
An error has occured which the software will attempt to correct.
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 appLogic()
Perform telemeter application logic.
int setupConfig(appConfigurator &config)
Setup an application configurator for the device section.
int appStartup()
Starts the telemetry log thread.
int checkRecordTimes(const telT &tel, telTs... tels)
Check the time of the last record for each telemetry type and make an entry if needed.
Software CRITICAL log entry.
Software ERR log entry.
Log entry recording the build-time git state.