11#ifndef app_MagAOXApp_hpp
12#define app_MagAOXApp_hpp
16#include <sys/syscall.h>
24#include <unordered_map>
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>
32#include "../common/environment.hpp"
33#include "../common/paths.hpp"
34#include "../common/defaults.hpp"
35#include "../common/config.hpp"
37#include "../logger/logFileRaw.hpp"
38#include "../logger/logManager.hpp"
40#include "../sys/thSetuid.hpp"
47using namespace mx::app;
71template <
bool _useINDI = true>
278 const std::
string &value,
327 bool m_elevated{
false };
451 template <
class thisPtr,
class Function>
456 pcf::IndiProperty &
thProp,
458 const std::string &
cpuset,
551 pcf::IndiProperty *
property{ 0 };
603 const std::string &
label =
"",
605 const std::string &
group =
""
616 const std::string &
elName,
619 const std::string &
elLabel =
""
627 template <
typename T>
629 const std::string &name,
636 const std::string &format,
639 const std::string &
label =
"",
641 const std::string &
group =
""
663 const std::string &name,
664 const std::string &
label =
"",
666 const std::string &
group =
""
676 const std::string &name,
677 const std::string &
label =
"",
679 const std::string &
group =
""
689 const std::string &name,
690 const std::vector<std::string> &elements,
694 const std::string &
label =
"",
696 const std::string &
group =
""
707 const std::string &name,
708 const std::vector<std::string> &elements,
710 const std::string &
label =
"",
712 const std::string &
group =
""
737 const pcf::IndiProperty::Type &
propType,
739 const pcf::IndiProperty::PropertyPermType &
propPerm,
742 const pcf::IndiProperty::PropertyStateType &
propState
755 int ( * )(
void *,
const pcf::IndiProperty & )
767 pcf::IndiProperty &prop,
769 const pcf::IndiProperty::Type &
propType,
770 const pcf::IndiProperty::PropertyPermType &
propPerm,
771 const pcf::IndiProperty::PropertyStateType &
propState,
772 int ( * )(
void *,
const pcf::IndiProperty & )
783 pcf::IndiProperty &prop,
785 const pcf::IndiProperty::Type &
propType,
786 const pcf::IndiProperty::PropertyPermType &
propPerm,
787 const pcf::IndiProperty::PropertyStateType &
propState,
788 const pcf::IndiProperty::SwitchRuleType &
propRule,
789 int ( * )(
void *,
const pcf::IndiProperty & )
800 pcf::IndiProperty &prop,
803 int ( * )(
void *,
const pcf::IndiProperty & )
864 template <
typename T>
866 const std::string &el,
868 pcf::IndiProperty::PropertyStateType
ipState = pcf::IndiProperty::Ok );
883 const std::string &el,
885 pcf::IndiProperty::PropertyStateType
ipState = pcf::IndiProperty::Ok );
894 const std::string &el,
895 const pcf::IndiElement::SwitchStateType &
newVal,
896 pcf::IndiProperty::PropertyStateType
ipState = pcf::IndiProperty::Ok );
910 template <
typename T>
912 pcf::IndiProperty &p,
913 const std::string &el,
915 pcf::IndiProperty::PropertyStateType
ipState = pcf::IndiProperty::Ok
927 template <
typename T>
929 const std::vector<std::string> &
els,
931 pcf::IndiProperty::PropertyStateType
newState =
932 pcf::IndiProperty::Ok
935 template <
typename T>
937 const std::vector<const char *> &
els,
939 pcf::IndiProperty::PropertyStateType
newState =
940 pcf::IndiProperty::Ok
948 template <
typename T>
961 template <
typename T>
963 const std::string &el,
980 const std::string &property,
997 const pcf::IndiProperty &
ipRecv
1132template <
bool _useINDI>
1136template <
bool _useINDI>
1139template <
bool _useINDI>
1142 if( m_self !=
nullptr )
1144 std::cerr <<
"Attempt to instantiate 2nd MagAOXApp. Exiting immediately.\n";
1151 getresuid( &m_euidReal, &m_euidCalled, &m_suid );
1154 m_log.parent(
this );
1157 config.m_sources =
true;
1158 config.configLog = configLog;
1170 if( MXLIB_UNCOMP_REPO_MODIFIED )
1176 log<git_state>(
git_state::messageT(
"mxlib", MXLIB_UNCOMP_CURRENT_SHA1, MXLIB_UNCOMP_REPO_MODIFIED ), gl );
1179template <
bool _useINDI>
1183 delete m_indiDriver;
1184 m_log.parent(
nullptr );
1189template <
bool _useINDI>
1195template <
bool _useINDI>
1204 MagAOXPath = tmpstr;
1217 m_configDir = MagAOXPath +
"/" + tmpstr;
1218 m_configPathGlobal = m_configDir +
"/magaox.conf";
1226 m_calibDir = MagAOXPath +
"/" + tmpstr;
1231 m_log.logPath( tmpstr );
1239 secretsPath = tmpstr;
1245 m_cpusetPath = tmpstr;
1249 if( m_configBase !=
"" )
1252 m_configPathUser = m_configDir +
"/" + m_configBase +
".conf";
1264 "The name of the application and its device name in INDI (if used), specifies the config file in the XWC config directory." );
1266 config.parseCommandLine( argc, argv,
"name" );
1267 config( m_configName,
"name" );
1269 if( m_configName ==
"" )
1271 m_configName = mx::ioutils::pathStem( invokedName );
1274 log<text_log>(
"Configuration Error: Application name (-n --name) not set." );
1280 m_configPathLocal = m_configDir +
"/" + m_configName +
".conf";
1283 if( registerIndiPropertyNew(
1284 m_indiP_state,
"fsm", pcf::IndiProperty::Text, pcf::IndiProperty::ReadOnly, pcf::IndiProperty::Idle, 0 ) <
1287 log<software_error>( { __FILE__, __LINE__,
"failed to register read only fsm_state property" } );
1290 m_indiP_state.add( pcf::IndiElement(
"state" ) );
1292 createStandardIndiRequestSw( m_indiP_clearFSMAlert,
"fsm_clear_alert",
"Clear FSM Alert",
"FSM" );
1293 if( registerIndiPropertyNew( m_indiP_clearFSMAlert, st_newCallBack_clearFSMAlert ) < 0 )
1295 log<software_error>( { __FILE__, __LINE__,
"failed to register new fsm_alert property" } );
1301template <
bool _useINDI>
1305 config.add(
"config.validate",
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." );
1316 config.add(
"loopPause",
1324 "The main loop pause time in ns" );
1327 "ignore_git",
"",
"ignore-git", argType::True,
"",
"",
false,
"bool",
"set to true to ignore git status to prevent the fsm_alert" );
1330 m_log.setupConfig( config );
1332 if( m_powerMgtEnabled )
1334 if( _useINDI ==
false )
1337 log<software_critical>( { __FILE__, __LINE__,
"power management is enabled but we are not using INDI" } );
1342 config.add(
"power.device",
1350 "Device controlling power for this app's device (INDI name)." );
1351 config.add(
"power.channel",
1359 "Channel on device for this app's device (INDI name)." );
1360 config.add(
"power.element",
1368 "INDI power state element name. Default is \"state\", only need to specify if different." );
1369 config.add(
"power.targetElement",
1371 "power.targetElement",
1377 "INDI power target element name. Default is \"target\", only need to specify if different." );
1378 config.add(
"power.powerOnWait",
1380 "power.powerOnWait",
1386 "Time after power-on to wait before continuing [sec]. Default is 0 sec, max is 3600 sec." );
1390template <
bool _useINDI>
1395 config( ig,
"ignore_git" );
1397 if( !ig && m_gitAlert )
1399 m_stateAlert =
true;
1403 if(config.isSet(
"config.validate"))
1405 m_configOnly =
true;
1409 m_log.logName( m_configName );
1410 m_log.loadConfig( config );
1413 config( m_loopPause,
"loopPause" );
1416 if( m_powerMgtEnabled )
1418 config( m_powerDevice,
"power.device" );
1419 config( m_powerChannel,
"power.channel" );
1420 config( m_powerElement,
"power.element" );
1421 config( m_powerTargetElement,
"power.targetElement" );
1423 if( m_powerDevice !=
"" && m_powerChannel !=
"" )
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 ) ) <
1431 log<software_error>( { __FILE__, __LINE__,
"failed to register set property" } );
1440 config( m_powerOnWait,
"power.powerOnWait" );
1441 if( m_powerOnWait > 3600 )
1443 log<text_log>(
"powerOnWait longer than 1 hour. Setting to 0.",
logPrio::LOG_ERROR );
1448template <
bool _useINDI>
1453 for(
auto it = config.m_targets.begin();
it != config.m_targets.end(); ++
it )
1455 if(
it->second.used ==
false )
1457 std::string
msg =
it->second.name;
1458 if( config.m_sources &&
it->second.sources.size() > 0 )
1460 msg +=
" [" +
it->second.sources[0] +
"]";
1468 if( config.m_unusedConfigs.size() > 0 )
1470 for(
auto it = config.m_unusedConfigs.begin();
it != config.m_unusedConfigs.end(); ++
it )
1472 if(
it->second.used ==
true )
1477 std::string
msg =
it->second.name;
1478 if( config.m_sources &&
it->second.sources.size() > 0 )
1480 msg +=
" [" +
it->second.sources[0] +
"]";
1489 if( config.nonOptions.size() > 0 )
1491 for(
size_t n = 0; n < config.nonOptions.size(); ++n )
1493 log<text_log>(
"Unrecognized command line argument: " + config.nonOptions[n],
logPrio::LOG_CRITICAL );
1500 if(m_shutdown ==
true)
1502 std::cerr <<
"\nThere were configuration errors.\n\n";
1506 std::cerr <<
"\nConfiguration is valid.\n\n";
1509 else if(m_shutdown ==
true)
1515template <
bool _useINDI>
1521#ifndef XWC_DISABLE_USER_CHECK
1522 struct stat logstat;
1524 if( stat( m_log.logPath().c_str(), &logstat ) < 0 )
1527 std::cerr <<
"\nCRITICAL: Can not stat the log path.\n\n";
1531 if( logstat.st_uid != geteuid() )
1534 std::cerr <<
"\nCRITICAL: You are running this app as the wrong user.\n\n";
1547 std::cerr <<
"\nCRITICAL: Failed to lock PID. Exiting.\n\n";
1556 m_log.logThreadStart();
1560 while( m_log.logThreadRunning() ==
false && w < 20 )
1563 std::this_thread::sleep_for( std::chrono::duration<unsigned long, std::nano>( 100000000 ) );
1567 if( m_log.logThreadRunning() ==
false )
1572 std::cerr <<
"\nCRITICAL: log thread not running. Exiting.\n\n";
1576 if( unlockPID() < 0 )
1578 log<software_error>( { __FILE__, __LINE__,
"error from unlockPID()" } );
1587 if( m_shutdown == 0 )
1589 if( setSigTermHandler() < 0 )
1593 log<software_critical>( { __FILE__, __LINE__,
"error from setSigTermHandler()" } );
1597 if( unlockPID() < 0 )
1599 log<software_error>( { __FILE__, __LINE__,
"error from unlockPID()" } );
1609 if( m_shutdown == 0 )
1613 if( appStartup() < 0 )
1617 log<software_critical>( { __FILE__, __LINE__,
"error from appStartup()" } );
1621 if( unlockPID() < 0 )
1623 log<software_error>( { __FILE__, __LINE__,
"error from unlockPID()" } );
1631 if( m_useINDI && m_shutdown == 0 )
1633 if( startINDI() < 0 )
1637 log<software_critical>( { __FILE__, __LINE__,
"INDI failed to start." } );
1642 if( appShutdown() < 0 )
1644 log<software_error>( { __FILE__, __LINE__,
"error from appShutdown()" } );
1647 if( unlockPID() < 0 )
1649 log<software_error>( { __FILE__, __LINE__,
"error from unlockPID()" } );
1657 if( m_powerMgtEnabled && m_shutdown == 0 )
1660 while( m_powerState < 0 && !m_shutdown )
1663 if( m_powerState < 0 )
1665 if( !stateLogged() )
1666 log<text_log>(
"waiting for power state" );
1677 if( m_powerState > 0 )
1683 m_powerOnCounter = 0;
1685 if( onPowerOff() < 0 )
1687 log<software_error>( { __FILE__, __LINE__,
"error from onPowerOff()" } );
1702 while( m_shutdown == 0 )
1705 if( m_powerMgtEnabled )
1709 if( m_powerState == 1 )
1711 m_powerOnCounter = 0;
1717 if( m_powerState == 0 )
1720 if( onPowerOff() < 0 )
1722 log<software_error>( { __FILE__, __LINE__,
"error from onPowerOff()" } );
1732 if( !m_powerMgtEnabled || m_powerState > 0 )
1734 if( appLogic() < 0 )
1736 log<software_error>( { __FILE__, __LINE__,
"error from appLogic()" } );
1741 else if( m_powerState == 0 )
1743 if( whilePowerOff() < 0 )
1745 log<software_error>( { __FILE__, __LINE__,
"error from whilePowerOff()" } );
1760 sendGetPropertySetList(
false );
1768 if( m_shutdown == 0 )
1770 std::this_thread::sleep_for( std::chrono::duration<unsigned long, std::nano>( m_loopPause ) );
1774 if( appShutdown() < 0 )
1776 log<software_error>( { __FILE__, __LINE__,
"error from appShutdown()" } );
1782 if( m_indiDriver !=
nullptr )
1784 pcf::IndiProperty ipSend;
1785 ipSend.setDevice( m_configName );
1788 m_indiDriver->sendDelProperty( ipSend );
1790 catch(
const std::exception &e )
1792 log<software_error>(
1793 { __FILE__, __LINE__, std::string(
"exception caught from sendDelProperty: " ) + e.what() } );
1796 m_indiDriver->quitProcess();
1797 m_indiDriver->deactivate();
1798 log<indidriver_stop>();
1801 if( unlockPID() < 0 )
1803 log<software_error>( { __FILE__, __LINE__,
"error from unlockPID()" } );
1810template <
bool _useINDI>
1811template <
typename logT,
int retval>
1814 m_log.template log<logT>(
msg, level );
1818template <
bool _useINDI>
1819template <
typename logT,
int retval>
1822 m_log.template log<logT>( level );
1826template <
bool _useINDI>
1831 logStdFormat( std::cerr, b );
1837 state( m_state,
true );
1840 if( _useINDI && m_indiDriver )
1842 pcf::IndiProperty
msg;
1843 msg.setDevice( m_configName );
1845 std::stringstream logstdf;
1846 logMinStdFormat( logstdf, b );
1848 msg.setMessage( logstdf.str() );
1854 tv.tv_usec = (
long int)( ( (
double)ts.
time_ns ) / 1e3 );
1856 msg.setTimeStamp( pcf::TimeStamp( tv ) );
1860 m_indiDriver->sendMessage(
msg );
1862 catch(
const std::exception &e )
1864 log<software_error>(
1865 { __FILE__, __LINE__, std::string(
"exception caught from sendMessage: " ) + e.what() } );
1870template <
bool _useINDI>
1873 const std::string &value,
1874 const std::string &source )
1876 m_log.template log<config_log>( { name, code, value, source } );
1879template <
bool _useINDI>
1882 struct sigaction act;
1886 act.sa_flags = SA_SIGINFO;
1887 sigemptyset( &set );
1891 if( sigaction( SIGTERM, &act, 0 ) < 0 )
1893 std::string logss =
"Setting handler for SIGTERM failed. Errno says: ";
1894 logss += strerror( errno );
1896 log<software_error>( { __FILE__, __LINE__, errno, 0, logss } );
1902 if( sigaction( SIGQUIT, &act, 0 ) < 0 )
1904 std::string logss =
"Setting handler for SIGQUIT failed. Errno says: ";
1905 logss += strerror( errno );
1907 log<software_error>( { __FILE__, __LINE__, errno, 0, logss } );
1913 if( sigaction( SIGINT, &act, 0 ) < 0 )
1915 std::string logss =
"Setting handler for SIGINT failed. Errno says: ";
1916 logss += strerror( errno );
1918 log<software_error>( { __FILE__, __LINE__, errno, 0, logss } );
1923 log<text_log>(
"Installed SIGTERM/SIGQUIT/SIGINT signal handler.",
logPrio::LOG_DEBUG );
1928template <
bool _useINDI>
1931 m_self->handlerSigTerm( signum, siginf, ucont );
1934template <
bool _useINDI>
1936 siginfo_t *siginf __attribute__( ( unused ) ),
1937 void *ucont __attribute__( ( unused ) ) )
1941 std::string signame;
1945 signame =
"SIGTERM";
1951 signame =
"SIGQUIT";
1957 std::string logss =
"Caught signal ";
1959 logss +=
". Shutting down.";
1961 std::cerr <<
"\n" << logss << std::endl;
1962 log<text_log>( logss );
1966void sigUsr1Handler(
int signum, siginfo_t *siginf,
void *ucont );
1968template <
bool _useINDI>
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 );
1979 log<software_error>( { __FILE__, __LINE__, errno, 0, logss } );
1987template <
bool _useINDI>
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 );
1998 log<software_error>( { __FILE__, __LINE__, errno, 0, logss } );
2006template <
bool _useINDI>
2011 std::string statusDir = sysPath;
2019 if( mkdir( statusDir.c_str(), S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH ) < 0 )
2021 if( errno != EEXIST )
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() } );
2031 statusDir += m_configName;
2033 pidFileName = statusDir +
"/pid";
2038 if( mkdir( statusDir.c_str(), S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH ) < 0 )
2040 if( errno != EEXIST )
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() } );
2050 std::ifstream pidIn;
2051 pidIn.open( pidFileName );
2061 std::stringstream procN;
2062 procN <<
"/proc/" << testPid <<
"/cmdline";
2064 std::ifstream procIn;
2065 std::string pidCmdLine;
2069 procIn.open( procN.str() );
2071 procIn >> pidCmdLine;
2076 log<software_critical>( { __FILE__, __LINE__, 0, 0,
"exception caught testing /proc/pid" } );
2084 size_t invokedPos = pidCmdLine.find( invokedName );
2087 size_t configPos = std::string::npos;
2088 if( invokedPos != std::string::npos )
2089 configPos = pidCmdLine.find( m_configName );
2092 if( invokedPos != std::string::npos && configPos != std::string::npos )
2095 std::stringstream logss;
2096 logss <<
"PID already locked (" << testPid <<
"). Time to die.";
2097 std::cerr << logss.str() << std::endl;
2112 std::ofstream pidOut;
2113 pidOut.open( pidFileName );
2115 if( !pidOut.good() )
2117 log<software_critical>( { __FILE__, __LINE__, errno, 0,
"could not open pid file for writing." } );
2126 std::stringstream logss;
2127 logss <<
"PID (" << m_pid <<
") locked.";
2128 log<text_log>( logss.str() );
2140template <
bool _useINDI>
2148 if( ::remove( pidFileName.c_str() ) < 0 )
2150 log<software_error>(
2151 { __FILE__, __LINE__, errno, 0, std::string(
"Failed to remove PID file: " ) + strerror( errno ) } );
2156 std::stringstream logss;
2157 logss <<
"PID (" << m_pid <<
") unlocked.";
2158 log<text_log>( logss.str() );
2163template <
bool _useINDI>
2164template <
class thisPtr,
class Function>
2168 pcf::IndiProperty &thProp,
2170 const std::string &cpuset,
2171 const std::string &thrdName,
2173 Function &&thrdStart )
2181 thrd = std::thread( thrdStart, thrdThis );
2183 catch(
const std::exception &e )
2185 log<software_error>(
2186 { __FILE__, __LINE__, std::string(
"Exception on " + thrdName +
" thread start: " ) + e.what() } );
2191 log<software_error>( { __FILE__, __LINE__,
"Unkown exception on " + thrdName +
" thread start" } );
2195 if( !thrd.joinable() )
2197 log<software_error>( { __FILE__, __LINE__, thrdName +
" thread did not start" } );
2209 sp.sched_priority = thrdPrio;
2223 rv = pthread_setschedparam( thrd.native_handle(), SCHED_OTHER, &sp );
2228 log<software_error>(
2232 "Setting " + thrdName +
" thread scheduler priority to " + std::to_string( thrdPrio ) +
" failed." } );
2236 log<text_log>( thrdName +
" thread scheduler priority set to " + std::to_string( thrdPrio ) );
2242 for(
int i = 0; i < 10; ++i )
2244 mx::sys::milliSleep( 100 );
2252 return log<
software_error, -1>( { __FILE__, __LINE__, errno,
"tpid for " + thrdName +
" not set." } );
2256 log<text_log>( thrdName +
" thread pid is " + std::to_string( tpid ) );
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 );
2275 std::string cpuFile = m_cpusetPath;
2276 cpuFile +=
"/" + cpuset;
2277 cpuFile +=
"/tasks";
2278 int wfd = open( cpuFile.c_str(), O_WRONLY );
2281 return log<
software_error, -1>( { __FILE__, __LINE__, errno,
"error from open for " + cpuFile } );
2285 snprintf( pids,
sizeof( pids ),
"%d", tpid );
2287 int w = write( wfd, pids, strnlen( pids,
sizeof( pids ) ) );
2288 if( w != (
int)strnlen( pids,
sizeof(pids) ) )
2290 return log<
software_error, -1>( { __FILE__, __LINE__, errno,
"error on write" } );
2304template <
bool _useINDI>
2310template <
bool _useINDI>
2322 log<state_change>( { m_state, s }, lvl );
2328 if( m_stateAlert != stateAlert && stateAlert ==
true )
2330 m_stateAlert = stateAlert;
2335 std::unique_lock<std::mutex>
lock( m_indiMutex,
2339 if(
lock.owns_lock() )
2342 pcf::IndiProperty::PropertyStateType stst =
INDI_IDLE;
2345 if( m_stateAlert ==
true )
2368template <
bool _useINDI>
2371 if( m_stateLogged > 0 )
2374 return m_stateLogged - 1;
2383template <
bool _useINDI>
2386 if( m_stateAlert ==
false )
2388 m_stateAlert =
false;
2391 pcf::IndiProperty::PropertyStateType stst =
INDI_IDLE;
2413template <
bool _useINDI>
2415 const std::string &propName,
2416 const std::string &label,
2417 const std::string &group )
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" ) );
2430 prop.setLabel( label );
2435 prop.setGroup( group );
2441template <
bool _useINDI>
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 )
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 );
2456 if( propLabel !=
"" )
2458 prop.setLabel( propLabel );
2461 if( propGroup !=
"" )
2463 prop.setGroup( propGroup );
2466 prop.add( pcf::IndiElement( elName ) );
2470 prop[elName].setLabel( elLabel );
2476template <
bool _useINDI>
2477template <
typename T>
2479 const std::string &name,
2483 const std::string &format,
2484 const std::string &label,
2485 const std::string &group )
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 );
2498 prop[
"current"].setFormat( format );
2501 prop.add( pcf::IndiElement(
"target" ) );
2502 prop[
"target"].setMin( min );
2503 prop[
"target"].setMax( max );
2504 prop[
"target"].setStep( step );
2507 prop[
"target"].setFormat( format );
2513 prop.setLabel( label );
2518 prop.setGroup( group );
2524template <
bool _useINDI>
2526 const std::string &propName,
2527 const std::string &propLabel,
2528 const std::string &propGroup )
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 );
2537 if( propLabel !=
"" )
2539 prop.setLabel( propLabel );
2542 if( propGroup !=
"" )
2544 prop.setGroup( propGroup );
2550template <
bool _useINDI>
2552 const std::string &name,
2553 const std::string &label,
2554 const std::string &group )
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 );
2564 prop.add( pcf::IndiElement(
"toggle", pcf::IndiElement::Off ) );
2569 prop.setLabel( label );
2574 prop.setGroup( group );
2580template <
bool _useINDI>
2582 const std::string &name,
2583 const std::string &label,
2584 const std::string &group )
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 );
2594 prop.add( pcf::IndiElement(
"request", pcf::IndiElement::Off ) );
2599 prop.setLabel( label );
2604 prop.setGroup( group );
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 )
2618 if( elements.size() == 0 )
2620 return log<
software_error, -1>( { __FILE__, __LINE__,
"elements vector has zero size" } );
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 );
2631 for(
size_t n = 0; n < elements.size(); ++n )
2633 pcf::IndiElement elem = pcf::IndiElement( elements[n], pcf::IndiElement::Off );
2634 elem.setLabel( elementLabels[n] );
2641 prop.setLabel( label );
2646 prop.setGroup( group );
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 )
2659 return createStandardIndiSelectionSw( prop, name, elements, elements, label, group );
2662template <
bool _useINDI>
2669 m_indiNewCallBacks.insert(
callBackValueType( prop.createUniqueKey(), { &prop, nullptr } ) );
2673 if( !result.second )
2676 { __FILE__, __LINE__,
"failed to insert INDI property: " + prop.createUniqueKey() } );
2679 catch( std::exception &e )
2681 return log<
software_error, -1>( { __FILE__, __LINE__, std::string(
"Exception caught: " ) + e.what() } );
2685 return log<
software_error, -1>( { __FILE__, __LINE__,
"Unknown exception caught." } );
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 )
2701 prop = pcf::IndiProperty(
propType );
2702 prop.setDevice( m_configName );
2703 prop.setName( propName );
2704 prop.setPerm( propPerm );
2705 prop.setState( propState );
2711 if( !result.second )
2714 { __FILE__, __LINE__,
"failed to insert INDI property: " + prop.createUniqueKey() } );
2717 catch( std::exception &e )
2719 return log<
software_error, -1>( { __FILE__, __LINE__, std::string(
"Exception caught: " ) + e.what() } );
2723 return log<
software_error, -1>( { __FILE__, __LINE__,
"Unknown exception caught." } );
2728template <
bool _useINDI>
2730 int ( *callBack )(
void *,
const pcf::IndiProperty &
ipRecv ) )
2737 callBackInsertResult result =
2738 m_indiNewCallBacks.insert( callBackValueType( prop.createUniqueKey(), { &prop, callBack } ) );
2740 if( !result.second )
2743 { __FILE__, __LINE__,
"failed to insert INDI property: " + prop.createUniqueKey() } );
2746 catch( std::exception &e )
2748 return log<
software_error, -1>( { __FILE__, __LINE__, std::string(
"Exception caught: " ) + e.what() } );
2752 return log<
software_error, -1>( { __FILE__, __LINE__,
"Unknown exception caught." } );
2758template <
bool _useINDI>
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 ) )
2769 prop = pcf::IndiProperty(
propType );
2770 prop.setDevice( m_configName );
2771 prop.setName( propName );
2772 prop.setPerm( propPerm );
2773 prop.setState( propState );
2775 return registerIndiPropertyNew( prop, callBack );
2778template <
bool _useINDI>
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 ) )
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 );
2799template <
bool _useINDI>
2801 const std::string &devName,
2802 const std::string &propName,
2803 int ( *callBack )(
void *,
const pcf::IndiProperty &
ipRecv ) )
2808 prop = pcf::IndiProperty();
2809 prop.setDevice( devName );
2810 prop.setName( propName );
2813 m_indiSetCallBacks.insert(
callBackValueType( prop.createUniqueKey(), { &prop, callBack } ) );
2817 if( !result.second )
2820 { __FILE__, __LINE__,
"failed to insert INDI property: " + prop.createUniqueKey() } );
2823 catch( std::exception &e )
2825 return log<
software_error, -1>( { __FILE__, __LINE__, std::string(
"Exception caught: " ) + e.what() } );
2829 return log<
software_error, -1>( { __FILE__, __LINE__,
"Unknown exception caught." } );
2835template <
bool _useINDI>
2843 driverFIFOPath +=
"/";
2846 m_driverInName = driverFIFOPath +
"/" + configName() +
".in";
2847 m_driverOutName = driverFIFOPath +
"/" + configName() +
".out";
2848 m_driverCtrlName = driverFIFOPath +
"/" + configName() +
".ctrl";
2854 mode_t prev = umask( 0 );
2857 if( mkfifo( m_driverInName.c_str(), S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP ) != 0 )
2859 if( errno != EEXIST )
2862 log<software_critical>( { __FILE__, __LINE__, errno, 0,
"mkfifo failed" } );
2869 if( mkfifo( m_driverOutName.c_str(), S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP ) != 0 )
2871 if( errno != EEXIST )
2875 log<software_critical>( { __FILE__, __LINE__, errno, 0,
"mkfifo failed" } );
2882 if( mkfifo( m_driverCtrlName.c_str(), S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP ) != 0 )
2884 if( errno != EEXIST )
2888 log<software_critical>( { __FILE__, __LINE__, errno, 0,
"mkfifo failed" } );
2899template <
bool _useINDI>
2906 if( createINDIFIFOS() < 0 )
2914 if( m_indiDriver !=
nullptr )
2916 m_indiDriver->quitProcess();
2917 m_indiDriver->deactivate();
2918 log<indidriver_stop>();
2919 delete m_indiDriver;
2920 m_indiDriver =
nullptr;
2927 log<software_critical>( { __FILE__, __LINE__, 0, 0,
"INDI Driver construction exception." } );
2932 if( m_indiDriver ==
nullptr )
2934 log<software_critical>( { __FILE__, __LINE__, 0, 0,
"INDI Driver construction failed." } );
2939 if( m_indiDriver->good() ==
false )
2941 log<software_critical>( { __FILE__, __LINE__, 0, 0,
"INDI Driver failed to open FIFOs." } );
2942 delete m_indiDriver;
2943 m_indiDriver =
nullptr;
2948 m_indiDriver->activate();
2949 log<indidriver_start>();
2951 sendGetPropertySetList(
true );
2956template <
bool _useINDI>
2960 if( !all && m_allDefsReceived )
2966 while(
it != m_indiSetCallBacks.end() )
2968 if( all ||
it->second.m_defReceived ==
false )
2970 if(
it->second.property )
2974 m_indiDriver->sendGetProperties( *(
it->second.property ) );
2976 catch(
const std::exception &e )
2978 log<software_error>( { __FILE__,
2980 "exception caught from sendGetProperties for " +
2981 it->second.property->getName() +
": " + e.what() } );
2985 it->second.m_defReceived =
false;
2991 m_allDefsReceived =
false;
2993 m_allDefsReceived =
true;
2996template <
bool _useINDI>
2999 handleSetProperty(
ipRecv );
3002template <
bool _useINDI>
3007 if( m_indiDriver ==
nullptr )
3011 if(
ipRecv.hasValidDevice() &&
ipRecv.getDevice() != m_indiDriver->getName() )
3017 if( !
ipRecv.hasValidName() )
3021 while(
it != m_indiNewCallBacks.end() )
3023 if(
it->second.property )
3027 m_indiDriver->sendDefProperty( *(
it->second.property ) );
3029 catch(
const std::exception &e )
3031 log<software_error>( { __FILE__,
3033 "exception caught from sendDefProperty for " +
3034 it->second.property->getName() +
": " + e.what() } );
3041 sendGetPropertySetList(
true );
3047 if( m_indiNewCallBacks.count(
ipRecv.createUniqueKey() ) == 0 )
3053 if( m_indiNewCallBacks[
ipRecv.createUniqueKey()].property )
3057 m_indiDriver->sendDefProperty( *( m_indiNewCallBacks[
ipRecv.createUniqueKey()].property ) );
3059 catch(
const std::exception &e )
3061 log<software_error>( { __FILE__,
3063 "exception caught from sendDefProperty for " +
3064 m_indiNewCallBacks[
ipRecv.createUniqueKey()].property->getName() +
": " +
3071template <
bool _useINDI>
3076 if( m_indiDriver ==
nullptr )
3080 if( m_indiNewCallBacks.count(
ipRecv.createUniqueKey() ) == 0 )
3082 log<software_debug>( { __FILE__, __LINE__,
"invalid NewProperty request for " +
ipRecv.createUniqueKey() } );
3086 int ( *callBack )(
void *,
const pcf::IndiProperty & ) = m_indiNewCallBacks[
ipRecv.createUniqueKey()].callBack;
3089 callBack(
this,
ipRecv );
3091 log<software_debug>( { __FILE__, __LINE__,
"NewProperty callback null for " +
ipRecv.createUniqueKey() } );
3096template <
bool _useINDI>
3101 if( m_indiDriver ==
nullptr )
3104 std::string key =
ipRecv.createUniqueKey();
3107 if( m_indiSetCallBacks.count( key ) > 0 )
3109 m_indiSetCallBacks[key].m_defReceived =
true;
3112 int ( *callBack )(
void *,
const pcf::IndiProperty & ) = m_indiSetCallBacks[key].callBack;
3114 callBack(
this,
ipRecv );
3126template <
bool _useINDI>
3127template <
typename T>
3129 const std::string &el,
3131 pcf::IndiProperty::PropertyStateType ipState )
3142template <
bool _useINDI>
3144 const std::string &el,
3146 pcf::IndiProperty::PropertyStateType ipState )
3148 updateIfChanged<std::string>( p, el, std::string( newVal ), ipState );
3151template <
bool _useINDI>
3153 const std::string &el,
3154 const pcf::IndiElement::SwitchStateType &newVal,
3155 pcf::IndiProperty::PropertyStateType ipState )
3166template <
bool _useINDI>
3167template <
typename T>
3169 const std::string &el,
3170 const std::vector<T> &newVals,
3171 pcf::IndiProperty::PropertyStateType ipState )
3179 std::vector<std::string> descriptors( newVals.size(), el );
3180 for(
size_t index = 0; index < newVals.size(); ++index )
3182 descriptors[index] += std::to_string( index );
3187template <
bool _useINDI>
3188template <
typename T>
3190 const std::vector<std::string> &els,
3191 const std::vector<T> &newVals,
3192 pcf::IndiProperty::PropertyStateType newState )
3203template <
bool _useINDI>
3204template <
typename T>
3206 const std::vector<const char *> &els,
3207 const std::vector<T> &newVals,
3208 pcf::IndiProperty::PropertyStateType newState )
3223template <
bool _useINDI>
3224template <
typename T>
3227 const pcf::IndiProperty &remoteProperty,
3230 if( remoteProperty.createUniqueKey() != localProperty.createUniqueKey() )
3235 if( !( remoteProperty.find(
"target" ) || remoteProperty.find(
"current" ) ) )
3242 if( remoteProperty.find(
"target" ) )
3244 localTarget = remoteProperty[
"target"].get<T>();
3250 if( remoteProperty.find(
"current" ) )
3252 localTarget = remoteProperty[
"current"].get<T>();
3276template <
typename T>
3279 return pcf::IndiProperty::Unknown;
3294template <
bool _useINDI>
3295template <
typename T>
3303 log<software_error>( { __FILE__, __LINE__,
"INDI communications not initialized." } );
3306 pcf::IndiProperty ipToSend = ipSend;
3310 ipToSend[el].setValue( newVal );
3314 log<software_error>(
3315 { __FILE__, __LINE__,
"Exception caught setting " + ipSend.createUniqueKey() +
"." + el } );
3319 int rv = m_indiDriver->sendNewProperty( ipToSend );
3322 log<software_error>( { __FILE__, __LINE__ } );
3329template <
bool _useINDI>
3337 return log<
software_error, -1>( { __FILE__, __LINE__,
"INDI communications not initialized." } );
3340 if( m_indiDriver->sendNewProperty( ipSend ) < 0 )
3348template <
bool _useINDI>
3354 pcf::IndiProperty ipSend( pcf::IndiProperty::Switch );
3358 ipSend.setDevice( device );
3359 ipSend.setName( property );
3360 ipSend.add( pcf::IndiElement(
"toggle" ) );
3362 catch( std::exception &e )
3364 return log<
software_error, -1>( { __FILE__, __LINE__, std::string(
"exception: " ) + e.what() } );
3367 if( onoff ==
false )
3369 ipSend[
"toggle"].setSwitchState( pcf::IndiElement::Off );
3373 ipSend[
"toggle"].setSwitchState( pcf::IndiElement::On );
3376 if( sendNewProperty( ipSend ) < 0 )
3379 { __FILE__, __LINE__,
"sendNewProperty failed for " + device +
"." +
property } );
3385template <
bool _useINDI>
3391template <
bool _useINDI>
3395 if(
ipRecv.createUniqueKey() != m_indiP_clearFSMAlert.createUniqueKey() )
3397 return log<
software_error, -1>( { __FILE__, __LINE__,
"wrong indi property received" } );
3400 if( !
ipRecv.find(
"request" ) )
3403 if(
ipRecv[
"request"].getSwitchState() == pcf::IndiElement::On )
3406 updateSwitchIfChanged( m_indiP_clearFSMAlert,
"request", pcf::IndiElement::Off,
INDI_IDLE );
3412template <
bool _useINDI>
3418template <
bool _useINDI>
3424template <
bool _useINDI>
3427 if( !m_powerMgtEnabled || m_powerOnWait == 0 || m_powerOnCounter < 0 )
3432 if( m_powerOnCounter * m_loopPause > ( (
double)m_powerOnWait ) * 1e9 )
3443template <
bool _useINDI>
3446 if( !m_powerMgtEnabled )
3449 return m_powerState;
3452template <
bool _useINDI>
3455 if( !m_powerMgtEnabled )
3458 return m_powerTargetState;
3461template <
bool _useINDI>
3469 ps =
ipRecv[m_powerElement].get<std::string>();
3475 else if( ps ==
"Off" )
3487 ps =
ipRecv[m_powerTargetElement].get<std::string>();
3491 m_powerTargetState = 1;
3493 else if( ps ==
"Off" )
3495 m_powerTargetState = 0;
3499 m_powerTargetState = -1;
3506template <
bool _useINDI>
3509 return m_configName;
3512template <
bool _useINDI>
3518template <
bool _useINDI>
3521 return m_driverInName;
3524template <
bool _useINDI>
3527 return m_driverOutName;
3530template <
bool _useINDI>
3533 return m_driverCtrlName;
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 ) \
3560 log<software_error>( { __FILE__, __LINE__, "error from threadStart for " #thrdName } ); \
3571#define XWCAPP_THREAD_CHECK( thrdSt, thrdName ) \
3574 if( pthread_tryjoin_np( thrdSt.native_handle(), 0 ) == 0 ) \
3576 log<software_error>( { __FILE__, __LINE__, #thrdName " thread has exited" } ); \
3582 log<software_error>( { __FILE__, __LINE__, #thrdName " thread has exited" } ); \
3591#define XWCAPP_THREAD_STOP( thrdSt ) \
3592 if( thrdSt.joinable() ) \
3594 pthread_kill( thrdSt.native_handle(), SIGUSR1 ); \
Internal class to manage setuid privilege escalation with RAII.
elevatedPrivileges(MagAOXApp *app)
The base-class for MagAO-X applications.
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).
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
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
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.
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.
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.
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.
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.
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).
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.
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.
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.
#define MAGAOX_default_loopPause
The default application loopPause.
#define MAGAOX_calibRelPath
The relative path to the calibration files.
#define MAGAOX_driverFIFORelPath
The relative path to the INDI driver FIFOs.
#define MAGAOX_configRelPath
The relative path to the configuration files.
#define MAGAOX_logRelPath
The relative path to the log directory.
#define MAGAOX_sysRelPath
The relative path to the system directory.
#define MAGAOX_path
The path to the MagAO-X system files.
#define MAGAOX_secretsRelPath
The relative path to the secrets directory. Used for storing passwords, etc.
#define MAGAOX_cpusetPath
The absolute path the the cpuset mount point.
#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.
std::shared_ptr< char > bufferPtrT
The log entry buffer smart pointer.
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.
MagAO-X INDI Driver Wrapper.
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.
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 >()
pcf::IndiProperty::Type propType< std::string >()
pcf::IndiProperty::Type propType< char * >()
void sigUsr1Handler(int signum, siginfo_t *siginf, void *ucont)
Empty signal handler. SIGUSR1 is used to interrupt sleep in various threads.
pcf::IndiProperty::Type propType< double >()
const pcf::IndiProperty & ipRecv
updateIfChanged(m_indiP_angle, "target", m_angle)
pcf::IndiProperty::Type propType()
std::unique_lock< std::mutex > lock(m_indiMutex)
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.
The standard MagAOX log manager, used for both process logs and telemetry streams.
A simple text log, a string-type log.
A fixed-width timespec structure.
nanosecT time_ns
Nanoseconds.
secT time_s
Time since the Unix epoch.