API
 
Loading...
Searching...
No Matches
MagAOXApp.hpp
Go to the documentation of this file.
1/** \file magAOXApp.hpp
2 * \brief The basic MagAO-X Application
3 * \author Jared R. Males (jaredmales@gmail.com)
4 *
5 * History:
6 * - 2017-12-24 created by JRM
7 *
8 * \ingroup app_files
9 */
10
11#ifndef app_MagAOXApp_hpp
12#define app_MagAOXApp_hpp
13
14#include <signal.h>
15#include <sys/stat.h>
16#include <sys/syscall.h>
17
18#include <cstdlib>
19#include <fstream>
20#include <sstream>
21#include <thread>
22#include <mutex>
23
24#include <unordered_map>
25
26#include <mx/mxlib.hpp>
27#include <mx/app/application.hpp>
28#include <mx/sys/environment.hpp>
29#include <mx/sys/timeUtils.hpp>
30#include <mx/ioutils/fileUtils.hpp>
31
32#include "../common/environment.hpp"
33#include "../common/paths.hpp"
34#include "../common/defaults.hpp"
35#include "../common/config.hpp"
36
37#include "../logger/logFileRaw.hpp"
38#include "../logger/logManager.hpp"
39
40#include "../sys/thSetuid.hpp"
41
42#include "stateCodes.hpp"
43#include "indiDriver.hpp"
44#include "indiMacros.hpp"
45#include "indiUtils.hpp"
46
47using namespace mx::app;
48
49using namespace MagAOX::logger;
50
51namespace MagAOX
52{
53namespace app
54{
55
56/// The base-class for MagAO-X applications.
57/**
58 * You can define a base configuration file for this class by defining
59 * \code
60 * m_configBase = "base_name";
61 * \endcode
62 * in the derived class constructor. This would be used, for instance to have a config common to
63 * all filter wheels.
64 *
65 *
66 * \todo make m_powerMgtEnabled a template parameter, and static_assert check if _useINDI== false and power management
67 * is true
68 *
69 * \ingroup magaoxapp
70 */
71template <bool _useINDI = true>
72class MagAOXApp : public application
73{
74 friend class MagAOXApp_test;
75
76 public:
77 /// The log manager type.
79
80 protected:
81 std::string MagAOXPath; ///< The base path of the MagAO-X system.
82
83 std::string m_configName; ///< The name of the configuration file (minus .conf).
84
85 std::string m_configDir; ///< The path to configuration files for MagAOX.
86
87 std::string m_configBase; ///< The name of a base config class for this app (minus .conf).
88
89 std::string m_calibDir; ///< The path to calibration files for MagAOX.
90
91 std::string sysPath; ///< The path to the system directory, for PID file, etc.
92
93 std::string secretsPath; ///< Path to the secrets directory, where passwords, etc, are stored.
94
95 unsigned long m_loopPause{ MAGAOX_default_loopPause }; /**< The time in nanoseconds to pause the main loop.
96 The appLogic() function of the derived class is called
97 every m_loopPause nanoseconds. Default is
98 1,000,000,000 ns. Config with loopPause=X.*/
99
100 int m_shutdown{ 0 }; ///< Flag to signal it's time to shutdown. When not 0, the main loop exits.
101
102 private:
103 /// Default c'tor is deleted.
104 MagAOXApp() = delete;
105
106 public:
107 /// Public c'tor. Handles uid, logs git repo status, and initializes static members.
108 /**
109 * Only one MagAOXApp can be instantiated per program. Hence this c'tor will issue exit(-1)
110 * if the static self-pointer m_self is already initialized.
111 *
112 * euid is set to 'real' to ensure that the application has normal privileges unless
113 * explicitly needed.
114 *
115 * Reference: http://man7.org/linux/man-pages/man2/getresuid.2.html
116 *
117 * The git repository status is required to create a MagAOXApp. Derived classes
118 * should include the results of running `gengithead.sh` and pass the defined
119 * sha1 and modified flags.
120 *
121 */
122 MagAOXApp( const std::string &git_sha1, ///< [in] The current SHA1 hash of the git repository
123 const bool git_modified ///< [in] Whether or not the repo is modified.
124 );
125
127
128 /// Get the value of the shutdown flag.
129 /**
130 * \returns the current value of m_shutdown
131 */
132 int shutdown();
133
134 /// Set the paths for config files
135 /** Replaces the mx::application defaults with the MagAO-X config system.
136 *
137 * This function parses the CL for "-n" or "--name".
138 *
139 *
140 * Do not override this unless you intend to depart from the MagAO-X standard.
141 */
142 virtual void setDefaults( int argc, ///< [in] standard command line result specifying number of arguments in argv
143 char **argv ///< [in] standard command line result containing the arguments.
144 );
145
146 /// The basic MagAO-X configuration setup method. Should not normally be overridden.
147 /** This method sets up the config system with the standard MagAO-X key=value pairs.
148 *
149 * Though it is virtual, it should not normally be overridden unless you need
150 * to depart from the MagAO-X standard.
151 *
152 * Setting up app specific config goes in setupConfig() implemented in the derived class.
153 */
155
156 /// The basic MagAO-X configuration processing method. Should not normally be overridden.
157 /** This method processes the standard MagAO-X key=value pairs.
158 *
159 * Though it is virtual, it should not normally be overridden unless you need
160 * to depart from the MagAO-X standard.
161 *
162 * Processing of app specific config goes in loadConfig() implemented by the derived class.
163 */
165
166 /// Check for unused and unrecognized config options and settings.
167 /** Logs the unused targets as warnings. Unrecognized and unused options are logged as critical, and m_shutdown is
168 * set. Any command line argument (not an option) will also be critical and cause shutdown.
169 */
171
172 /// The execute method implementing the standard main loop. Should not normally be overridden.
173 /** Performs final startup steps. That is:
174 * - Verifies correct effective user-id by comparison to logs directory.
175 * - PID locking lockPID()
176 * - log thread startup by logThreadStart()
177 * - signal handling installation by setSigTermHandler()
178 * - appStartup() is called
179 * - INDI communications started by startINDI()
180 * - power state is checked, pausing if unknown (if being managed)
181 *
182 * Errors in the above steps will cause a process exit.
183 *
184 * Then commences the main event loop.
185 * Conditions on entry to the main loop:
186 * - PID locked
187 * - Log thread running
188 * - Signal handling installed
189 * - appStartup successful
190 * - INDI communications started successfully (if being used)
191 * - power state known (if being managed)
192 *
193 * In the event loop, the power state is checked (if being managed). If power is off, then onPowerOff is called.
194 * If power is on, or power is not managed, appLogic is called. These methods are implemented in derived classes,
195 * and are called every m_loopPause interval.
196 *
197 * If an error is returned by either onPowerOff or appLogic, or a signal is handled, then the shutdown is managed.
198 * This includes shutting down INDI, calling appShutdown, and unlocking the PID. The log thread will shutdown.
199 */
201
202 /** \name Pure Virtual Functions
203 * Derived applications must implement these.
204 * @{
205 */
206
207 /// Any tasks to perform prior to the main event loop go here.
208 /** This is called after signal handling is installed. FSM state is
209 * stateCodes::INITIALIZED when this is called.
210 *
211 * Set m_shutdown = 1 on any fatal errors here.
212 */
214
215 /// This is where derived applications implement their main FSM logic.
216 /** This will be called every m_loopPause nanoseconds until the application terminates.
217 *
218 * FSM state will be whatever it is on exti from appStartup.
219 *
220 * Should return -1 on an any unrecoverable errors which will caues app to terminate. Could also set m_shutdown=1.
221 * Return 0 on success, or at least intent to continue.
222 *
223 */
224 virtual int appLogic() = 0;
225
226 /// Any tasks to perform after main loop exit go here.
227 /** Should be able to handle case where appStartup and/or appLogic have not run.
228 */
230
231 ///@} -- Pure Virtual Functions
232
233 /** \name Logging
234 * @{
235 */
236 public:
238
239 /// Make a log entry
240 /** Wrapper for logManager::log
241 *
242 * \tparam logT the log entry type
243 * \tparam retval the value returned by this method.
244 *
245 */
246 template <typename logT, int retval = 0>
247 static int log( const typename logT::messageT &msg, ///< [in] the message to log
248 logPrioT level = logPrio::LOG_DEFAULT /**< [in] [optional] the log level. The default
249 is used if not specified.*/
250 );
251
252 /// Make a log entry
253 /** Wrapper for logManager::log
254 *
255 * \tparam logT the log entry type
256 * \tparam retval the value returned by this method.
257 *
258 */
259 template <typename logT, int retval = 0>
260 static int log( logPrioT level = logPrio::LOG_DEFAULT /**< [in] [optional] the log level. The default is
261 used if not specified.*/
262 );
263
264 /// Handle a log message from the logging system
265 /** This is a callback from the logManager, and is called when the log thread is processing log entries.
266 *
267 * Decides whether to display to stderr and whether to send via INDI.
268 */
270
271 protected:
272 /// Callback for config system logging.
273 /** Called by appConfigurator each time a value is set using the config() operator.
274 * You never need to call this directly.
275 */
276 static void configLog( const std::string &name, ///< [in] The name of the config value
277 const int &code, ///< [in] numeric code specifying the type
278 const std::string &value, ///< [in] the value read by the config system
279 const std::string &source ///< [in] the source of the value.
280 );
281
282 ///@} -- logging
283
284 /** \name Signal Handling
285 * @{libMagAOX/logger/types/software_log.hpp
286 */
287 private:
289 *m_self; ///< Static pointer to this (set in constructor). Used to test whether a a MagAOXApp is already
290 ///< instatiated (a fatal error) and used for getting out of static signal handlers.
291
292 /// Sets the handler for SIGTERM, SIGQUIT, and SIGINT.
294
295 /// The handler called when SIGTERM, SIGQUIT, or SIGINT is received. Just a wrapper for handlerSigTerm.
296 static void _handlerSigTerm( int signum, ///< [in] specifies the signal.
297 siginfo_t *siginf, ///< [in] ignored by MagAOXApp
298 void *ucont ///< [in] ignored by MagAOXApp
299 );
300
301 /// Handles SIGTERM, SIGQUIT, and SIGINT. Sets m_shutdown to 1 and logs the signal.
302 void handlerSigTerm( int signum, ///< [in] specifies the signal.
303 siginfo_t *siginf, ///< [in] ignored by MagAOXApp
304 void *ucont ///< [in] ignored by MagAOXApp
305 );
306
307 ///@} -- Signal Handling
308
309 /** \name Privilege Management
310 * @{
311 */
312 private:
313 uid_t m_euidReal; ///< The real user id of the proces (i.e. the lower privileged id of the user)
314 uid_t m_euidCalled; ///< The user id of the process as called (i.e. the higher privileged id of the owner, root if
315 ///< setuid).
316 uid_t m_suid; ///< The save-set user id of the process
317
318 protected:
319 /// Internal class to manage setuid privilege escalation with RAII
320 /** Upon construction this elevates to the called user id, root in a setuid process.
321 * Restores privileges to real user id upon destruction (i.e. when it goes out of scope).
322 */
324 {
325 private:
327 bool m_elevated{ false };
328
329 public:
331 {
332 m_app = app;
333 elevate();
334 }
335
336 void elevate()
337 {
338 if( m_elevated )
339 return;
340
341 m_app->setEuidCalled();
342 m_elevated = true;
343 }
344
345 void restore()
346 {
347 if( !m_elevated )
348 return;
349
350 m_app->setEuidReal();
351 m_elevated = false;
352 }
353
355 {
356 restore();
357 }
358 };
359
360 private:
361 /// Set the effective user ID to the called value, i.e. the highest possible.
362 /** If setuid is set on the file, this will be super-user privileges.
363 *
364 * Reference: http://pubs.opengroup.org/onlinepubs/009695399/functions/seteuid.html
365 *
366 * \returns 0 on success
367 * \returns -1 on error from setuid().
368 */
370
371 /// Set the effective user ID to the real value, i.e. the file owner.
372 /**
373 * Reference: http://pubs.opengroup.org/onlinepubs/009695399/functions/seteuid.html
374 *
375 * \returns 0 on success
376 * \returns -1 on error from setuid().
377 */
379
380 ///@} -- Privilege Management
381
382 protected:
383 /** \name PID Locking
384 *
385 * Each MagAOXApp has a PID lock file in the system directory. The app will not
386 * startup if it detects that the PID is already locked, preventing duplicates. This is
387 * based on the configured name, not the invoked name (argv[0]).
388 *
389 * @{
390 */
391
392 std::string pidFileName; ///< The name of the PID file
393
394 pid_t m_pid{ 0 }; ///< This process's PID
395
396 /// Attempt to lock the PID by writing it to a file. Fails if a process is already running with the same config
397 /// name.
398 /** First checks the PID file for an existing PID. If found, interrogates /proc to determine if that process is
399 * running and if so if the command line matches. If a matching process is currently running, then this returns an
400 * error.
401 *
402 * Will not fail if a PID file exists but the stored PID does not correspond to a running process with the same
403 * command line name.
404 *
405 * Reference: https://linux.die.net/man/3/getpid
406 *
407 * \returns 0 on success.
408 * \returns -1 on any error, including creating the PID file or if this app is already running.
409 */
410 int lockPID();
411
412 /// Remove the PID file.
414
415 ///@} -- PID Locking
416
417 /** \name cpusets
418 * The path to the cpusets mount is configured by the environment varialbe defined by MAGOX_env_cpuset
419 * in environment.hpp. This environment varialbe is normally named "CGROUPS1_CPUSET_MOUNTPOINT". If
420 * the environment variable is not set, the default defined by MAGAOX_cpusetPath in paths.hpp is used.
421 * This is normally "/opt/MagAOX/cpuset/"
422 */
423
425
426 /** \name Threads
427 *
428 * @{
429 */
430 public:
431 /// Start a thread, using this class's privileges to set priority, etc.
432 /**
433 * The thread initialization synchronizer `bool` is set to true at the beginning
434 * of this function, then is set to false once all initialization is complete. The
435 * thread exec function should wait until this is false before doing _anything_ except
436 * setting the pid. This is to avoid privilege escalation bugs.
437 *
438 * The interface of the thread start function is:
439 \code
440 static void impl::myThreadStart( impl * o )
441 {
442 o->myThreadExec(); //A member function which actually executes the thread
443 }
444 \endcode
445 * where `impl` is the derived class, and `mThreadStart` and `myThreadExec` are members
446 * of `impl`.
447 *
448 * \returns 0 on success
449 * \returns -1 on error
450 */
451 template <class thisPtr, class Function>
452 int threadStart( std::thread &thrd, /**< [out] The thread object to start executing */
453 bool &thrdInit, /**< [in/out] The thread initilization synchronizer. */
454 pid_t &tpid, /**< [in/out] The thread pid to be filled in by thrdStart
455 immediately upon call*/
456 pcf::IndiProperty &thProp, /**< [in/out] The INDI property to publish the thread details */
457 int thrdPrio, /**< [in] The r/t priority to set for this thread */
458 const std::string &cpuset, /**< [in] the cpuset to place this thread on. Ignored if "". */
459 const std::string &thrdName, /**< [in] The name of the thread (just for logging) */
460 thisPtr *thrdThis, /**< [in] The `this` pointer to pass to the thread starter function */
461 Function &&thrdStart /**< [in] The thread starting function, a static function taking a
462 `this` pointer as argument. */
463 );
464
465 ///@} -- Threads
466
467 /** \name Application State
468 *
469 * @{
470 */
471 private:
472 stateCodes::stateCodeT m_state{ stateCodes::UNINITIALIZED }; ///< The application's state. Never ever set this
473 ///< directly, use state(const stateCodeT & s).
474
475 bool m_stateAlert{ false }; // Flag to control whether the FSM is in an alert state. Once set, only user
476 // acknowledgement can change this.
477
478 bool m_gitAlert{ false }; // Flag set if there is a git modified warning to alert on.
479
480 int m_stateLogged{ 0 }; /**< Counter and flag for use to log errors just once.
481 Never ever access directly, use stateLogged().*/
482
483 public:
484 /// Get the current state code
485 /** \returns m_state
486 */
488
489 /// Set the current state code
490 /*
491 * If it is a change, the state change is logged. Also resets m_stateLogged to 0.
492 *
493 * Will also update INDI if there is a change.
494 */
495 void state( const stateCodes::stateCodeT &s, ///< [in] The new application state
496 bool stateAlert = false ///< [in] [optional] flag to set the alert state of the FSM property.
497 );
498
499 /// Updates and returns the value of m_stateLogged. Will be 0 on first call after a state change, >0 afterwards.
500 /** This method exists to facilitate logging the reason for a state change once, but not
501 * logging it on subsequent event loops. Returns the current value upon entry, but updates
502 * before returning so that the next call returns the incremented value. Example usage:
503 * \code
504 if( connection_failed ) //some condition set this to true
505 {
506 state( stateCodes::NOTCONNECTED );
507 if(!stateLogged()) log<text_log>("Not connected");
508 }
509 \endcode
510 * In this example, the log entry is made the first time the state changes. If there are no changes to a
511 * different state in the mean time, then when the event loop gets here again and decides it is not connected,
512 * the log entry will not be made.
513 *
514 * \returns current value of m_stateLogged, that is the value before it is incremented.
515 */
517
518 ///@} --Application State
519
520 private:
521 /// Clear the FSM alert state.
522 /** This can only be done from within this class, and this
523 * should only be possible via user action via INDI.
524 */
526
527 /** \name INDI Interface
528 *
529 * For reference: "Get" and "New" refer to properties we own. "Set" refers to properties owned by others.
530 * So we respond to GetProperties by listing our own properties, and NewProperty is a request to change
531 * a property we own. Whereas SetProperty is a notification that someone else has changed a property.
532 *
533 * @{
534 */
535 protected:
536 /// Flag controlling whether INDI is used. If false, then no INDI code executes.
537 constexpr static bool m_useINDI = _useINDI;
538
539 ///\todo instead of making this public, provide an accessor.
540 public:
541 /// The INDI driver wrapper. Constructed and initialized by execute, which starts and stops communications.
543
544 /// Mutex for locking INDI communications.
545 std::mutex m_indiMutex;
546
547 protected:
548 /// Structure to hold the call-back details for handling INDI communications.
550 {
551 pcf::IndiProperty *property{ 0 }; ///< A pointer to an INDI property.
552 int ( *callBack )( void *, const pcf::IndiProperty & ){ 0 }; /**< The function to call for a new or
553 set property.*/
554
555 bool m_defReceived{ false }; /**< Flag indicating that a DefProperty has been received
556 after a GetProperty.*/
557 };
558
559 public:
560 /// Value type of the indiCallBack map.
561 typedef std::pair<std::string, indiCallBack> callBackValueType;
562
563 /// Iterator type of the indiCallBack map.
564 typedef typename std::unordered_map<std::string, indiCallBack>::iterator callBackIterator;
565
566 /// Return type of insert on the indiCallBack map.
567 typedef std::pair<callBackIterator, bool> callBackInsertResult;
568
569 protected:
570 /// Map to hold the NewProperty indiCallBacks for this App, with fast lookup by property name.
571 /** The key for these is the property name.
572 */
573 std::unordered_map<std::string, indiCallBack> m_indiNewCallBacks;
574
575 /// Map to hold the SetProperty indiCallBacks for this App, with fast lookup by property name.
576 /** The key for these is device.name
577 */
578 std::unordered_map<std::string, indiCallBack> m_indiSetCallBacks;
579
580 protected:
581 /// Flag indicating that all registered Set properties have been updated since last Get.
582 bool m_allDefsReceived{ false };
583
584 /// Full path name of the INDI driver input FIFO.
585 std::string m_driverInName;
586
587 /// Full path name of the INDI driver output FIFO.
588 std::string m_driverOutName;
589
590 /// Full path name of the INDI driver control FIFO.
591 /** This is currently only used to signal restarts.
592 */
593 std::string m_driverCtrlName;
594
595 public:
596 /// Create a standard R/W INDI Text property with target and current elements.
597 /**
598 * \returns 0 on success
599 * \returns -1 on error
600 */
601 int createStandardIndiText( pcf::IndiProperty &prop, /**< [out] the property to create and setup */
602 const std::string &propName, /**< [in] the name of the property */
603 const std::string &label = "", /**< [in] [optional] the GUI label suggestion for this
604 property */
605 const std::string &group = "" /**< [in] [optional] the group for this property */
606 );
607
608 /// Create a standard ReadOnly INDI Text property, with at least one element.
609 /**
610 * \returns 0 on success
611 * \returns -1 on error
612 */
613 int
614 createROIndiText( pcf::IndiProperty &prop, ///< [out] the property to create and setup
615 const std::string &propName, ///< [in] the name of the property
616 const std::string &elName, ///< [in] the name of the element
617 const std::string &propLabel = "", ///< [in] [optional] the GUI label suggestion for this property
618 const std::string &propGroup = "", ///< [in] [optional] the group for this property
619 const std::string &elLabel = "" ///< [in] [optional] the GUI label suggestion for the element
620 );
621
622 /// Create a standard R/W INDI Number property with target and current elements.
623 /**
624 * \returns 0 on success
625 * \returns -1 on error
626 */
627 template <typename T>
628 int createStandardIndiNumber( pcf::IndiProperty &prop, /**< [out] the property to create and setup */
629 const std::string &name, /**< [in] the name of the property */
630 const T &min, /**< [in] the minimum value for the elements, applied
631 to both target and current */
632 const T &max, /**< [in] the minimum value for the elements, applied
633 to both target and current */
634 const T &step, /**< [in] the step size for the elements, applied to
635 both target and current */
636 const std::string &format, /**< [in] the _ value for the elements, applied to
637 both target and current. Set to "" to use
638 the MagAO-X standard for type. */
639 const std::string &label = "", /**< [in] [optional] the GUI label suggestion for
640 this property */
641 const std::string &group = "" /**< [in] [optional] the group for this property*/
642 );
643
644 /// Create a ReadOnly INDI Number property
645 /**
646 * \returns 0 on success
647 * \returns -1 on error
648 */
649 int createROIndiNumber( pcf::IndiProperty &prop, /**< [out] the property to create and setup */
650 const std::string &propName, /**< [in] the name of the property */
651 const std::string &propLabel = "", /**< [in] [optional] the GUI label suggestion for this
652 property */
653 const std::string &propGroup = "" /**< [in] [optional] the group for this property */
654 );
655
656 /// Create a standard R/W INDI switch with a single toggle element.
657 /** This switch is intended to function like an on/off toggle switch.
658 *
659 * \returns 0 on success
660 * \returns -1 on error
661 */
662 int createStandardIndiToggleSw( pcf::IndiProperty &prop, /**< [out] the property to create and setup */
663 const std::string &name, /**< [in] the name of the property */
664 const std::string &label = "", /**< [in] [optional] the GUI label suggestion for
665 this property */
666 const std::string &group = "" /**< [in] [optional] the group for this property */
667 );
668
669 /// Create a standard R/W INDI switch with a single request element.
670 /** This switch is intended to function like a momentary switch.
671 *
672 * \returns 0 on success
673 * \returns -1 on error
674 */
675 int createStandardIndiRequestSw( pcf::IndiProperty &prop, /**< [out] the property to create and setup */
676 const std::string &name, /**< [in] the name of the property */
677 const std::string &label = "", /**< [in] [optional] the GUI label suggestion
678 for this property */
679 const std::string &group = "" /**< [in] [optional] the group for this property */
680 );
681
682 /// Create a standard R/W INDI selection (one of many) switch with vector of elements and element labels
683 /** This switch is intended to function like drop down menu.
684 *
685 * \returns 0 on success
686 * \returns -1 on error
687 */
688 int createStandardIndiSelectionSw( pcf::IndiProperty &prop, /**< [out] the property to create and setup */
689 const std::string &name, /**< [in] the name of the property, */
690 const std::vector<std::string> &elements, /**< [in] the element names to give to
691 the switches */
692 const std::vector<std::string> &elementLabels, /**< [in] the element labels to
693 give to the switches */
694 const std::string &label = "", /**< [in] [optional] the GUI label suggestion for
695 this property */
696 const std::string &group = "" /**< [in] [optional] the group for this property */
697 );
698
699 /// Create a standard R/W INDI selection (one of many) switch with vector of elements using the element strings as
700 /// their own labels
701 /** This switch is intended to function like drop down menu.
702 *
703 * \returns 0 on success
704 * \returns -1 on error
705 */
706 int createStandardIndiSelectionSw( pcf::IndiProperty &prop, ///< [out] the property to create and setup
707 const std::string &name, ///< [in] the name of the property,
708 const std::vector<std::string> &elements, /**< [in] the element names to give to
709 the switches */
710 const std::string &label = "", /**< [in] [optional] the GUI label suggestion for
711 this property */
712 const std::string &group = "" ///< [in] [optional] the group for this property
713 );
714
715 /// Register an INDI property which is read only.
716 /** This version requires the property be fully set up.
717 *
718 * \returns 0 on success.
719 * \returns -1 on error.
720 *
721 */
722 int registerIndiPropertyReadOnly( pcf::IndiProperty &prop /**< [in] the property to register, must be
723 completely setup */ );
724
725 /// Register an INDI property which is read only.
726 /** This verison sets up the INDI property according to the arguments.
727 *
728 * \returns 0 on success.
729 * \returns -1 on error.
730 *
731 */
732 int registerIndiPropertyReadOnly( pcf::IndiProperty &prop, /**< [out] the property to
733 register, will
734 be configured */
735 const std::string &propName, /**< [in] the name of the
736 property */
737 const pcf::IndiProperty::Type &propType, /**< [in] the type of the
738 property */
739 const pcf::IndiProperty::PropertyPermType &propPerm, /**< [in] the permissions
740 of the property
741 */
742 const pcf::IndiProperty::PropertyStateType &propState /**< [in] the state of the
743 property */
744 );
745
746 /// Register an INDI property which is exposed for others to request a New Property for.
747 /** In this version the supplied IndiProperty must be fully set up before passing in.
748 *
749 * \returns 0 on success.
750 * \returns -1 on error.
751 *
752 */
753 int registerIndiPropertyNew( pcf::IndiProperty &prop, /**< [in] the property to register,
754 must be fully set up */
755 int ( * )( void *, const pcf::IndiProperty & ) /**< [in] the callback for changing
756 the property */
757 );
758
759 /// Register an INDI property which is exposed for others to request a New Property for.
760 /** This verison sets up the INDI property according to the arguments.
761 *
762 * \returns 0 on success.
763 * \returns -1 on error.
764 *
765 */
767 pcf::IndiProperty &prop, ///< [out] the property to register
768 const std::string &propName, ///< [in] the name of the property
769 const pcf::IndiProperty::Type &propType, ///< [in] the type of the property
770 const pcf::IndiProperty::PropertyPermType &propPerm, ///< [in] the permissions of the property
771 const pcf::IndiProperty::PropertyStateType &propState, ///< [in] the state of the property
772 int ( * )( void *, const pcf::IndiProperty & ) ///< [in] the callback for changing the property
773 );
774
775 /// Register an INDI property which is exposed for others to request a New Property for, with a switch rule
776 /** This verison sets up the INDI property according to the arguments.
777 *
778 * \returns 0 on success.
779 * \returns -1 on error.
780 *
781 */
783 pcf::IndiProperty &prop, ///< [out] the property to register
784 const std::string &propName, ///< [in] the name of the property
785 const pcf::IndiProperty::Type &propType, ///< [in] the type of the property
786 const pcf::IndiProperty::PropertyPermType &propPerm, ///< [in] the permissions of the property
787 const pcf::IndiProperty::PropertyStateType &propState, ///< [in] the state of the property
788 const pcf::IndiProperty::SwitchRuleType &propRule, ///< [in] the switch rule type
789 int ( * )( void *, const pcf::IndiProperty & ) ///< [in] the callback for changing the property
790 );
791
792 /// Register an INDI property which is monitored for updates from others.
793 /**
794 *
795 * \returns 0 on success.
796 * \returns -1 on error.
797 *
798 */
800 pcf::IndiProperty &prop, ///< [out] the property to register
801 const std::string &devName, ///< [in] the device which owns this property
802 const std::string &propName, ///< [in] the name of the property
803 int ( * )( void *, const pcf::IndiProperty & ) ///< [in] the callback for processing the property change
804 );
805
806 protected:
807 /// Create the INDI FIFOs
808 /** Changes permissions to max available and creates the
809 * FIFOs at the configured path.
810 */
812
813 /// Start INDI Communications
814 /**
815 * \returns 0 on success
816 * \returns -1 on error. This is fatal.
817 */
819
820 public:
821 void sendGetPropertySetList( bool all = false );
822
823 /// Handler for the DEF INDI properties notification
824 /** Uses the properties registered in m_indiSetCallBacks to process the notification. This is called by
825 * m_indiDriver's indiDriver::handleDefProperties.
826 */
827 void handleDefProperty( const pcf::IndiProperty &ipRecv /**< [in] The property being sent. */ );
828
829 /// Handler for the get INDI properties request
830 /** Uses the properties registered in m_indiCallBacks to respond to the request. This is called by
831 * m_indiDriver's indiDriver::handleGetProperties.
832 */
833 void handleGetProperties( const pcf::IndiProperty &ipRecv /**< [in] The property being requested. */ );
834
835 /// Handler for the new INDI property request
836 /** Uses the properties registered in m_indiCallBacks to respond to the request, looking up the callback for this
837 * property and calling it.
838 *
839 * This is called by m_indiDriver's indiDriver::handleGetProperties.
840 *
841 * \todo handle errors, are they FATAL?
842 */
843 void handleNewProperty( const pcf::IndiProperty &ipRecv /**< [in] The property being changed. */ );
844
845 /// Handler for the set INDI property request
846 /**
847 *
848 * This is called by m_indiDriver's indiDriver::handleSetProperties.
849 *
850 * \todo handle errors, are they FATAL?
851 */
852 void handleSetProperty( const pcf::IndiProperty &ipRecv /**< [in] The property being changed. */ );
853
854 protected:
855 /// Update an INDI property element value if it has changed.
856 /** Will only peform a SetProperty if the new element value has changed
857 * compared to the stored value, or if the property state has changed.
858 *
859 * This comparison is done in the true
860 * type of the value.
861 *
862 * For a property with multiple elements, you should use the vector version to minimize network traffic.
863 */
864 template <typename T>
865 void updateIfChanged( pcf::IndiProperty &p, ///< [in/out] The property containing the element to possibly update
866 const std::string &el, ///< [in] The element name
867 const T &newVal, ///< [in] the new value
868 pcf::IndiProperty::PropertyStateType ipState = pcf::IndiProperty::Ok );
869
870 /// Update an INDI property element value if it has changed.
871 /** Will only peform a SetProperty if the new element value has changed
872 * compared to the stored value, or if the property state has changed.
873 *
874 * This comparison is done in the true
875 * type of the value.
876 *
877 * This is a specialization for `const char *` to `std::string`.
878 *
879 * For a property with multiple elements, you should use the vector version to minimize network traffic.
880 * \overload
881 */
882 void updateIfChanged( pcf::IndiProperty &p, ///< [in/out] The property containing the element to possibly update
883 const std::string &el, ///< [in] The element name
884 const char *newVal, ///< [in] the new value
885 pcf::IndiProperty::PropertyStateType ipState = pcf::IndiProperty::Ok );
886
887 /// Update an INDI switch element value if it has changed.
888 /** Will only peform a SetProperty if the new element switch state has changed, or the propery state
889 * has changed.
890 *
891 */
892 void
893 updateSwitchIfChanged( pcf::IndiProperty &p, ///< [in/out] The property containing the element to possibly update
894 const std::string &el, ///< [in] The element name
895 const pcf::IndiElement::SwitchStateType &newVal, ///< [in] the new value
896 pcf::IndiProperty::PropertyStateType ipState = pcf::IndiProperty::Ok );
897
898 /// Update an INDI property if values have changed.
899 /** Will only peform a SetProperty if at least one value has changed
900 * compared to the stored value, or if the property state has changed.
901 *
902 * Constructs the element names for each value as elX where X is the index of the vector.
903 *
904 * This comparison is done in the true
905 * type of the value.
906 *
907 *
908 * \overload
909 */
910 template <typename T>
912 pcf::IndiProperty &p, ///< [in/out] The property containing the element to possibly update
913 const std::string &el, ///< [in] Beginning of each element name
914 const std::vector<T> &newVals, ///< [in] the new values
915 pcf::IndiProperty::PropertyStateType ipState = pcf::IndiProperty::Ok ///< [in] [optional] the new state
916 );
917
918 /// Update an INDI property if values have changed.
919 /** Will only peform a SetProperty if at least one value has changed
920 * compared to the stored value, or if the property state has changed.
921 *
922 * This comparison is done in the true
923 * type of the value.
924 *
925 * \overload
926 */
927 template <typename T>
928 void updateIfChanged( pcf::IndiProperty &p, ///< [in/out] The property containing the element to possibly update
929 const std::vector<std::string> &els, ///< [in] String vector of element names
930 const std::vector<T> &newVals, ///< [in] the new values
931 pcf::IndiProperty::PropertyStateType newState =
932 pcf::IndiProperty::Ok ///< [in] [optional] The state of the property
933 );
934
935 template <typename T>
936 void updatesIfChanged( pcf::IndiProperty &p, ///< [in/out] The property containing the element to possibly update
937 const std::vector<const char *> &els, ///< [in] String vector of element names
938 const std::vector<T> &newVals, ///< [in] the new values
939 pcf::IndiProperty::PropertyStateType newState =
940 pcf::IndiProperty::Ok ///< [in] [optional] The state of the property
941 );
942
943 /// Get the target element value from an new property
944 /**
945 * \returns 0 on success
946 * \returns -1 on error
947 */
948 template <typename T>
949 int indiTargetUpdate( pcf::IndiProperty &localProperty, ///< [out] The local property to update
950 T &localTarget, ///< [out] The local value to update
951 const pcf::IndiProperty &remoteProperty, ///< [in] the new property received
952 bool setBusy = true ///< [in] [optional] set property to busy if true
953 );
954
955 /// Send a newProperty command to another device (using the INDI Client interface)
956 /** Copies the input IndiProperty, then updates the element with the new value.
957 *
958 * \returns 0 on success.
959 * \returns -1 on an errory.
960 */
961 template <typename T>
962 int sendNewProperty( const pcf::IndiProperty &ipSend, ///< [in] The property to send a "new" INDI command for
963 const std::string &el, ///< [in] The element of the property to change
964 const T &newVal ///< [in] The value to request for the element.
965 );
966 /// Send a newProperty command to another device (using the INDI Client interface)
967 /**
968 *
969 * \returns 0 on success.
970 * \returns -1 on an error, which will be logged
971 */
972 int sendNewProperty( const pcf::IndiProperty &ipSend /**< [in] The property to send a "new" INDI command for */ );
973
974 /// Send a new property commmand for a standard toggle switch
975 /**
976 * \returns 0 on success
977 * \returns -1 on an error, which will be logged.
978 */
979 int sendNewStandardIndiToggle( const std::string &device, ///< [in] The device name
980 const std::string &property, ///< [in] The property name
981 bool onoff ///< [in] Switch state to send: true = on, false = off
982 );
983
984 /// indi Property to report the application state.
985 pcf::IndiProperty m_indiP_state;
986
987 /// indi Property to clear an FSM alert.
988 pcf::IndiProperty m_indiP_clearFSMAlert;
989
990 /// The static callback function to be registered for requesting to clear the FSM alert
991 /**
992 * \returns 0 on success.
993 * \returns -1 on error.
994 */
995 static int st_newCallBack_clearFSMAlert( void *app, /**< [in] a pointer to this, will be
996 static_cast-ed to MagAOXApp. */
997 const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with
998 the new property request. */
999 );
1000
1001 /// The callback called by the static version, to actually process the FSM Alert Clear request.
1002 /**
1003 * \returns 0 on success.
1004 * \returns -1 on error.
1005 */
1006 int newCallBack_clearFSMAlert( const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with
1007 the new property request.*/ );
1008
1009 ///@} --INDI Interface
1010
1011 /** \name Power Management
1012 * For devices which have remote power management (e.g. from one of the PDUs) we implement
1013 * a standard power state monitoring and management component for the FSM. This needs to be enabled
1014 * in the derived app constructor. To stay enabled, m_powerDevice and m_powerChannel must be
1015 * not empty strings after the configuration. These could be set in the derived app defaults.
1016 *
1017 * If power management is enabled, then while power is off, appLogic will not be called.
1018 * Instead a parrallel set of virtual functions is called, onPowerOff (to allow apps to
1019 * perform cleanup) and whilePowerOff (to allow apps to keep variables updated, etc).
1020 * Note that these could merely call appLogic if desired.
1021 *
1022 */
1023 protected:
1024 bool m_powerMgtEnabled{ false }; ///< Flag controls whether power mgt is used. Set this in the constructor of a
1025 ///< derived app. If true, then if after loadConfig the powerDevice and
1026 ///< powerChannel are empty, then the app will exit with a critical error.
1027
1028 /* Configurables . . . */
1029 std::string m_powerDevice; ///< The INDI device name of the power controller
1030 std::string m_powerChannel; ///< The INDI property name of the channel controlling this device's power.
1031 std::string m_powerElement{ "state" }; ///< The INDI element name to monitor for this device's power state.
1032 std::string m_powerTargetElement{ "target" }; ///< The INDI element name to monitor for this device's power state.
1033
1034 unsigned long m_powerOnWait{ 0 }; ///< Time in sec to wait for device to boot after power on.
1035
1036 /* Power on waiting counter . . . */
1037 int m_powerOnCounter{ -1 }; ///< Counts numer of loops after power on, implements delay for device bootup. If -1,
1038 ///< then device was NOT powered off on app startup.
1039
1040 /* Power state . . . */
1041 int m_powerState{ -1 }; ///< Current power state, 1=On, 0=Off, -1=Unk.
1042 int m_powerTargetState{ -1 }; ///< Current target power state, 1=On, 0=Off, -1=Unk.
1043
1044 pcf::IndiProperty m_indiP_powerChannel; ///< INDI property used to communicate power state.
1045
1046 /// This method is called when the change to poweroff is detected.
1047 /**
1048 * \returns 0 on success.
1049 * \returns -1 on any error which means the app should exit.
1050 */
1051 virtual int onPowerOff();
1052
1053 /// This method is called while the power is off, once per FSM loop.
1054 /**
1055 * \returns 0 on success.
1056 * \returns -1 on any error which means the app should exit.
1057 */
1058 virtual int whilePowerOff();
1059
1060 /// This method tests whether the power on wait time has elapsed.
1061 /** You would call this once per appLogic loop while in state POWERON. While false, you would return 0.
1062 * Once it becomes true, take post-power-on actions and go on with life.
1063 *
1064 * \returns true if the time since POWERON is greater than the power-on wait, or if power management is not enabled
1065 * \returns false otherwise
1066 */
1068
1069 public:
1070 /// Returns the current power state.
1071 /** If power management is not enabled, this always returns 1=On.
1072 *
1073 * \returns -1 if power state is unknown
1074 * \returns 0 if power is off
1075 * \returns 1 if power is on or m_powerMgtEnabled==false
1076 */
1078
1079 /// Returns the target power state.
1080 /** If power management is not enabled, this always returns 1=On.
1081 *
1082 * \returns -1 if target power state is unknown
1083 * \returns 0 if target power state is off
1084 * \returns 1 if target power is on or m_powerMgtEnabled==false
1085 */
1087
1089
1090 ///@} Power Management
1091
1092 public:
1093 /** \name Member Accessors
1094 *
1095 * @{
1096 */
1097
1098 /// Get the config name
1099 /**
1100 * \returns the current value of m_configName
1101 */
1102 std::string configName();
1103
1104 /// Get the config directory
1105 /**
1106 * \returns the current value of m_configDir
1107 */
1108 std::string configDir();
1109
1110 /// Get the INDI input FIFO file name
1111 /**
1112 * \returns the current value of m_driverInName
1113 */
1114 std::string driverInName();
1115
1116 /// Get the INDI output FIFO file name
1117 /**
1118 * \returns the current value of m_driverOutName
1119 */
1120 std::string driverOutName();
1121
1122 /// Get the INDI control FIFO file name
1123 /**
1124 * \returns the current value of m_driverCtrlName
1125 */
1126 std::string driverCtrlName();
1127
1128 ///@} --Member Accessors
1129};
1130
1131// Set self pointer to null so app starts up uninitialized.
1132template <bool _useINDI>
1134
1135// Define the logger
1136template <bool _useINDI>
1138
1139template <bool _useINDI>
1140MagAOXApp<_useINDI>::MagAOXApp( const std::string &git_sha1, const bool git_modified )
1141{
1142 if( m_self != nullptr )
1143 {
1144 std::cerr << "Attempt to instantiate 2nd MagAOXApp. Exiting immediately.\n";
1145 exit( -1 );
1146 }
1147
1148 m_self = this;
1149
1150 // Get the uids of this process.
1151 getresuid( &m_euidReal, &m_euidCalled, &m_suid );
1152 setEuidReal(); // immediately step down to unpriveleged uid.
1153
1154 m_log.parent( this );
1155
1156 // Set up config logging
1157 config.m_sources = true;
1158 config.configLog = configLog;
1159
1160 // We log the current GIT status.
1162 if( git_modified )
1163 {
1165 m_gitAlert = true;
1166 }
1167 log<git_state>( git_state::messageT( "MagAOX", git_sha1, git_modified ), gl );
1168
1169 gl = logPrio::LOG_INFO;
1170 if( MXLIB_UNCOMP_REPO_MODIFIED )
1171 {
1173 m_gitAlert = true;
1174 }
1175
1176 log<git_state>( git_state::messageT( "mxlib", MXLIB_UNCOMP_CURRENT_SHA1, MXLIB_UNCOMP_REPO_MODIFIED ), gl );
1177}
1178
1179template <bool _useINDI>
1181{
1182 if( m_indiDriver )
1183 delete m_indiDriver;
1184 m_log.parent( nullptr );
1185
1187}
1188
1189template <bool _useINDI>
1191{
1192 return m_shutdown;
1193}
1194
1195template <bool _useINDI>
1197 char **argv ) // virtual
1198{
1199 std::string tmpstr;
1200
1201 tmpstr = mx::sys::getEnv( MAGAOX_env_path );
1202 if( tmpstr != "" )
1203 {
1204 MagAOXPath = tmpstr;
1205 }
1206 else
1207 {
1208 MagAOXPath = MAGAOX_path;
1209 }
1210
1211 // Set the config path relative to MagAOXPath
1212 tmpstr = mx::sys::getEnv( MAGAOX_env_config );
1213 if( tmpstr == "" )
1214 {
1215 tmpstr = MAGAOX_configRelPath;
1216 }
1217 m_configDir = MagAOXPath + "/" + tmpstr;
1218 m_configPathGlobal = m_configDir + "/magaox.conf";
1219
1220 // Set the calib path relative to MagAOXPath
1221 tmpstr = mx::sys::getEnv( MAGAOX_env_calib );
1222 if( tmpstr == "" )
1223 {
1224 tmpstr = MAGAOX_calibRelPath;
1225 }
1226 m_calibDir = MagAOXPath + "/" + tmpstr;
1227
1228 // Setup default log path
1229 tmpstr = MagAOXPath + "/" + MAGAOX_logRelPath;
1230
1231 m_log.logPath( tmpstr );
1232
1233 // Setup default sys path
1234 tmpstr = MagAOXPath + "/" + MAGAOX_sysRelPath;
1235 sysPath = tmpstr;
1236
1237 // Setup default secrets path
1238 tmpstr = MagAOXPath + "/" + MAGAOX_secretsRelPath;
1239 secretsPath = tmpstr;
1240
1241 // Setup default cpuset path
1242 tmpstr = mx::sys::getEnv( MAGAOX_env_cpuset );
1243 if( tmpstr != "" )
1244 {
1245 m_cpusetPath = tmpstr;
1246 }
1247 // else we stick with the default
1248
1249 if( m_configBase != "" )
1250 {
1251 // We use mx::application's configPathUser for this components base config file
1252 m_configPathUser = m_configDir + "/" + m_configBase + ".conf";
1253 }
1254
1255 // Parse CL just to get the "name".
1256 config.add( "name",
1257 "n",
1258 "name",
1259 argType::Required,
1260 "",
1261 "",
1262 true,
1263 "string",
1264 "The name of the application and its device name in INDI (if used), specifies the config file in the XWC config directory." );
1265
1266 config.parseCommandLine( argc, argv, "name" );
1267 config( m_configName, "name" );
1268
1269 if( m_configName == "" )
1270 {
1271 m_configName = mx::ioutils::pathStem( invokedName );
1272 if(!doHelp)
1273 {
1274 log<text_log>( "Configuration Error: Application name (-n --name) not set." );
1275 doHelp = true;
1276 }
1277 }
1278
1279 // We use mx::application's configPathLocal for this component's config file
1280 m_configPathLocal = m_configDir + "/" + m_configName + ".conf";
1281
1282 // Now we can setup common INDI properties
1283 if( registerIndiPropertyNew(
1284 m_indiP_state, "fsm", pcf::IndiProperty::Text, pcf::IndiProperty::ReadOnly, pcf::IndiProperty::Idle, 0 ) <
1285 0 )
1286 {
1287 log<software_error>( { __FILE__, __LINE__, "failed to register read only fsm_state property" } );
1288 }
1289
1290 m_indiP_state.add( pcf::IndiElement( "state" ) );
1291
1292 createStandardIndiRequestSw( m_indiP_clearFSMAlert, "fsm_clear_alert", "Clear FSM Alert", "FSM" );
1293 if( registerIndiPropertyNew( m_indiP_clearFSMAlert, st_newCallBack_clearFSMAlert ) < 0 )
1294 {
1295 log<software_error>( { __FILE__, __LINE__, "failed to register new fsm_alert property" } );
1296 }
1297
1298 return;
1299}
1300
1301template <bool _useINDI>
1303{
1304 // Validate config
1305 config.add( "config.validate",
1306 "",
1307 "config.validate",
1308 argType::True,
1309 "",
1310 "",
1311 false,
1312 "bool",
1313 "Validate the configuration. App will exit after loading the configuration, but before entering the event loop. Errors from configuratin processing will be shown. Always safe to run." );
1314
1315 // App stuff
1316 config.add( "loopPause",
1317 "p",
1318 "loopPause",
1319 argType::Required,
1320 "",
1321 "loopPause",
1322 false,
1323 "unsigned long",
1324 "The main loop pause time in ns" );
1325
1326 config.add(
1327 "ignore_git", "", "ignore-git", argType::True, "", "", false, "bool", "set to true to ignore git status to prevent the fsm_alert" );
1328
1329 // Logger Stuff
1330 m_log.setupConfig( config );
1331
1332 if( m_powerMgtEnabled )
1333 {
1334 if( _useINDI == false )
1335 {
1336 // If this condition obtains, we should not go on because it means we'll never leave power off!!!
1337 log<software_critical>( { __FILE__, __LINE__, "power management is enabled but we are not using INDI" } );
1338 m_shutdown = true;
1339 }
1340
1341 // Power Management
1342 config.add( "power.device",
1343 "",
1344 "power.device",
1345 argType::Required,
1346 "power",
1347 "device",
1348 false,
1349 "string",
1350 "Device controlling power for this app's device (INDI name)." );
1351 config.add( "power.channel",
1352 "",
1353 "power.channel",
1354 argType::Required,
1355 "power",
1356 "channel",
1357 false,
1358 "string",
1359 "Channel on device for this app's device (INDI name)." );
1360 config.add( "power.element",
1361 "",
1362 "power.element",
1363 argType::Required,
1364 "power",
1365 "element",
1366 false,
1367 "string",
1368 "INDI power state element name. Default is \"state\", only need to specify if different." );
1369 config.add( "power.targetElement",
1370 "",
1371 "power.targetElement",
1372 argType::Required,
1373 "power",
1374 "targetElement",
1375 false,
1376 "string",
1377 "INDI power target element name. Default is \"target\", only need to specify if different." );
1378 config.add( "power.powerOnWait",
1379 "",
1380 "power.powerOnWait",
1381 argType::Required,
1382 "power",
1383 "powerOnWait",
1384 false,
1385 "int",
1386 "Time after power-on to wait before continuing [sec]. Default is 0 sec, max is 3600 sec." );
1387 }
1388}
1389
1390template <bool _useINDI>
1392{
1393 //--------- Ignore Git State --------//
1394 bool ig{ false };
1395 config( ig, "ignore_git" );
1396
1397 if( !ig && m_gitAlert )
1398 {
1399 m_stateAlert = true;
1400 }
1401
1402 //--------- Config Validation Mode --------//
1403 if(config.isSet("config.validate"))
1404 {
1405 m_configOnly = true; //m_configOnly is from mx::application
1406 }
1407
1408 //---------- Setup the logger ----------//
1409 m_log.logName( m_configName );
1410 m_log.loadConfig( config );
1411
1412 //--------- Loop Pause Time --------//
1413 config( m_loopPause, "loopPause" );
1414
1415 //--------Power Management --------//
1416 if( m_powerMgtEnabled )
1417 {
1418 config( m_powerDevice, "power.device" );
1419 config( m_powerChannel, "power.channel" );
1420 config( m_powerElement, "power.element" );
1421 config( m_powerTargetElement, "power.targetElement" );
1422
1423 if( m_powerDevice != "" && m_powerChannel != "" )
1424 {
1425 log<text_log>( "enabling power management: " + m_powerDevice + "." + m_powerChannel + "." + m_powerElement +
1426 "/" + m_powerTargetElement );
1427 if( registerIndiPropertySet(
1428 m_indiP_powerChannel, m_powerDevice, m_powerChannel, INDI_SETCALLBACK( m_indiP_powerChannel ) ) <
1429 0 )
1430 {
1431 log<software_error>( { __FILE__, __LINE__, "failed to register set property" } );
1432 }
1433 }
1434 else
1435 {
1436 log<text_log>( "power management not configured!", logPrio::LOG_CRITICAL );
1437 m_shutdown = true;
1438 }
1439
1440 config( m_powerOnWait, "power.powerOnWait" );
1441 if( m_powerOnWait > 3600 )
1442 {
1443 log<text_log>( "powerOnWait longer than 1 hour. Setting to 0.", logPrio::LOG_ERROR );
1444 }
1445 }
1446}
1447
1448template <bool _useINDI>
1450{
1451 // This checks for unused but valid config options and arguments, and logs them.
1452 // This will catch options we aren't actually using but are configured(debugging).
1453 for( auto it = config.m_targets.begin(); it != config.m_targets.end(); ++it )
1454 {
1455 if( it->second.used == false )
1456 {
1457 std::string msg = it->second.name;
1458 if( config.m_sources && it->second.sources.size() > 0 )
1459 {
1460 msg += " [" + it->second.sources[0] + "]";
1461 }
1462 log<text_log>( "Unused config target: " + msg, logPrio::LOG_WARNING );
1463 }
1464 }
1465
1466 // This checks for invalid/unknown config options and arguments, and logs them.
1467 // This diagnosis problems in the config file
1468 if( config.m_unusedConfigs.size() > 0 )
1469 {
1470 for( auto it = config.m_unusedConfigs.begin(); it != config.m_unusedConfigs.end(); ++it )
1471 {
1472 if( it->second.used == true )
1473 {
1474 continue;
1475 }
1476
1477 std::string msg = it->second.name;
1478 if( config.m_sources && it->second.sources.size() > 0 )
1479 {
1480 msg += " [" + it->second.sources[0] + "]";
1481 }
1482 log<text_log>( "Unrecognized config setting: " + msg, logPrio::LOG_CRITICAL );
1483
1484 m_shutdown = true;
1485 }
1486 }
1487
1488 //MagAO-X does not use non-option CLI arguments. Presence probably points to a typo (missing - or --).
1489 if( config.nonOptions.size() > 0 )
1490 {
1491 for( size_t n = 0; n < config.nonOptions.size(); ++n )
1492 {
1493 log<text_log>( "Unrecognized command line argument: " + config.nonOptions[n], logPrio::LOG_CRITICAL );
1494 }
1495 m_shutdown = true;
1496 }
1497
1498 if(m_configOnly) //validation mode
1499 {
1500 if(m_shutdown == true)
1501 {
1502 std::cerr << "\nThere were configuration errors.\n\n";
1503 }
1504 else
1505 {
1506 std::cerr << "\nConfiguration is valid.\n\n";
1507 }
1508 }
1509 else if(m_shutdown == true)
1510 {
1511 doHelp = true; //Causes mx::application to print help and exit.
1512 }
1513}
1514
1515template <bool _useINDI>
1517{
1518//----------------------------------------//
1519// Check user
1520//----------------------------------------//
1521#ifndef XWC_DISABLE_USER_CHECK
1522 struct stat logstat;
1523
1524 if( stat( m_log.logPath().c_str(), &logstat ) < 0 )
1525 {
1526 state( stateCodes::FAILURE );
1527 std::cerr << "\nCRITICAL: Can not stat the log path.\n\n";
1528 return -1;
1529 }
1530
1531 if( logstat.st_uid != geteuid() )
1532 {
1533 state( stateCodes::FAILURE );
1534 std::cerr << "\nCRITICAL: You are running this app as the wrong user.\n\n";
1535 return -1;
1536 }
1537#endif
1538
1539 //----------------------------------------//
1540 // Get the PID Lock
1541 //----------------------------------------//
1542 if( lockPID() < 0 )
1543 {
1544 state( stateCodes::FAILURE );
1545
1546 // We don't log this, because it won't be logged anyway.
1547 std::cerr << "\nCRITICAL: Failed to lock PID. Exiting.\n\n";
1548
1549 // Return immediately, not safe to go on.
1550 return -1;
1551 }
1552
1553 /* ***************************** */
1554 /* start logging */
1555 /* ***************************** */
1556 m_log.logThreadStart(); // no return type
1557
1558 // Give up to 2 secs to make sure log thread has time to get started and try to open a file.
1559 int w = 0;
1560 while( m_log.logThreadRunning() == false && w < 20 )
1561 {
1562 // Sleep for 100 msec
1563 std::this_thread::sleep_for( std::chrono::duration<unsigned long, std::nano>( 100000000 ) );
1564 ++w;
1565 }
1566
1567 if( m_log.logThreadRunning() == false )
1568 {
1569 state( stateCodes::FAILURE );
1570
1571 // We don't log this, because it won't be logged anyway.
1572 std::cerr << "\nCRITICAL: log thread not running. Exiting.\n\n";
1573
1574 m_shutdown = 1; //just in case, though this should not have an effect yet.
1575
1576 if( unlockPID() < 0 )
1577 {
1578 log<software_error>( { __FILE__, __LINE__, "error from unlockPID()" } );
1579 }
1580
1581 return -1;
1582 }
1583
1584 /* ***************************** */
1585 /* signal handling */
1586 /* ***************************** */
1587 if( m_shutdown == 0 )
1588 {
1589 if( setSigTermHandler() < 0 )
1590 {
1591 state( stateCodes::FAILURE );
1592
1593 log<software_critical>( { __FILE__, __LINE__, "error from setSigTermHandler()" } );
1594
1595 m_shutdown = 1; //just in case, though this should not have an effect yet.
1596
1597 if( unlockPID() < 0 )
1598 {
1599 log<software_error>( { __FILE__, __LINE__, "error from unlockPID()" } );
1600 }
1601
1602 return -1;
1603 }
1604 }
1605
1606 /* ***************************** */
1607 /* appStartup() */
1608 /* ***************************** */
1609 if( m_shutdown == 0 )
1610 {
1611 state( stateCodes::INITIALIZED );
1612
1613 if( appStartup() < 0 )
1614 {
1615 state( stateCodes::FAILURE );
1616
1617 log<software_critical>( { __FILE__, __LINE__, "error from appStartup()" } );
1618
1619 m_shutdown = 1; //just in case, though this should not have an effect yet.
1620
1621 if( unlockPID() < 0 )
1622 {
1623 log<software_error>( { __FILE__, __LINE__, "error from unlockPID()" } );
1624 }
1625
1626 return -1;
1627 }
1628 }
1629
1630 //====Begin INDI Communications
1631 if( m_useINDI && m_shutdown == 0 ) // if we're using INDI and not already dead, that is
1632 {
1633 if( startINDI() < 0 )
1634 {
1635 state( stateCodes::FAILURE );
1636
1637 log<software_critical>( { __FILE__, __LINE__, "INDI failed to start." } );
1638
1639 m_shutdown = 1; //have to set so that child event loops know to exit
1640
1641 // Have to call appShutdown since appStartup was called
1642 if( appShutdown() < 0 )
1643 {
1644 log<software_error>( { __FILE__, __LINE__, "error from appShutdown()" } );
1645 }
1646
1647 if( unlockPID() < 0 )
1648 {
1649 log<software_error>( { __FILE__, __LINE__, "error from unlockPID()" } );
1650 }
1651
1652 return -1;
1653 }
1654 }
1655
1656 // We have to wait for power status to become available
1657 if( m_powerMgtEnabled && m_shutdown == 0 )
1658 {
1659 int nwaits = 0;
1660 while( m_powerState < 0 && !m_shutdown )
1661 {
1662 sleep( 1 );
1663 if( m_powerState < 0 )
1664 {
1665 if( !stateLogged() )
1666 log<text_log>( "waiting for power state" );
1667 }
1668
1669 ++nwaits;
1670 if( nwaits == 30 )
1671 {
1672 log<text_log>( "stalled waiting for power state", logPrio::LOG_ERROR );
1673 state( stateCodes::ERROR );
1674 }
1675 }
1676
1677 if( m_powerState > 0 )
1678 {
1679 state( stateCodes::POWERON );
1680 }
1681 else
1682 {
1683 m_powerOnCounter = 0;
1684 state( stateCodes::POWEROFF );
1685 if( onPowerOff() < 0 )
1686 {
1687 log<software_error>( { __FILE__, __LINE__, "error from onPowerOff()" } );
1688 m_shutdown = 1;
1689 }
1690 }
1691 }
1692
1693 // This is the main event loop.
1694 /* Conditions on entry:
1695 * -- PID locked
1696 * -- Log thread running
1697 * -- Signal handling installed
1698 * -- appStartup() successful
1699 * -- INDI communications started successfully (if being used)
1700 * -- power state known (if being managed)
1701 */
1702 while( m_shutdown == 0 )
1703 {
1704 // First check power state.
1705 if( m_powerMgtEnabled )
1706 {
1707 if( state() == stateCodes::POWEROFF )
1708 {
1709 if( m_powerState == 1 )
1710 {
1711 m_powerOnCounter = 0;
1712 state( stateCodes::POWERON );
1713 }
1714 }
1715 else // Any other state
1716 {
1717 if( m_powerState == 0 )
1718 {
1719 state( stateCodes::POWEROFF );
1720 if( onPowerOff() < 0 )
1721 {
1722 log<software_error>( { __FILE__, __LINE__, "error from onPowerOff()" } );
1723 m_shutdown = 1;
1724 continue;
1725 }
1726 }
1727 // We don't do anything if m_powerState is -1, which is a startup condition.
1728 }
1729 }
1730
1731 // Only run appLogic if power is on, or we are not managing power.
1732 if( !m_powerMgtEnabled || m_powerState > 0 )
1733 {
1734 if( appLogic() < 0 )
1735 {
1736 log<software_error>( { __FILE__, __LINE__, "error from appLogic()" } );
1737 m_shutdown = 1;
1738 continue;
1739 }
1740 }
1741 else if( m_powerState == 0 )
1742 {
1743 if( whilePowerOff() < 0 )
1744 {
1745 log<software_error>( { __FILE__, __LINE__, "error from whilePowerOff()" } );
1746 m_shutdown = 1;
1747 continue;
1748 }
1749 }
1750
1751 /** \todo Need a heartbeat update here.
1752 */
1753
1754 if( m_useINDI )
1755 {
1756 // Checkup on the INDI properties we're monitoring.
1757 // This will make sure we are up-to-date if indiserver restarts without us.
1758 // And handles cases where we miss a Def becuase the other driver wasn't started up
1759 // when we sent our Get.
1760 sendGetPropertySetList( false ); // Only does anything if it needs to be done.
1761 }
1762
1763 // This is purely to make sure INDI is up to date in case
1764 // mutex was locked on last attempt.
1765 state( state() );
1766
1767 // Pause loop unless shutdown is set
1768 if( m_shutdown == 0 )
1769 {
1770 std::this_thread::sleep_for( std::chrono::duration<unsigned long, std::nano>( m_loopPause ) );
1771 }
1772 }
1773
1774 if( appShutdown() < 0 )
1775 {
1776 log<software_error>( { __FILE__, __LINE__, "error from appShutdown()" } );
1777 }
1778
1779 state( stateCodes::SHUTDOWN );
1780
1781 // Stop INDI communications
1782 if( m_indiDriver != nullptr )
1783 {
1784 pcf::IndiProperty ipSend;
1785 ipSend.setDevice( m_configName );
1786 try
1787 {
1788 m_indiDriver->sendDelProperty( ipSend );
1789 }
1790 catch( const std::exception &e )
1791 {
1792 log<software_error>(
1793 { __FILE__, __LINE__, std::string( "exception caught from sendDelProperty: " ) + e.what() } );
1794 }
1795
1796 m_indiDriver->quitProcess();
1797 m_indiDriver->deactivate();
1798 log<indidriver_stop>();
1799 }
1800
1801 if( unlockPID() < 0 )
1802 {
1803 log<software_error>( { __FILE__, __LINE__, "error from unlockPID()" } );
1804 }
1805
1806 sleep( 1 );
1807 return 0;
1808}
1809
1810template <bool _useINDI>
1811template <typename logT, int retval>
1812int MagAOXApp<_useINDI>::log( const typename logT::messageT &msg, logPrioT level )
1813{
1814 m_log.template log<logT>( msg, level );
1815 return retval;
1816}
1817
1818template <bool _useINDI>
1819template <typename logT, int retval>
1821{
1822 m_log.template log<logT>( level );
1823 return retval;
1824}
1825
1826template <bool _useINDI>
1828{
1830 {
1831 logStdFormat( std::cerr, b );
1832 std::cerr << "\n";
1833 }
1834
1836 {
1837 state( m_state, true ); // For anything worse than error, we set the FSM state to alert
1838 }
1839
1840 if( _useINDI && m_indiDriver )
1841 {
1842 pcf::IndiProperty msg;
1843 msg.setDevice( m_configName );
1844
1845 std::stringstream logstdf;
1846 logMinStdFormat( logstdf, b );
1847
1848 msg.setMessage( logstdf.str() );
1849
1850 // Set the INDI prop timespec to match the log entry
1852 timeval tv;
1853 tv.tv_sec = ts.time_s;
1854 tv.tv_usec = (long int)( ( (double)ts.time_ns ) / 1e3 );
1855
1856 msg.setTimeStamp( pcf::TimeStamp( tv ) );
1857
1858 try
1859 {
1860 m_indiDriver->sendMessage( msg );
1861 }
1862 catch( const std::exception &e )
1863 {
1864 log<software_error>(
1865 { __FILE__, __LINE__, std::string( "exception caught from sendMessage: " ) + e.what() } );
1866 }
1867 }
1868}
1869
1870template <bool _useINDI>
1871void MagAOXApp<_useINDI>::configLog( const std::string &name,
1872 const int &code,
1873 const std::string &value,
1874 const std::string &source )
1875{
1876 m_log.template log<config_log>( { name, code, value, source } );
1877}
1878
1879template <bool _useINDI>
1881{
1882 struct sigaction act;
1883 sigset_t set;
1884
1885 act.sa_sigaction = &MagAOXApp<_useINDI>::_handlerSigTerm;
1886 act.sa_flags = SA_SIGINFO;
1887 sigemptyset( &set );
1888 act.sa_mask = set;
1889
1890 errno = 0;
1891 if( sigaction( SIGTERM, &act, 0 ) < 0 )
1892 {
1893 std::string logss = "Setting handler for SIGTERM failed. Errno says: ";
1894 logss += strerror( errno );
1895
1896 log<software_error>( { __FILE__, __LINE__, errno, 0, logss } );
1897
1898 return -1;
1899 }
1900
1901 errno = 0;
1902 if( sigaction( SIGQUIT, &act, 0 ) < 0 )
1903 {
1904 std::string logss = "Setting handler for SIGQUIT failed. Errno says: ";
1905 logss += strerror( errno );
1906
1907 log<software_error>( { __FILE__, __LINE__, errno, 0, logss } );
1908
1909 return -1;
1910 }
1911
1912 errno = 0;
1913 if( sigaction( SIGINT, &act, 0 ) < 0 )
1914 {
1915 std::string logss = "Setting handler for SIGINT failed. Errno says: ";
1916 logss += strerror( errno );
1917
1918 log<software_error>( { __FILE__, __LINE__, errno, 0, logss } );
1919
1920 return -1;
1921 }
1922
1923 log<text_log>( "Installed SIGTERM/SIGQUIT/SIGINT signal handler.", logPrio::LOG_DEBUG );
1924
1925 return 0;
1926}
1927
1928template <bool _useINDI>
1929void MagAOXApp<_useINDI>::_handlerSigTerm( int signum, siginfo_t *siginf, void *ucont )
1930{
1931 m_self->handlerSigTerm( signum, siginf, ucont );
1932}
1933
1934template <bool _useINDI>
1936 siginfo_t *siginf __attribute__( ( unused ) ),
1937 void *ucont __attribute__( ( unused ) ) )
1938{
1939 m_shutdown = 1;
1940
1941 std::string signame;
1942 switch( signum )
1943 {
1944 case SIGTERM:
1945 signame = "SIGTERM";
1946 break;
1947 case SIGINT:
1948 signame = "SIGINT";
1949 break;
1950 case SIGQUIT:
1951 signame = "SIGQUIT";
1952 break;
1953 default:
1954 signame = "OTHER";
1955 }
1956
1957 std::string logss = "Caught signal ";
1958 logss += signame;
1959 logss += ". Shutting down.";
1960
1961 std::cerr << "\n" << logss << std::endl;
1962 log<text_log>( logss );
1963}
1964
1965/// Empty signal handler. SIGUSR1 is used to interrupt sleep in various threads.
1966void sigUsr1Handler( int signum, siginfo_t *siginf, void *ucont );
1967
1968template <bool _useINDI>
1970{
1971 errno = 0;
1972 if( sys::th_seteuid( m_euidCalled ) < 0 )
1973 {
1974 std::string logss = "Setting effective user id to euidCalled (";
1975 logss += mx::ioutils::convertToString<int>( m_euidCalled );
1976 logss += ") failed. Errno says: ";
1977 logss += strerror( errno );
1978
1979 log<software_error>( { __FILE__, __LINE__, errno, 0, logss } );
1980
1981 return -1;
1982 }
1983
1984 return 0;
1985}
1986
1987template <bool _useINDI>
1989{
1990 errno = 0;
1991 if( sys::th_seteuid( m_euidReal ) < 0 )
1992 {
1993 std::string logss = "Setting effective user id to euidReal (";
1994 logss += mx::ioutils::convertToString<int>( m_euidReal );
1995 logss += ") failed. Errno says: ";
1996 logss += strerror( errno );
1997
1998 log<software_error>( { __FILE__, __LINE__, errno, 0, logss } );
1999
2000 return -1;
2001 }
2002
2003 return 0;
2004}
2005
2006template <bool _useINDI>
2008{
2009 m_pid = getpid();
2010
2011 std::string statusDir = sysPath;
2012
2013 // Get the maximum privileges available
2014 elevatedPrivileges elPriv( this );
2015
2016 // Create statusDir root with read/write/search permissions for owner and group, and with read/search permissions
2017 // for others.
2018 errno = 0;
2019 if( mkdir( statusDir.c_str(), S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH ) < 0 )
2020 {
2021 if( errno != EEXIST )
2022 {
2023 std::stringstream logss;
2024 logss << "Failed to create root of statusDir (" << statusDir << "). Errno says: " << strerror( errno );
2025 log<software_critical>( { __FILE__, __LINE__, errno, 0, logss.str() } );
2026 return -1;
2027 }
2028 }
2029
2030 statusDir += "/";
2031 statusDir += m_configName;
2032
2033 pidFileName = statusDir + "/pid";
2034
2035 // Create statusDir with read/write/search permissions for owner and group, and with read/search permissions for
2036 // others.
2037 errno = 0;
2038 if( mkdir( statusDir.c_str(), S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH ) < 0 )
2039 {
2040 if( errno != EEXIST )
2041 {
2042 std::stringstream logss;
2043 logss << "Failed to create statusDir (" << statusDir << "). Errno says: " << strerror( errno );
2044 log<software_critical>( { __FILE__, __LINE__, errno, 0, logss.str() } );
2045 return -1;
2046 }
2047
2048 // If here, then we need to check the pid file.
2049
2050 std::ifstream pidIn;
2051 pidIn.open( pidFileName );
2052
2053 if( pidIn.good() ) // PID file exists, now read its contents and compare to proc/<pid>/cmdline
2054 {
2055 // Read PID from file
2056 pid_t testPid;
2057 pidIn >> testPid;
2058 pidIn.close();
2059
2060 // Get command line used to start this process from /proc
2061 std::stringstream procN;
2062 procN << "/proc/" << testPid << "/cmdline";
2063
2064 std::ifstream procIn;
2065 std::string pidCmdLine;
2066
2067 try
2068 {
2069 procIn.open( procN.str() );
2070 if( procIn.good() )
2071 procIn >> pidCmdLine;
2072 procIn.close();
2073 }
2074 catch( ... )
2075 {
2076 log<software_critical>( { __FILE__, __LINE__, 0, 0, "exception caught testing /proc/pid" } );
2077 return -1;
2078 }
2079
2080 // If pidCmdLine == "" at this point we just allow the rest of the
2081 // logic to run...
2082
2083 // Search for invokedName in command line.
2084 size_t invokedPos = pidCmdLine.find( invokedName );
2085
2086 // If invokedName found, then we check for configName.
2087 size_t configPos = std::string::npos;
2088 if( invokedPos != std::string::npos )
2089 configPos = pidCmdLine.find( m_configName );
2090
2091 // Check if PID is already locked by this program+config combo:
2092 if( invokedPos != std::string::npos && configPos != std::string::npos )
2093 {
2094 // This means that this app already exists for this config, and we need to die.
2095 std::stringstream logss;
2096 logss << "PID already locked (" << testPid << "). Time to die.";
2097 std::cerr << logss.str() << std::endl;
2098
2099 log<text_log>( logss.str(), logPrio::LOG_CRITICAL );
2100
2101 return -1;
2102 }
2103 }
2104 else
2105 {
2106 // No PID File so we should just go on.
2107 pidIn.close();
2108 }
2109 }
2110
2111 // Now write current PID to file and go on with life.
2112 std::ofstream pidOut;
2113 pidOut.open( pidFileName );
2114
2115 if( !pidOut.good() )
2116 {
2117 log<software_critical>( { __FILE__, __LINE__, errno, 0, "could not open pid file for writing." } );
2118 // euidReal();
2119 return -1;
2120 }
2121
2122 pidOut << m_pid;
2123
2124 pidOut.close();
2125
2126 std::stringstream logss;
2127 logss << "PID (" << m_pid << ") locked.";
2128 log<text_log>( logss.str() );
2129
2130 // Go back to regular privileges
2131 /*if( euidReal() < 0 )
2132 {
2133 log<software_error>({__FILE__, __LINE__, 0, 0, "Seeting euid to real failed."});
2134 return -1;
2135 }*/
2136
2137 return 0;
2138}
2139
2140template <bool _useINDI>
2142{
2143 { // scope for elPriv
2144
2145 // Get the maximum privileges available
2146 elevatedPrivileges elPriv( this );
2147
2148 if( ::remove( pidFileName.c_str() ) < 0 )
2149 {
2150 log<software_error>(
2151 { __FILE__, __LINE__, errno, 0, std::string( "Failed to remove PID file: " ) + strerror( errno ) } );
2152 return -1;
2153 }
2154 }
2155
2156 std::stringstream logss;
2157 logss << "PID (" << m_pid << ") unlocked.";
2158 log<text_log>( logss.str() );
2159
2160 return 0;
2161}
2162
2163template <bool _useINDI>
2164template <class thisPtr, class Function>
2166 bool &thrdInit,
2167 pid_t &tpid,
2168 pcf::IndiProperty &thProp,
2169 int thrdPrio,
2170 const std::string &cpuset,
2171 const std::string &thrdName,
2172 thisPtr *thrdThis,
2173 Function &&thrdStart )
2174{
2175 thrdInit = true;
2176
2177 tpid = 0;
2178
2179 try
2180 {
2181 thrd = std::thread( thrdStart, thrdThis );
2182 }
2183 catch( const std::exception &e )
2184 {
2185 log<software_error>(
2186 { __FILE__, __LINE__, std::string( "Exception on " + thrdName + " thread start: " ) + e.what() } );
2187 return -1;
2188 }
2189 catch( ... )
2190 {
2191 log<software_error>( { __FILE__, __LINE__, "Unkown exception on " + thrdName + " thread start" } );
2192 return -1;
2193 }
2194
2195 if( !thrd.joinable() )
2196 {
2197 log<software_error>( { __FILE__, __LINE__, thrdName + " thread did not start" } );
2198 return -1;
2199 }
2200
2201 // Now set the RT priority.
2202
2203 if( thrdPrio < 0 )
2204 thrdPrio = 0;
2205 if( thrdPrio > 99 )
2206 thrdPrio = 99;
2207
2208 sched_param sp;
2209 sp.sched_priority = thrdPrio;
2210
2211 int rv = 0;
2212
2213 { // scope for elPriv
2214 // Get the maximum privileges available
2215 elevatedPrivileges elPriv( this );
2216
2217 // We set return value based on result from sched_setscheduler
2218 // But we make sure to restore privileges no matter what happens.
2219 errno = 0;
2220 if( thrdPrio > 0 )
2221 rv = pthread_setschedparam( thrd.native_handle(), MAGAOX_RT_SCHED_POLICY, &sp );
2222 else
2223 rv = pthread_setschedparam( thrd.native_handle(), SCHED_OTHER, &sp );
2224 }
2225
2226 if( rv < 0 )
2227 {
2228 log<software_error>(
2229 { __FILE__,
2230 __LINE__,
2231 errno,
2232 "Setting " + thrdName + " thread scheduler priority to " + std::to_string( thrdPrio ) + " failed." } );
2233 }
2234 else
2235 {
2236 log<text_log>( thrdName + " thread scheduler priority set to " + std::to_string( thrdPrio ) );
2237 }
2238
2239 // Wait for tpid to be filled in, but only for one total second.
2240 if( tpid == 0 )
2241 {
2242 for( int i = 0; i < 10; ++i )
2243 {
2244 mx::sys::milliSleep( 100 );
2245 if( tpid != 0 )
2246 break;
2247 }
2248 }
2249
2250 if( tpid == 0 )
2251 {
2252 return log<software_error, -1>( { __FILE__, __LINE__, errno, "tpid for " + thrdName + " not set." } );
2253 }
2254 else
2255 {
2256 log<text_log>( thrdName + " thread pid is " + std::to_string( tpid ) );
2257
2258 if( _useINDI )
2259 {
2260 thProp = pcf::IndiProperty( pcf::IndiProperty::Number );
2261 thProp.setDevice( configName() );
2262 thProp.setName( std::string( "th-" ) + thrdName );
2263 thProp.setPerm( pcf::IndiProperty::ReadOnly );
2264 thProp.setState( pcf::IndiProperty::Idle );
2265 thProp.add( pcf::IndiElement( "pid" ) );
2266 thProp["pid"] = tpid;
2267 thProp.add( pcf::IndiElement( "prio" ) );
2268 thProp["prio"] = thrdPrio;
2269 registerIndiPropertyReadOnly( thProp );
2270 }
2271
2272 if( cpuset != "" )
2273 {
2274 elevatedPrivileges ep( this );
2275 std::string cpuFile = m_cpusetPath;
2276 cpuFile += "/" + cpuset;
2277 cpuFile += "/tasks";
2278 int wfd = open( cpuFile.c_str(), O_WRONLY );
2279 if( wfd < 0 )
2280 {
2281 return log<software_error, -1>( { __FILE__, __LINE__, errno, "error from open for " + cpuFile } );
2282 }
2283
2284 char pids[128];
2285 snprintf( pids, sizeof( pids ), "%d", tpid );
2286
2287 int w = write( wfd, pids, strnlen( pids, sizeof( pids ) ) );
2288 if( w != (int)strnlen( pids, sizeof(pids) ) )
2289 {
2290 return log<software_error, -1>( { __FILE__, __LINE__, errno, "error on write" } );
2291 }
2292
2293 close( wfd );
2294
2295 log<text_log>( "moved " + thrdName + " to cpuset " + cpuset, logPrio::LOG_NOTICE );
2296 }
2297 }
2298
2299 thrdInit = false;
2300
2301 return 0;
2302}
2303
2304template <bool _useINDI>
2306{
2307 return m_state;
2308}
2309
2310template <bool _useINDI>
2312{
2313 // Only log anything if it's a change
2314 if( m_state != s )
2315 {
2317 if( s == stateCodes::ERROR )
2318 lvl = logPrio::LOG_ERROR;
2319 if( s == stateCodes::FAILURE )
2321
2322 log<state_change>( { m_state, s }, lvl );
2323
2324 m_state = s;
2325 m_stateLogged = 0;
2326 }
2327
2328 if( m_stateAlert != stateAlert && stateAlert == true )
2329 {
2330 m_stateAlert = stateAlert;
2331 log<text_log>( "FSM alert set", logPrio::LOG_WARNING );
2332 }
2333
2334 // Check to make sure INDI is up to date
2335 std::unique_lock<std::mutex> lock( m_indiMutex,
2336 std::try_to_lock ); // Lock the mutex before conducting INDI communications.
2337
2338 // Note this is called every execute loop to make sure we update eventually
2339 if( lock.owns_lock() )
2340 {
2341 ///\todo move this to a function in stateCodes
2342 pcf::IndiProperty::PropertyStateType stst = INDI_IDLE;
2343
2344 // If it's already in the "ALERT" state, then this can't take it out of it.
2345 if( m_stateAlert == true )
2346 {
2347 stst = INDI_ALERT;
2348 }
2349 else
2350 {
2351 if( m_state == stateCodes::READY )
2352 stst = INDI_OK;
2353 else if( m_state == stateCodes::OPERATING || m_state == stateCodes::HOMING ||
2354 m_state == stateCodes::CONFIGURING )
2355 stst = INDI_BUSY;
2356 else if( m_state < stateCodes::NODEVICE )
2357 stst = INDI_ALERT;
2358 else if( m_state <= stateCodes::LOGGEDIN )
2359 stst = INDI_IDLE;
2360 else if( m_state == stateCodes::NOTHOMED || m_state == stateCodes::SHUTDOWN )
2361 stst = INDI_IDLE;
2362 }
2363
2364 updateIfChanged( m_indiP_state, "state", stateCodes::codeText( m_state ), stst );
2365 }
2366}
2367
2368template <bool _useINDI>
2370{
2371 if( m_stateLogged > 0 )
2372 {
2373 ++m_stateLogged;
2374 return m_stateLogged - 1;
2375 }
2376 else
2377 {
2378 m_stateLogged = 1;
2379 return 0;
2380 }
2381}
2382
2383template <bool _useINDI>
2385{
2386 if( m_stateAlert == false )
2387 return 0;
2388 m_stateAlert = false;
2389 log<text_log>( "FSM alert cleared", logPrio::LOG_WARNING );
2390
2391 pcf::IndiProperty::PropertyStateType stst = INDI_IDLE;
2392
2393 if( m_state == stateCodes::READY )
2394 stst = INDI_OK;
2395 else if( m_state == stateCodes::OPERATING || m_state == stateCodes::HOMING || m_state == stateCodes::CONFIGURING )
2396 stst = INDI_BUSY;
2397 else if( m_state < stateCodes::NODEVICE )
2398 stst = INDI_ALERT;
2399 else if( m_state <= stateCodes::LOGGEDIN )
2400 stst = INDI_IDLE;
2401 else if( m_state == stateCodes::NOTHOMED || m_state == stateCodes::SHUTDOWN )
2402 stst = INDI_IDLE;
2403
2404 updateIfChanged( m_indiP_state, "state", stateCodes::codeText( m_state ), stst );
2405
2406 return 0;
2407}
2408
2409/*-------------------------------------------------------------------------------------*/
2410/* INDI Support */
2411/*-------------------------------------------------------------------------------------*/
2412
2413template <bool _useINDI>
2415 const std::string &propName,
2416 const std::string &label,
2417 const std::string &group )
2418{
2419 prop = pcf::IndiProperty( pcf::IndiProperty::Text );
2420 prop.setDevice( configName() );
2421 prop.setName( propName );
2422 prop.setPerm( pcf::IndiProperty::ReadWrite );
2423 prop.setState( pcf::IndiProperty::Idle );
2424 prop.add( pcf::IndiElement( "current" ) );
2425 prop.add( pcf::IndiElement( "target" ) );
2426
2427 // Don't set "" just in case libcommon does something with defaults
2428 if( label != "" )
2429 {
2430 prop.setLabel( label );
2431 }
2432
2433 if( group != "" )
2434 {
2435 prop.setGroup( group );
2436 }
2437
2438 return 0;
2439}
2440
2441template <bool _useINDI>
2442int MagAOXApp<_useINDI>::createROIndiText( pcf::IndiProperty &prop,
2443 const std::string &propName,
2444 const std::string &elName,
2445 const std::string &propLabel,
2446 const std::string &propGroup,
2447 const std::string &elLabel )
2448{
2449 prop = pcf::IndiProperty( pcf::IndiProperty::Text );
2450 prop.setDevice( configName() );
2451 prop.setName( propName );
2452 prop.setPerm( pcf::IndiProperty::ReadOnly );
2453 prop.setState( pcf::IndiProperty::Idle );
2454
2455 // Don't set "" just in case libcommon does something with defaults
2456 if( propLabel != "" )
2457 {
2458 prop.setLabel( propLabel );
2459 }
2460
2461 if( propGroup != "" )
2462 {
2463 prop.setGroup( propGroup );
2464 }
2465
2466 prop.add( pcf::IndiElement( elName ) );
2467
2468 if( elLabel != "" )
2469 {
2470 prop[elName].setLabel( elLabel );
2471 }
2472
2473 return 0;
2474}
2475
2476template <bool _useINDI>
2477template <typename T>
2479 const std::string &name,
2480 const T &min,
2481 const T &max,
2482 const T &step,
2483 const std::string &format,
2484 const std::string &label,
2485 const std::string &group )
2486{
2487 prop = pcf::IndiProperty( pcf::IndiProperty::Number );
2488 prop.setDevice( configName() );
2489 prop.setName( name );
2490 prop.setPerm( pcf::IndiProperty::ReadWrite );
2491 prop.setState( pcf::IndiProperty::Idle );
2492 prop.add( pcf::IndiElement( "current" ) );
2493 prop["current"].setMin( min );
2494 prop["current"].setMax( max );
2495 prop["current"].setStep( step );
2496 if( format != "" ) // don't override defaults
2497 {
2498 prop["current"].setFormat( format );
2499 }
2500
2501 prop.add( pcf::IndiElement( "target" ) );
2502 prop["target"].setMin( min );
2503 prop["target"].setMax( max );
2504 prop["target"].setStep( step );
2505 if( format != "" ) // don't override defaults
2506 {
2507 prop["target"].setFormat( format );
2508 }
2509
2510 // Don't set "" just in case libcommon does something with defaults
2511 if( label != "" )
2512 {
2513 prop.setLabel( label );
2514 }
2515
2516 if( group != "" )
2517 {
2518 prop.setGroup( group );
2519 }
2520
2521 return 0;
2522}
2523
2524template <bool _useINDI>
2525int MagAOXApp<_useINDI>::createROIndiNumber( pcf::IndiProperty &prop,
2526 const std::string &propName,
2527 const std::string &propLabel,
2528 const std::string &propGroup )
2529{
2530 prop = pcf::IndiProperty( pcf::IndiProperty::Number );
2531 prop.setDevice( configName() );
2532 prop.setName( propName );
2533 prop.setPerm( pcf::IndiProperty::ReadOnly );
2534 prop.setState( pcf::IndiProperty::Idle );
2535
2536 // Don't set "" just in case libcommon does something with defaults
2537 if( propLabel != "" )
2538 {
2539 prop.setLabel( propLabel );
2540 }
2541
2542 if( propGroup != "" )
2543 {
2544 prop.setGroup( propGroup );
2545 }
2546
2547 return 0;
2548}
2549
2550template <bool _useINDI>
2552 const std::string &name,
2553 const std::string &label,
2554 const std::string &group )
2555{
2556 prop = pcf::IndiProperty( pcf::IndiProperty::Switch );
2557 prop.setDevice( configName() );
2558 prop.setName( name );
2559 prop.setPerm( pcf::IndiProperty::ReadWrite );
2560 prop.setState( pcf::IndiProperty::Idle );
2561 prop.setRule( pcf::IndiProperty::AtMostOne );
2562
2563 // Add the toggle element initialized to Off
2564 prop.add( pcf::IndiElement( "toggle", pcf::IndiElement::Off ) );
2565
2566 // Don't set "" just in case libcommon does something with defaults
2567 if( label != "" )
2568 {
2569 prop.setLabel( label );
2570 }
2571
2572 if( group != "" )
2573 {
2574 prop.setGroup( group );
2575 }
2576
2577 return 0;
2578}
2579
2580template <bool _useINDI>
2582 const std::string &name,
2583 const std::string &label,
2584 const std::string &group )
2585{
2586 prop = pcf::IndiProperty( pcf::IndiProperty::Switch );
2587 prop.setDevice( configName() );
2588 prop.setName( name );
2589 prop.setPerm( pcf::IndiProperty::ReadWrite );
2590 prop.setState( pcf::IndiProperty::Idle );
2591 prop.setRule( pcf::IndiProperty::AtMostOne );
2592
2593 // Add the toggle element initialized to Off
2594 prop.add( pcf::IndiElement( "request", pcf::IndiElement::Off ) );
2595
2596 // Don't set "" just in case libcommon does something with defaults
2597 if( label != "" )
2598 {
2599 prop.setLabel( label );
2600 }
2601
2602 if( group != "" )
2603 {
2604 prop.setGroup( group );
2605 }
2606
2607 return 0;
2608}
2609
2610template <bool _useINDI>
2612 const std::string &name,
2613 const std::vector<std::string> &elements,
2614 const std::vector<std::string> &elementLabels,
2615 const std::string &label,
2616 const std::string &group )
2617{
2618 if( elements.size() == 0 )
2619 {
2620 return log<software_error, -1>( { __FILE__, __LINE__, "elements vector has zero size" } );
2621 }
2622
2623 prop = pcf::IndiProperty( pcf::IndiProperty::Switch );
2624 prop.setDevice( configName() );
2625 prop.setName( name );
2626 prop.setPerm( pcf::IndiProperty::ReadWrite );
2627 prop.setState( pcf::IndiProperty::Idle );
2628 prop.setRule( pcf::IndiProperty::OneOfMany );
2629
2630 // Add the toggle element initialized to Off
2631 for( size_t n = 0; n < elements.size(); ++n )
2632 {
2633 pcf::IndiElement elem = pcf::IndiElement( elements[n], pcf::IndiElement::Off );
2634 elem.setLabel( elementLabels[n] );
2635 prop.add( elem );
2636 }
2637
2638 // Don't set "" just in case libcommon does something with defaults
2639 if( label != "" )
2640 {
2641 prop.setLabel( label );
2642 }
2643
2644 if( group != "" )
2645 {
2646 prop.setGroup( group );
2647 }
2648
2649 return 0;
2650}
2651
2652template <bool _useINDI>
2654 const std::string &name,
2655 const std::vector<std::string> &elements,
2656 const std::string &label,
2657 const std::string &group )
2658{
2659 return createStandardIndiSelectionSw( prop, name, elements, elements, label, group );
2660}
2661
2662template <bool _useINDI>
2664{
2665 if( !m_useINDI )
2666 return 0;
2667
2668 callBackInsertResult result =
2669 m_indiNewCallBacks.insert( callBackValueType( prop.createUniqueKey(), { &prop, nullptr } ) );
2670
2671 try
2672 {
2673 if( !result.second )
2674 {
2675 return log<software_error, -1>(
2676 { __FILE__, __LINE__, "failed to insert INDI property: " + prop.createUniqueKey() } );
2677 }
2678 }
2679 catch( std::exception &e )
2680 {
2681 return log<software_error, -1>( { __FILE__, __LINE__, std::string( "Exception caught: " ) + e.what() } );
2682 }
2683 catch( ... )
2684 {
2685 return log<software_error, -1>( { __FILE__, __LINE__, "Unknown exception caught." } );
2686 }
2687
2688 return 0;
2689}
2690
2691template <bool _useINDI>
2693 const std::string &propName,
2694 const pcf::IndiProperty::Type &propType,
2695 const pcf::IndiProperty::PropertyPermType &propPerm,
2696 const pcf::IndiProperty::PropertyStateType &propState )
2697{
2698 if( !m_useINDI )
2699 return 0;
2700
2701 prop = pcf::IndiProperty( propType );
2702 prop.setDevice( m_configName );
2703 prop.setName( propName );
2704 prop.setPerm( propPerm );
2705 prop.setState( propState );
2706
2707 callBackInsertResult result = m_indiNewCallBacks.insert( callBackValueType( propName, { &prop, nullptr } ) );
2708
2709 try
2710 {
2711 if( !result.second )
2712 {
2713 return log<software_error, -1>(
2714 { __FILE__, __LINE__, "failed to insert INDI property: " + prop.createUniqueKey() } );
2715 }
2716 }
2717 catch( std::exception &e )
2718 {
2719 return log<software_error, -1>( { __FILE__, __LINE__, std::string( "Exception caught: " ) + e.what() } );
2720 }
2721 catch( ... )
2722 {
2723 return log<software_error, -1>( { __FILE__, __LINE__, "Unknown exception caught." } );
2724 }
2725 return 0;
2726}
2727
2728template <bool _useINDI>
2729int MagAOXApp<_useINDI>::registerIndiPropertyNew( pcf::IndiProperty &prop,
2730 int ( *callBack )( void *, const pcf::IndiProperty &ipRecv ) )
2731{
2732 if( !m_useINDI )
2733 return 0;
2734
2735 try
2736 {
2737 callBackInsertResult result =
2738 m_indiNewCallBacks.insert( callBackValueType( prop.createUniqueKey(), { &prop, callBack } ) );
2739
2740 if( !result.second )
2741 {
2742 return log<software_error, -1>(
2743 { __FILE__, __LINE__, "failed to insert INDI property: " + prop.createUniqueKey() } );
2744 }
2745 }
2746 catch( std::exception &e )
2747 {
2748 return log<software_error, -1>( { __FILE__, __LINE__, std::string( "Exception caught: " ) + e.what() } );
2749 }
2750 catch( ... )
2751 {
2752 return log<software_error, -1>( { __FILE__, __LINE__, "Unknown exception caught." } );
2753 }
2754
2755 return 0;
2756}
2757
2758template <bool _useINDI>
2759int MagAOXApp<_useINDI>::registerIndiPropertyNew( pcf::IndiProperty &prop,
2760 const std::string &propName,
2761 const pcf::IndiProperty::Type &propType,
2762 const pcf::IndiProperty::PropertyPermType &propPerm,
2763 const pcf::IndiProperty::PropertyStateType &propState,
2764 int ( *callBack )( void *, const pcf::IndiProperty &ipRecv ) )
2765{
2766 if( !m_useINDI )
2767 return 0;
2768
2769 prop = pcf::IndiProperty( propType );
2770 prop.setDevice( m_configName );
2771 prop.setName( propName );
2772 prop.setPerm( propPerm );
2773 prop.setState( propState );
2774
2775 return registerIndiPropertyNew( prop, callBack );
2776}
2777
2778template <bool _useINDI>
2779int MagAOXApp<_useINDI>::registerIndiPropertyNew( pcf::IndiProperty &prop,
2780 const std::string &propName,
2781 const pcf::IndiProperty::Type &propType,
2782 const pcf::IndiProperty::PropertyPermType &propPerm,
2783 const pcf::IndiProperty::PropertyStateType &propState,
2784 const pcf::IndiProperty::SwitchRuleType &propRule,
2785 int ( *callBack )( void *, const pcf::IndiProperty &ipRecv ) )
2786{
2787 if( !m_useINDI )
2788 return 0;
2789
2790 prop = pcf::IndiProperty( propType );
2791 prop.setDevice( m_configName );
2792 prop.setName( propName );
2793 prop.setPerm( propPerm );
2794 prop.setState( propState );
2795 prop.setRule( propRule );
2796 return registerIndiPropertyNew( prop, callBack );
2797}
2798
2799template <bool _useINDI>
2801 const std::string &devName,
2802 const std::string &propName,
2803 int ( *callBack )( void *, const pcf::IndiProperty &ipRecv ) )
2804{
2805 if( !m_useINDI )
2806 return 0;
2807
2808 prop = pcf::IndiProperty();
2809 prop.setDevice( devName );
2810 prop.setName( propName );
2811
2812 callBackInsertResult result =
2813 m_indiSetCallBacks.insert( callBackValueType( prop.createUniqueKey(), { &prop, callBack } ) );
2814
2815 try
2816 {
2817 if( !result.second )
2818 {
2819 return log<software_error, -1>(
2820 { __FILE__, __LINE__, "failed to insert INDI property: " + prop.createUniqueKey() } );
2821 }
2822 }
2823 catch( std::exception &e )
2824 {
2825 return log<software_error, -1>( { __FILE__, __LINE__, std::string( "Exception caught: " ) + e.what() } );
2826 }
2827 catch( ... )
2828 {
2829 return log<software_error, -1>( { __FILE__, __LINE__, "Unknown exception caught." } );
2830 }
2831
2832 return 0;
2833}
2834
2835template <bool _useINDI>
2837{
2838 if( !m_useINDI )
2839 return 0;
2840
2841 ///\todo make driver FIFO path full configurable.
2842 std::string driverFIFOPath = MAGAOX_path;
2843 driverFIFOPath += "/";
2844 driverFIFOPath += MAGAOX_driverFIFORelPath;
2845
2846 m_driverInName = driverFIFOPath + "/" + configName() + ".in";
2847 m_driverOutName = driverFIFOPath + "/" + configName() + ".out";
2848 m_driverCtrlName = driverFIFOPath + "/" + configName() + ".ctrl";
2849
2850 // Get max permissions
2851 elevatedPrivileges elPriv( this );
2852
2853 // Clear the file mode creation mask so mkfifo does what we want. Don't forget to restore it.
2854 mode_t prev = umask( 0 );
2855
2856 errno = 0;
2857 if( mkfifo( m_driverInName.c_str(), S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP ) != 0 )
2858 {
2859 if( errno != EEXIST )
2860 {
2861 umask( prev );
2862 log<software_critical>( { __FILE__, __LINE__, errno, 0, "mkfifo failed" } );
2863 log<text_log>( "Failed to create input FIFO.", logPrio::LOG_CRITICAL );
2864 return -1;
2865 }
2866 }
2867
2868 errno = 0;
2869 if( mkfifo( m_driverOutName.c_str(), S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP ) != 0 )
2870 {
2871 if( errno != EEXIST )
2872 {
2873 umask( prev );
2874 // euidReal();
2875 log<software_critical>( { __FILE__, __LINE__, errno, 0, "mkfifo failed" } );
2876 log<text_log>( "Failed to create ouput FIFO.", logPrio::LOG_CRITICAL );
2877 return -1;
2878 }
2879 }
2880
2881 errno = 0;
2882 if( mkfifo( m_driverCtrlName.c_str(), S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP ) != 0 )
2883 {
2884 if( errno != EEXIST )
2885 {
2886 umask( prev );
2887 // euidReal();
2888 log<software_critical>( { __FILE__, __LINE__, errno, 0, "mkfifo failed" } );
2889 log<text_log>( "Failed to create ouput FIFO.", logPrio::LOG_CRITICAL );
2890 return -1;
2891 }
2892 }
2893
2894 umask( prev );
2895 // euidReal();
2896 return 0;
2897}
2898
2899template <bool _useINDI>
2901{
2902 if( !m_useINDI )
2903 return 0;
2904
2905 //===== Create the FIFOs for INDI communications ====
2906 if( createINDIFIFOS() < 0 )
2907 {
2908 return -1;
2909 }
2910
2911 //======= Instantiate the indiDriver
2912 try
2913 {
2914 if( m_indiDriver != nullptr )
2915 {
2916 m_indiDriver->quitProcess();
2917 m_indiDriver->deactivate();
2918 log<indidriver_stop>();
2919 delete m_indiDriver;
2920 m_indiDriver = nullptr;
2921 }
2922
2923 m_indiDriver = new indiDriver<MagAOXApp>( this, m_configName, "0", "0" );
2924 }
2925 catch( ... )
2926 {
2927 log<software_critical>( { __FILE__, __LINE__, 0, 0, "INDI Driver construction exception." } );
2928 return -1;
2929 }
2930
2931 // Check for INDI failure
2932 if( m_indiDriver == nullptr )
2933 {
2934 log<software_critical>( { __FILE__, __LINE__, 0, 0, "INDI Driver construction failed." } );
2935 return -1;
2936 }
2937
2938 // Check for INDI failure to open the FIFOs
2939 if( m_indiDriver->good() == false )
2940 {
2941 log<software_critical>( { __FILE__, __LINE__, 0, 0, "INDI Driver failed to open FIFOs." } );
2942 delete m_indiDriver;
2943 m_indiDriver = nullptr;
2944 return -1;
2945 }
2946
2947 //======= Now we start talkin'
2948 m_indiDriver->activate();
2949 log<indidriver_start>();
2950
2951 sendGetPropertySetList( true );
2952
2953 return 0;
2954}
2955
2956template <bool _useINDI>
2958{
2959 // Unless forced by all, we only do anything if allDefs are not received yet
2960 if( !all && m_allDefsReceived )
2961 return;
2962
2963 callBackIterator it = m_indiSetCallBacks.begin();
2964
2965 int nowFalse = 0;
2966 while( it != m_indiSetCallBacks.end() )
2967 {
2968 if( all || it->second.m_defReceived == false )
2969 {
2970 if( it->second.property )
2971 {
2972 try
2973 {
2974 m_indiDriver->sendGetProperties( *( it->second.property ) );
2975 }
2976 catch( const std::exception &e )
2977 {
2978 log<software_error>( { __FILE__,
2979 __LINE__,
2980 "exception caught from sendGetProperties for " +
2981 it->second.property->getName() + ": " + e.what() } );
2982 }
2983 }
2984
2985 it->second.m_defReceived = false;
2986 ++nowFalse;
2987 }
2988 ++it;
2989 }
2990 if( nowFalse != 0 )
2991 m_allDefsReceived = false;
2992 if( nowFalse == 0 )
2993 m_allDefsReceived = true;
2994}
2995
2996template <bool _useINDI>
2997void MagAOXApp<_useINDI>::handleDefProperty( const pcf::IndiProperty &ipRecv )
2998{
2999 handleSetProperty( ipRecv ); // We have the same response to both Def and Set.
3000}
3001
3002template <bool _useINDI>
3003void MagAOXApp<_useINDI>::handleGetProperties( const pcf::IndiProperty &ipRecv )
3004{
3005 if( !m_useINDI )
3006 return;
3007 if( m_indiDriver == nullptr )
3008 return;
3009
3010 // Ignore if not our device
3011 if( ipRecv.hasValidDevice() && ipRecv.getDevice() != m_indiDriver->getName() )
3012 {
3013 return;
3014 }
3015
3016 // Send all properties if requested.
3017 if( !ipRecv.hasValidName() )
3018 {
3019 callBackIterator it = m_indiNewCallBacks.begin();
3020
3021 while( it != m_indiNewCallBacks.end() )
3022 {
3023 if( it->second.property )
3024 {
3025 try
3026 {
3027 m_indiDriver->sendDefProperty( *( it->second.property ) );
3028 }
3029 catch( const std::exception &e )
3030 {
3031 log<software_error>( { __FILE__,
3032 __LINE__,
3033 "exception caught from sendDefProperty for " +
3034 it->second.property->getName() + ": " + e.what() } );
3035 }
3036 }
3037 ++it;
3038 }
3039
3040 // This is a possible INDI server restart, so we re-register for all notifications.
3041 sendGetPropertySetList( true );
3042
3043 return;
3044 }
3045
3046 // Check if we actually have this.
3047 if( m_indiNewCallBacks.count( ipRecv.createUniqueKey() ) == 0 )
3048 {
3049 return;
3050 }
3051
3052 // Otherwise send just the requested property, if property is not null
3053 if( m_indiNewCallBacks[ipRecv.createUniqueKey()].property )
3054 {
3055 try
3056 {
3057 m_indiDriver->sendDefProperty( *( m_indiNewCallBacks[ipRecv.createUniqueKey()].property ) );
3058 }
3059 catch( const std::exception &e )
3060 {
3061 log<software_error>( { __FILE__,
3062 __LINE__,
3063 "exception caught from sendDefProperty for " +
3064 m_indiNewCallBacks[ipRecv.createUniqueKey()].property->getName() + ": " +
3065 e.what() } );
3066 }
3067 }
3068 return;
3069}
3070
3071template <bool _useINDI>
3072void MagAOXApp<_useINDI>::handleNewProperty( const pcf::IndiProperty &ipRecv )
3073{
3074 if( !m_useINDI )
3075 return;
3076 if( m_indiDriver == nullptr )
3077 return;
3078
3079 // Check if this is a valid name for us.
3080 if( m_indiNewCallBacks.count( ipRecv.createUniqueKey() ) == 0 )
3081 {
3082 log<software_debug>( { __FILE__, __LINE__, "invalid NewProperty request for " + ipRecv.createUniqueKey() } );
3083 return;
3084 }
3085
3086 int ( *callBack )( void *, const pcf::IndiProperty & ) = m_indiNewCallBacks[ipRecv.createUniqueKey()].callBack;
3087
3088 if( callBack )
3089 callBack( this, ipRecv );
3090
3091 log<software_debug>( { __FILE__, __LINE__, "NewProperty callback null for " + ipRecv.createUniqueKey() } );
3092
3093 return;
3094}
3095
3096template <bool _useINDI>
3097void MagAOXApp<_useINDI>::handleSetProperty( const pcf::IndiProperty &ipRecv )
3098{
3099 if( !m_useINDI )
3100 return;
3101 if( m_indiDriver == nullptr )
3102 return;
3103
3104 std::string key = ipRecv.createUniqueKey();
3105
3106 // Check if this is valid
3107 if( m_indiSetCallBacks.count( key ) > 0 )
3108 {
3109 m_indiSetCallBacks[key].m_defReceived = true; // record that we got this Def/Set
3110
3111 // And call the callback
3112 int ( *callBack )( void *, const pcf::IndiProperty & ) = m_indiSetCallBacks[key].callBack;
3113 if( callBack )
3114 callBack( this, ipRecv );
3115
3116 ///\todo log an error here because callBack should not be null
3117 }
3118 else
3119 {
3120 ///\todo log invalid SetProperty request.
3121 }
3122
3123 return;
3124}
3125
3126template <bool _useINDI>
3127template <typename T>
3128void MagAOXApp<_useINDI>::updateIfChanged( pcf::IndiProperty &p,
3129 const std::string &el,
3130 const T &newVal,
3131 pcf::IndiProperty::PropertyStateType ipState )
3132{
3133 if( !_useINDI )
3134 return;
3135
3136 if( !m_indiDriver )
3137 return;
3138
3139 indi::updateIfChanged( p, el, newVal, m_indiDriver, ipState );
3140}
3141
3142template <bool _useINDI>
3143void MagAOXApp<_useINDI>::updateIfChanged( pcf::IndiProperty &p,
3144 const std::string &el,
3145 const char *newVal,
3146 pcf::IndiProperty::PropertyStateType ipState )
3147{
3148 updateIfChanged<std::string>( p, el, std::string( newVal ), ipState );
3149}
3150
3151template <bool _useINDI>
3153 const std::string &el,
3154 const pcf::IndiElement::SwitchStateType &newVal,
3155 pcf::IndiProperty::PropertyStateType ipState )
3156{
3157 if( !_useINDI )
3158 return;
3159
3160 if( !m_indiDriver )
3161 return;
3162
3163 indi::updateSwitchIfChanged( p, el, newVal, m_indiDriver, ipState );
3164}
3165
3166template <bool _useINDI>
3167template <typename T>
3168void MagAOXApp<_useINDI>::updateIfChanged( pcf::IndiProperty &p,
3169 const std::string &el,
3170 const std::vector<T> &newVals,
3171 pcf::IndiProperty::PropertyStateType ipState )
3172{
3173 if( !_useINDI )
3174 return;
3175
3176 if( !m_indiDriver )
3177 return;
3178
3179 std::vector<std::string> descriptors( newVals.size(), el );
3180 for( size_t index = 0; index < newVals.size(); ++index )
3181 {
3182 descriptors[index] += std::to_string( index );
3183 }
3184 indi::updateIfChanged( p, descriptors, newVals, m_indiDriver );
3185}
3186
3187template <bool _useINDI>
3188template <typename T>
3189void MagAOXApp<_useINDI>::updateIfChanged( pcf::IndiProperty &p,
3190 const std::vector<std::string> &els,
3191 const std::vector<T> &newVals,
3192 pcf::IndiProperty::PropertyStateType newState )
3193{
3194 if( !_useINDI )
3195 return;
3196
3197 if( !m_indiDriver )
3198 return;
3199
3200 indi::updateIfChanged( p, els, newVals, m_indiDriver, newState );
3201}
3202
3203template <bool _useINDI>
3204template <typename T>
3205void MagAOXApp<_useINDI>::updatesIfChanged( pcf::IndiProperty &p,
3206 const std::vector<const char *> &els,
3207 const std::vector<T> &newVals,
3208 pcf::IndiProperty::PropertyStateType newState )
3209{
3210 if( !_useINDI )
3211 {
3212 return;
3213 }
3214
3215 if( !m_indiDriver )
3216 {
3217 return;
3218 }
3219
3220 indi::updatesIfChanged( p, els, newVals, m_indiDriver, newState );
3221}
3222
3223template <bool _useINDI>
3224template <typename T>
3225int MagAOXApp<_useINDI>::indiTargetUpdate( pcf::IndiProperty &localProperty,
3226 T &localTarget,
3227 const pcf::IndiProperty &remoteProperty,
3228 bool setBusy )
3229{
3230 if( remoteProperty.createUniqueKey() != localProperty.createUniqueKey() )
3231 {
3232 return log<text_log, -1>( "INDI property names do not match", logPrio::LOG_ERROR );
3233 }
3234
3235 if( !( remoteProperty.find( "target" ) || remoteProperty.find( "current" ) ) )
3236 {
3237 return log<text_log, -1>( "no target or current element in INDI property", logPrio::LOG_ERROR );
3238 }
3239
3240 bool set = false;
3241
3242 if( remoteProperty.find( "target" ) )
3243 {
3244 localTarget = remoteProperty["target"].get<T>();
3245 set = true;
3246 }
3247
3248 if( !set )
3249 {
3250 if( remoteProperty.find( "current" ) )
3251 {
3252 localTarget = remoteProperty["current"].get<T>();
3253 set = true;
3254 }
3255 }
3256
3257 if( !set )
3258 {
3259 return log<text_log, -1>( "no non-empty value found in INDI property", logPrio::LOG_ERROR );
3260 }
3261
3262 if( setBusy )
3263 {
3264 updateIfChanged( localProperty, "target", localTarget, INDI_BUSY );
3265 }
3266 else
3267 {
3268 updateIfChanged( localProperty, "target", localTarget );
3269 }
3270
3271 return 0;
3272}
3273
3274/// \todo move propType to an INDI utils file, and document.
3275
3276template <typename T>
3277pcf::IndiProperty::Type propType()
3278{
3279 return pcf::IndiProperty::Unknown;
3280}
3281
3282template <>
3283pcf::IndiProperty::Type propType<char *>();
3284
3285template <>
3286pcf::IndiProperty::Type propType<std::string>();
3287
3288template <>
3289pcf::IndiProperty::Type propType<int>();
3290
3291template <>
3292pcf::IndiProperty::Type propType<double>();
3293
3294template <bool _useINDI>
3295template <typename T>
3296int MagAOXApp<_useINDI>::sendNewProperty( const pcf::IndiProperty &ipSend, const std::string &el, const T &newVal )
3297{
3298 if( !_useINDI )
3299 return 0;
3300
3301 if( !m_indiDriver )
3302 {
3303 log<software_error>( { __FILE__, __LINE__, "INDI communications not initialized." } );
3304 return -1;
3305 }
3306 pcf::IndiProperty ipToSend = ipSend;
3307
3308 try
3309 {
3310 ipToSend[el].setValue( newVal );
3311 }
3312 catch( ... )
3313 {
3314 log<software_error>(
3315 { __FILE__, __LINE__, "Exception caught setting " + ipSend.createUniqueKey() + "." + el } );
3316 return -1;
3317 }
3318
3319 int rv = m_indiDriver->sendNewProperty( ipToSend );
3320 if( rv < 0 )
3321 {
3322 log<software_error>( { __FILE__, __LINE__ } );
3323 return -1;
3324 }
3325
3326 return 0;
3327}
3328
3329template <bool _useINDI>
3330int MagAOXApp<_useINDI>::sendNewProperty( const pcf::IndiProperty &ipSend )
3331{
3332 if( !_useINDI )
3333 return 0;
3334
3335 if( !m_indiDriver )
3336 {
3337 return log<software_error, -1>( { __FILE__, __LINE__, "INDI communications not initialized." } );
3338 }
3339
3340 if( m_indiDriver->sendNewProperty( ipSend ) < 0 )
3341 {
3342 return log<software_error, -1>( { __FILE__, __LINE__ } );
3343 }
3344
3345 return 0;
3346}
3347
3348template <bool _useINDI>
3349int MagAOXApp<_useINDI>::sendNewStandardIndiToggle( const std::string &device, const std::string &property, bool onoff )
3350{
3351 if( !_useINDI )
3352 return 0;
3353
3354 pcf::IndiProperty ipSend( pcf::IndiProperty::Switch );
3355
3356 try
3357 {
3358 ipSend.setDevice( device );
3359 ipSend.setName( property );
3360 ipSend.add( pcf::IndiElement( "toggle" ) );
3361 }
3362 catch( std::exception &e )
3363 {
3364 return log<software_error, -1>( { __FILE__, __LINE__, std::string( "exception: " ) + e.what() } );
3365 }
3366
3367 if( onoff == false )
3368 {
3369 ipSend["toggle"].setSwitchState( pcf::IndiElement::Off );
3370 }
3371 else
3372 {
3373 ipSend["toggle"].setSwitchState( pcf::IndiElement::On );
3374 }
3375
3376 if( sendNewProperty( ipSend ) < 0 )
3377 {
3378 return log<software_error, -1>(
3379 { __FILE__, __LINE__, "sendNewProperty failed for " + device + "." + property } );
3380 }
3381
3382 return 0;
3383}
3384
3385template <bool _useINDI>
3386int MagAOXApp<_useINDI>::st_newCallBack_clearFSMAlert( void *app, const pcf::IndiProperty &ipRecv )
3387{
3388 return static_cast<MagAOXApp<_useINDI> *>( app )->newCallBack_clearFSMAlert( ipRecv );
3389}
3390
3391template <bool _useINDI>
3393{
3394
3395 if( ipRecv.createUniqueKey() != m_indiP_clearFSMAlert.createUniqueKey() )
3396 {
3397 return log<software_error, -1>( { __FILE__, __LINE__, "wrong indi property received" } );
3398 }
3399
3400 if( !ipRecv.find( "request" ) )
3401 return 0;
3402
3403 if( ipRecv["request"].getSwitchState() == pcf::IndiElement::On )
3404 {
3405 clearFSMAlert();
3406 updateSwitchIfChanged( m_indiP_clearFSMAlert, "request", pcf::IndiElement::Off, INDI_IDLE );
3407 }
3408
3409 return 0;
3410}
3411
3412template <bool _useINDI>
3414{
3415 return 0;
3416}
3417
3418template <bool _useINDI>
3420{
3421 return 0;
3422}
3423
3424template <bool _useINDI>
3426{
3427 if( !m_powerMgtEnabled || m_powerOnWait == 0 || m_powerOnCounter < 0 )
3428 {
3429 return true;
3430 }
3431
3432 if( m_powerOnCounter * m_loopPause > ( (double)m_powerOnWait ) * 1e9 )
3433 {
3434 return true;
3435 }
3436 else
3437 {
3438 ++m_powerOnCounter;
3439 return false;
3440 }
3441}
3442
3443template <bool _useINDI>
3445{
3446 if( !m_powerMgtEnabled )
3447 return 1;
3448
3449 return m_powerState;
3450}
3451
3452template <bool _useINDI>
3454{
3455 if( !m_powerMgtEnabled )
3456 return 1;
3457
3458 return m_powerTargetState;
3459}
3460
3461template <bool _useINDI>
3462INDI_SETCALLBACK_DEFN( MagAOXApp<_useINDI>, m_indiP_powerChannel )
3463( const pcf::IndiProperty &ipRecv )
3464{
3465 std::string ps;
3466
3467 if( ipRecv.find( m_powerElement ) )
3468 {
3469 ps = ipRecv[m_powerElement].get<std::string>();
3470
3471 if( ps == "On" )
3472 {
3473 m_powerState = 1;
3474 }
3475 else if( ps == "Off" )
3476 {
3477 m_powerState = 0;
3478 }
3479 else
3480 {
3481 m_powerState = -1;
3482 }
3483 }
3484
3485 if( ipRecv.find( m_powerTargetElement ) )
3486 {
3487 ps = ipRecv[m_powerTargetElement].get<std::string>();
3488
3489 if( ps == "On" )
3490 {
3491 m_powerTargetState = 1;
3492 }
3493 else if( ps == "Off" )
3494 {
3495 m_powerTargetState = 0;
3496 }
3497 else
3498 {
3499 m_powerTargetState = -1;
3500 }
3501 }
3502
3503 return 0;
3504}
3505
3506template <bool _useINDI>
3508{
3509 return m_configName;
3510}
3511
3512template <bool _useINDI>
3514{
3515 return m_configDir;
3516}
3517
3518template <bool _useINDI>
3520{
3521 return m_driverInName;
3522}
3523
3524template <bool _useINDI>
3526{
3527 return m_driverOutName;
3528}
3529
3530template <bool _useINDI>
3532{
3533 return m_driverCtrlName;
3534}
3535
3536extern template class MagAOXApp<true>;
3537extern template class MagAOXApp<false>;
3538
3539} // namespace app
3540} // namespace MagAOX
3541
3542/// Error handling wrapper for the threadStart function of the XWCApp
3543/** This should be placed in appLogic for each thread managed by an MagAOXApp.
3544 * On error, this will cause the app to shutdown.
3545 * \see MagAOXApp::threadStart
3546 *
3547 * \param thrdSt [out] (std::thread) The thread object to start executing
3548 * \param thrdInit [in/out] (bool) The thread initilization synchronizer.
3549 * \param thrdId [in/out] (pid_t) The thread pid to be filled in by thrdStart immediately upon call
3550 * \param thrdProp [in/out](pcf::IndiProperty) The INDI property to publish the thread details
3551 * \param thrdPrio [in] (int) The r/t priority to set for this thread
3552 * \param thrdCpuset [in] (const std::string &) the cpuset to place this thread on. Ignored if "".
3553 * \param thrdName [in] (const std::string &) The name of the thread (just for logging)
3554 * \param thrdStart [in] (function) The thread starting function, a static function taking a `this` pointer as
3555 * argument.
3556 */
3557#define XWCAPP_THREAD_START( thrdSt, thrdInit, thrdId, thrdProp, thrdPrio, thrdCpuset, thrdName, thrdStart ) \
3558 if( threadStart( thrdSt, thrdInit, thrdId, thrdProp, thrdPrio, thrdCpuset, thrdName, this, thrdStart ) < 0 ) \
3559 { \
3560 log<software_error>( { __FILE__, __LINE__, "error from threadStart for " #thrdName } ); \
3561 return -1; \
3562 }
3563
3564/// Error handling wrapper for checking on thread status with tryjoin
3565/** This should be placed in appLogic for each thread managed by an MagAOXApp. If the
3566 * thread has exited or otherwise causes an error the app will exit.
3567 *
3568 * \param thrdSt [in] (std::thread) The thread object to start executing
3569 * \param thrdName [in] (const std::string &) The name of the thread (just for logging)
3570 */
3571#define XWCAPP_THREAD_CHECK( thrdSt, thrdName ) \
3572 try \
3573 { \
3574 if( pthread_tryjoin_np( thrdSt.native_handle(), 0 ) == 0 ) \
3575 { \
3576 log<software_error>( { __FILE__, __LINE__, #thrdName " thread has exited" } ); \
3577 return -1; \
3578 } \
3579 } \
3580 catch( ... ) \
3581 { \
3582 log<software_error>( { __FILE__, __LINE__, #thrdName " thread has exited" } ); \
3583 return -1; \
3584 }
3585
3586/// Error handlng wrapper for stopping a thread
3587/** This should be placed in appShutdown for each thread managed by an MagAOXApp.
3588 *
3589 * \param thrdSt [in] (std::thread) The thread object to start executing
3590 */
3591#define XWCAPP_THREAD_STOP( thrdSt ) \
3592 if( thrdSt.joinable() ) \
3593 { \
3594 pthread_kill( thrdSt.native_handle(), SIGUSR1 ); \
3595 try \
3596 { \
3597 thrdSt.join(); \
3598 } \
3599 catch( ... ) \
3600 { \
3601 } \
3602 }
3603
3604#endif // app_MagAOXApp_hpp
Internal class to manage setuid privilege escalation with RAII.
The base-class for MagAO-X applications.
Definition MagAOXApp.hpp:73
void handleDefProperty(const pcf::IndiProperty &ipRecv)
Handler for the DEF INDI properties notification.
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.
static void _handlerSigTerm(int signum, siginfo_t *siginf, void *ucont)
The handler called when SIGTERM, SIGQUIT, or SIGINT is received. Just a wrapper for handlerSigTerm.
void handleSetProperty(const pcf::IndiProperty &ipRecv)
Handler for the set INDI property request.
int sendNewStandardIndiToggle(const std::string &device, const std::string &property, bool onoff)
Send a new property commmand for a standard toggle switch.
std::string m_configName
The name of the configuration file (minus .conf).
Definition MagAOXApp.hpp:83
virtual int onPowerOff()
This method is called when the change to poweroff is detected.
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, const std::string &propName, const pcf::IndiProperty::Type &propType, const pcf::IndiProperty::PropertyPermType &propPerm, const pcf::IndiProperty::PropertyStateType &propState, int(*)(void *, const pcf::IndiProperty &))
Register an INDI property which is exposed for others to request a New Property for.
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.
bool m_allDefsReceived
Flag indicating that all registered Set properties have been updated since last Get.
void handlerSigTerm(int signum, siginfo_t *siginf, void *ucont)
Handles SIGTERM, SIGQUIT, and SIGINT. Sets m_shutdown to 1 and logs the signal.
std::string driverCtrlName()
Get the INDI control FIFO file name.
int createStandardIndiNumber(pcf::IndiProperty &prop, const std::string &name, const T &min, const T &max, const T &step, const std::string &format, const std::string &label="", const std::string &group="")
Create a standard R/W INDI Number property with target and current elements.
virtual int execute()
The execute method implementing the standard main loop. Should not normally be overridden.
static void configLog(const std::string &name, const int &code, const std::string &value, const std::string &source)
Callback for config system logging.
INDI_SETCALLBACK_DECL(MagAOXApp, m_indiP_powerChannel)
int powerState()
Returns the current power state.
std::string pidFileName
The name of the PID file.
void sendGetPropertySetList(bool all=false)
unsigned long m_loopPause
Definition MagAOXApp.hpp:95
void updateIfChanged(pcf::IndiProperty &p, const std::string &el, const char *newVal, pcf::IndiProperty::PropertyStateType ipState=pcf::IndiProperty::Ok)
Update an INDI property element value if it has changed.
int m_shutdown
Flag to signal it's time to shutdown. When not 0, the main loop exits.
std::unordered_map< std::string, indiCallBack > m_indiNewCallBacks
Map to hold the NewProperty indiCallBacks for this App, with fast lookup by property name.
int setEuidReal()
Set the effective user ID to the real value, i.e. the file owner.
void updateIfChanged(pcf::IndiProperty &p, const std::vector< std::string > &els, const std::vector< T > &newVals, pcf::IndiProperty::PropertyStateType newState=pcf::IndiProperty::Ok)
Update an INDI property if values have changed.
virtual int appLogic()=0
This is where derived applications implement their main FSM logic.
pcf::IndiProperty m_indiP_powerChannel
INDI property used to communicate power state.
std::unordered_map< std::string, indiCallBack >::iterator callBackIterator
Iterator type of the indiCallBack map.
int createINDIFIFOS()
Create the INDI FIFOs.
int shutdown()
Get the value of the shutdown flag.
stateCodes::stateCodeT m_state
int createStandardIndiSelectionSw(pcf::IndiProperty &prop, const std::string &name, const std::vector< std::string > &elements, const std::string &label="", const std::string &group="")
int m_powerState
Current power state, 1=On, 0=Off, -1=Unk.
std::string m_powerDevice
The INDI device name of the power controller.
friend class MagAOXApp_test
Definition MagAOXApp.hpp:74
int sendNewProperty(const pcf::IndiProperty &ipSend)
Send a newProperty command to another device (using the INDI Client interface)
void state(const stateCodes::stateCodeT &s, bool stateAlert=false)
Set the current state code.
indiDriver< MagAOXApp > * m_indiDriver
The INDI driver wrapper. Constructed and initialized by execute, which starts and stops communication...
void handleNewProperty(const pcf::IndiProperty &ipRecv)
Handler for the new INDI property request.
int powerStateTarget()
Returns the target power state.
void updateSwitchIfChanged(pcf::IndiProperty &p, const std::string &el, const pcf::IndiElement::SwitchStateType &newVal, pcf::IndiProperty::PropertyStateType ipState=pcf::IndiProperty::Ok)
Update an INDI switch element value if it has changed.
std::string m_powerElement
The INDI element name to monitor for this device's power state.
std::pair< std::string, indiCallBack > callBackValueType
Value type of the indiCallBack map.
std::string m_calibDir
The path to calibration files for MagAOX.
Definition MagAOXApp.hpp:89
std::string configDir()
Get the config directory.
static int st_newCallBack_clearFSMAlert(void *app, const pcf::IndiProperty &ipRecv)
The static callback function to be registered for requesting to clear the FSM alert.
int stateLogged()
Updates and returns the value of m_stateLogged. Will be 0 on first call after a state change,...
int newCallBack_clearFSMAlert(const pcf::IndiProperty &ipRecv)
The callback called by the static version, to actually process the FSM Alert Clear request.
static logManagerT m_log
MagAOXApp(const std::string &git_sha1, const bool git_modified)
Public c'tor. Handles uid, logs git repo status, and initializes static members.
static int log(const typename logT::messageT &msg, logPrioT level=logPrio::LOG_DEFAULT)
Make a log entry.
void handleGetProperties(const pcf::IndiProperty &ipRecv)
Handler for the get INDI properties request.
virtual int appShutdown()=0
Any tasks to perform after main loop exit go here.
uid_t m_euidReal
The real user id of the proces (i.e. the lower privileged id of the user)
std::string m_configDir
The path to configuration files for MagAOX.
Definition MagAOXApp.hpp:85
MagAOXApp()=delete
Default c'tor is deleted.
int createROIndiNumber(pcf::IndiProperty &prop, const std::string &propName, const std::string &propLabel="", const std::string &propGroup="")
Create a ReadOnly INDI Number property.
int registerIndiPropertyReadOnly(pcf::IndiProperty &prop)
Register an INDI property which is read only.
void updateIfChanged(pcf::IndiProperty &p, const std::string &el, const std::vector< T > &newVals, pcf::IndiProperty::PropertyStateType ipState=pcf::IndiProperty::Ok)
Update an INDI property if values have changed.
std::string driverOutName()
Get the INDI output FIFO file name.
virtual void setDefaults(int argc, char **argv)
Set the paths for config files.
std::pair< callBackIterator, bool > callBackInsertResult
Return type of insert on the indiCallBack map.
pcf::IndiProperty m_indiP_clearFSMAlert
indi Property to clear an FSM alert.
static constexpr bool m_useINDI
Flag controlling whether INDI is used. If false, then no INDI code executes.
virtual int whilePowerOff()
This method is called while the power is off, once per FSM loop.
bool powerOnWaitElapsed()
This method tests whether the power on wait time has elapsed.
logger::logManager< MagAOXApp< _useINDI >, logFileRaw > logManagerT
The log manager type.
Definition MagAOXApp.hpp:78
pid_t m_pid
This process's PID.
std::string m_powerChannel
The INDI property name of the channel controlling this device's power.
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::string secretsPath
Path to the secrets directory, where passwords, etc, are stored.
Definition MagAOXApp.hpp:93
unsigned long m_powerOnWait
Time in sec to wait for device to boot after power on.
std::string m_powerTargetElement
The INDI element name to monitor for this device's power state.
int clearFSMAlert()
Clear the FSM alert state.
virtual void setupBasicConfig()
The basic MagAO-X configuration setup method. Should not normally be overridden.
std::mutex m_indiMutex
Mutex for locking INDI communications.
std::string m_configBase
The name of a base config class for this app (minus .conf).
Definition MagAOXApp.hpp:87
std::string configName()
Get the config name.
int createStandardIndiSelectionSw(pcf::IndiProperty &prop, const std::string &name, const std::vector< std::string > &elements, const std::vector< std::string > &elementLabels, const std::string &label="", const std::string &group="")
Create a standard R/W INDI selection (one of many) switch with vector of elements and element labels.
std::string m_driverOutName
Full path name of the INDI driver output FIFO.
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.
std::string sysPath
The path to the system directory, for PID file, etc.
Definition MagAOXApp.hpp:91
int setSigTermHandler()
Sets the handler for SIGTERM, SIGQUIT, and SIGINT.
virtual int appStartup()=0
Any tasks to perform prior to the main event loop go here.
int startINDI()
Start INDI Communications.
virtual void checkConfig()
Check for unused and unrecognized config options and settings.
int sendNewProperty(const pcf::IndiProperty &ipSend, const std::string &el, const T &newVal)
Send a newProperty command to another device (using the INDI Client interface)
int createStandardIndiText(pcf::IndiProperty &prop, const std::string &propName, const std::string &label="", const std::string &group="")
Create a standard R/W INDI Text property with target and current elements.
~MagAOXApp() noexcept(true)
virtual void loadBasicConfig()
The basic MagAO-X configuration processing method. Should not normally be overridden.
void logMessage(bufferPtrT &b)
Handle a log message from the logging system.
void updatesIfChanged(pcf::IndiProperty &p, const std::vector< const char * > &els, const std::vector< T > &newVals, pcf::IndiProperty::PropertyStateType newState=pcf::IndiProperty::Ok)
int registerIndiPropertySet(pcf::IndiProperty &prop, const std::string &devName, const std::string &propName, int(*)(void *, const pcf::IndiProperty &))
Register an INDI property which is monitored for updates from others.
int registerIndiPropertyNew(pcf::IndiProperty &prop, const std::string &propName, const pcf::IndiProperty::Type &propType, const pcf::IndiProperty::PropertyPermType &propPerm, const pcf::IndiProperty::PropertyStateType &propState, const pcf::IndiProperty::SwitchRuleType &propRule, int(*)(void *, const pcf::IndiProperty &))
Register an INDI property which is exposed for others to request a New Property for,...
int registerIndiPropertyReadOnly(pcf::IndiProperty &prop, const std::string &propName, const pcf::IndiProperty::Type &propType, const pcf::IndiProperty::PropertyPermType &propPerm, const pcf::IndiProperty::PropertyStateType &propState)
Register an INDI property which is read only.
std::string driverInName()
Get the INDI input FIFO file name.
int indiTargetUpdate(pcf::IndiProperty &localProperty, T &localTarget, const pcf::IndiProperty &remoteProperty, bool setBusy=true)
Get the target element value from an new property.
int unlockPID()
Remove the PID file.
static MagAOXApp * m_self
std::string MagAOXPath
The base path of the MagAO-X system.
Definition MagAOXApp.hpp:81
std::string m_driverCtrlName
Full path name of the INDI driver control FIFO.
int m_powerTargetState
Current target power state, 1=On, 0=Off, -1=Unk.
pcf::IndiProperty m_indiP_state
indi Property to report the application state.
int setEuidCalled()
Set the effective user ID to the called value, i.e. the highest possible.
uid_t m_suid
The save-set user id of the process.
std::unordered_map< std::string, indiCallBack > m_indiSetCallBacks
Map to hold the SetProperty indiCallBacks for this App, with fast lookup by property name.
std::string m_driverInName
Full path name of the INDI driver input FIFO.
A class to manage raw binary log files.
#define MAGAOX_RT_SCHED_POLICY
The real-time scheduling policy.
Definition config.hpp:21
#define MAGAOX_default_loopPause
The default application loopPause.
Definition defaults.hpp:49
#define MAGAOX_calibRelPath
The relative path to the calibration files.
Definition paths.hpp:36
#define MAGAOX_driverFIFORelPath
The relative path to the INDI driver FIFOs.
Definition paths.hpp:85
#define MAGAOX_configRelPath
The relative path to the configuration files.
Definition paths.hpp:29
#define MAGAOX_logRelPath
The relative path to the log directory.
Definition paths.hpp:50
#define MAGAOX_sysRelPath
The relative path to the system directory.
Definition paths.hpp:64
#define MAGAOX_path
The path to the MagAO-X system files.
Definition paths.hpp:22
#define MAGAOX_secretsRelPath
The relative path to the secrets directory. Used for storing passwords, etc.
Definition paths.hpp:71
#define MAGAOX_cpusetPath
The absolute path the the cpuset mount point.
Definition paths.hpp:99
#define MAGAOX_env_path
Environment variable setting the MagAO-X path.
#define MAGAOX_env_calib
Environment variable setting the relative calib path.
#define MAGAOX_env_config
Environment variable setting the relative config path.
#define MAGAOX_env_cpuset
Environment variable setting the relative calib path.
#define INDI_SETCALLBACK_DEFN(class, prop)
Define the callback for a set property request.
#define INDI_SETCALLBACK(prop)
Get the name of the static callback wrapper for a set property.
int8_t logPrioT
The type of the log priority code.
Definition logDefs.hpp:21
std::shared_ptr< char > bufferPtrT
The log entry buffer smart pointer.
Definition logHeader.hpp:58
static int logLevel(bufferPtrT &logBuffer, const logPrioT &lvl)
Set the level of a log entry in a logBuffer header.
static int timespec(bufferPtrT &logBuffer, const timespecX &ts)
Set the timespec of a log entry.
@ OPERATING
The device is operating, other than homing.
@ POWEROFF
The device power is off.
@ NODEVICE
No device exists for the application to control.
@ SHUTDOWN
The application has shutdown, set just after calling appShutdown().
@ NOTHOMED
The device has not been homed.
@ HOMING
The device is homing.
@ FAILURE
The application has failed, should be used when m_shutdown is set for an error.
@ CONFIGURING
The application is configuring the device.
@ 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.
@ LOGGEDIN
The application has logged into the device or service.
@ UNINITIALIZED
The application is unitialized, the default.
@ INITIALIZED
The application has been initialized, set just before calling appStartup().
@ POWERON
The device power is on.
int th_seteuid(uid_t euid)
Sets the effective user id of the calling thread, rather than the whole process.
Definition thSetuid.cpp:18
MagAO-X INDI Driver Wrapper.
Macros for INDI.
MagAO-X INDI Utilities.
#define INDI_IDLE
Definition indiUtils.hpp:27
#define INDI_BUSY
Definition indiUtils.hpp:29
#define INDI_ALERT
Definition indiUtils.hpp:30
#define INDI_OK
Definition indiUtils.hpp:28
void updateIfChanged(pcf::IndiProperty &p, const std::string &el, const T &newVal, indiDriverT *indiDriver, pcf::IndiProperty::PropertyStateType newState=pcf::IndiProperty::Ok)
Update the value of the INDI element, but only if it has changed.
Definition indiUtils.hpp:92
void updateSwitchIfChanged(pcf::IndiProperty &p, const std::string &el, const pcf::IndiElement::SwitchStateType &newVal, indiDriverT *indiDriver, pcf::IndiProperty::PropertyStateType newState=pcf::IndiProperty::Ok)
Update the value of the INDI element, but only if it has changed.
void updatesIfChanged(pcf::IndiProperty &p, const elVecT &els, const std::vector< T > &newVals, indiDriverT *indiDriver, pcf::IndiProperty::PropertyStateType newState=pcf::IndiProperty::Ok)
Update the elements of an INDI propery, but only if there has been a change in at least one.
pcf::IndiProperty::Type propType< int >()
Definition MagAOXApp.cpp:31
std::stringstream msg
pcf::IndiProperty::Type propType< std::string >()
Definition MagAOXApp.cpp:25
pcf::IndiProperty::Type propType< char * >()
Definition MagAOXApp.cpp:19
void sigUsr1Handler(int signum, siginfo_t *siginf, void *ucont)
Empty signal handler. SIGUSR1 is used to interrupt sleep in various threads.
Definition MagAOXApp.cpp:42
pcf::IndiProperty::Type propType< double >()
Definition MagAOXApp.cpp:37
const pcf::IndiProperty & ipRecv
updateIfChanged(m_indiP_angle, "target", m_angle)
pcf::IndiProperty::Type propType()
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_WARNING
A condition has occurred which may become an error, but the process continues.
static constexpr logPrioT LOG_ERROR
An error has occured which the software will attempt to correct.
static constexpr logPrioT LOG_DEBUG
Used for debugging.
MagAO-X Application States.
Structure to hold the call-back details for handling INDI communications.
int(* callBack)(void *, const pcf::IndiProperty &)
int16_t stateCodeT
The type of the state code.
static std::string codeText(const stateCodeT &stateCode)
Get an ASCII string corresponding to an application stateCode.
The type of the input message.
Definition git_state.hpp:36
The standard MagAOX log manager, used for both process logs and telemetry streams.
Software ERR log entry.
A simple text log, a string-type log.
Definition text_log.hpp:24
A fixed-width timespec structure.
Definition timespecX.hpp:35
nanosecT time_ns
Nanoseconds.
Definition timespecX.hpp:37
secT time_s
Time since the Unix epoch.
Definition timespecX.hpp:36