7#ifndef cacaoInterface_hpp
8#define cacaoInterface_hpp
10#include "../../libMagAOX/libMagAOX.hpp"
11#include "../../magaox_git_version.h"
131 const std::string &
param,
132 const std::string &
val
138 const std::string &
param,
143 const std::string &
param );
146 const std::string &
param );
277 config.add(
"loop.number",
"",
"loop.number", argType::Required,
"loop",
"number",
false,
"string",
"the loop number");
429 const std::string & param,
430 const std::string & val
444 if(w != (
int)
comout.size())
458 const std::string & param,
466 const std::string & param
482 if(w != (
int)
comout.size())
494 while(
rfd < 0 &&
nr < 500)
498 mx::sys::milliSleep(10);
523 size_t ned =
instr.find_last_not_of(
" \t\r\n");
525 if(
ned == std::string::npos ||
ned == 0)
543 const std::string & param
559 if(w != (
int)
comout.size())
570 while(
rfd < 0 &&
nr < 500)
574 mx::sys::milliSleep(10);
599 size_t ned =
instr.find_last_not_of(
" \t\r\n");
601 if(
ned == std::string::npos ||
ned == 0)
635 calsrc +=
"/calib_dir.txt";
735 if(
setFPSVal(
"mfilt",
"loopON", std::string(
"ON")) != 0)
749 if(
setFPSVal(
"mfilt",
"loopON", std::string(
"OFF")) != 0)
769 if(
setFPSVal(
"mfilt",
"loopZERO", std::string(
"ON")) != 0)
817 catch(
const std::exception&
e)
843 mx::sys::milliSleep(500);
854 if(!
ipRecv.find(
"toggle"))
return 0;
856 std::unique_lock<std::mutex>
lock(m_indiMutex);
858 if(
ipRecv[
"toggle"].getSwitchState() == pcf::IndiElement::On)
862 else if(
ipRecv[
"toggle"].getSwitchState() == pcf::IndiElement::Off)
867 log<software_error>({__FILE__,__LINE__,
"switch state fall through."});
878 if(
ipRecv.find(
"current"))
880 current =
ipRecv[
"current"].get<
double>();
885 target =
ipRecv[
"target"].get<
double>();
888 if(target == -1) target = current;
895 std::lock_guard<std::mutex> guard(m_indiMutex);
897 m_gain_target = target;
908 if(!
ipRecv.find(
"request"))
return 0;
910 std::unique_lock<std::mutex>
lock(m_indiMutex);
912 if(
ipRecv[
"request"].getSwitchState() == pcf::IndiElement::On)
927 if(
ipRecv.find(
"current"))
929 current =
ipRecv[
"current"].get<
double>();
934 target =
ipRecv[
"target"].get<
double>();
937 if(target == -1) target = current;
944 std::lock_guard<std::mutex> guard(m_indiMutex);
946 m_multCoeff_target = target;
949 return setMultCoeff();
959 if(
ipRecv.find(
"current"))
961 current =
ipRecv[
"current"].get<
double>();
966 target =
ipRecv[
"target"].get<
double>();
969 if(target == -1) target = current;
976 std::lock_guard<std::mutex> guard(m_indiMutex);
977 m_maxLim_target = target;
999 static float gain {-1000};
1000 static float multcoef {0};
1001 static float limit {0};
The base-class for MagAO-X applications.
void updateIfChanged(pcf::IndiProperty &p, const std::string &el, const T &newVal, pcf::IndiProperty::PropertyStateType ipState=pcf::IndiProperty::Ok)
Update an INDI property element value if it has changed.
int createStandardIndiRequestSw(pcf::IndiProperty &prop, const std::string &name, const std::string &label="", const std::string &group="")
Create a standard R/W INDI switch with a single request element.
stateCodes::stateCodeT state()
Get the current state code.
int registerIndiPropertyNew(pcf::IndiProperty &prop, int(*)(void *, const pcf::IndiProperty &))
Register an INDI property which is exposed for others to request a New Property for.
int createStandardIndiToggleSw(pcf::IndiProperty &prop, const std::string &name, const std::string &label="", const std::string &group="")
Create a standard R/W INDI switch with a single toggle element.
int m_shutdown
Flag to signal it's time to shutdown. When not 0, the main loop exits.
int shutdown()
Get the value of the shutdown flag.
void updateSwitchIfChanged(pcf::IndiProperty &p, const std::string &el, const pcf::IndiElement::SwitchStateType &newVal, pcf::IndiProperty::PropertyStateType ipState=pcf::IndiProperty::Ok)
Update an INDI switch element value if it has changed.
int stateLogged()
Updates and returns the value of m_stateLogged. Will be 0 on first call after a state change,...
static int log(const typename logT::messageT &msg, logPrioT level=logPrio::LOG_DEFAULT)
Make a log entry.
int registerIndiPropertyReadOnly(pcf::IndiProperty &prop)
Register an INDI property which is read only.
int createROIndiText(pcf::IndiProperty &prop, const std::string &propName, const std::string &elName, const std::string &propLabel="", const std::string &propGroup="", const std::string &elLabel="")
Create a standard ReadOnly INDI Text property, with at least one element.
std::mutex m_indiMutex
Mutex for locking INDI communications.
int threadStart(std::thread &thrd, bool &thrdInit, pid_t &tpid, pcf::IndiProperty &thProp, int thrdPrio, const std::string &cpuset, const std::string &thrdName, thisPtr *thrdThis, Function &&thrdStart)
Start a thread, using this class's privileges to set priority, etc.
The MagAO-X CACAO Interface.
int setMultCoeff()
Set loop multiplication coefficient to the value of m_multCoeff_target;.
pcf::IndiProperty m_indiP_loopZero
int m_loopState
The loop state. 0 = off, 1 = paused (on, 0 gain), 2 = on.
int checkLoopProcesses()
Check if the loop processes are running.
INDI_NEWCALLBACK_DECL(cacaoInterface, m_indiP_maxLim)
bool m_fmThreadInit
Initialization flag for the file monitoring thread.
virtual int appLogic()
Implementation of the FSM for cacaoInterface.
std::string getFPSValNum(const std::string &fps, const std::string ¶m)
void fmThreadExec()
File monitoring thread function.
float m_gain_target
The target loop gain.
static void fmThreadStart(cacaoInterface *c)
File monitoring thread starter function.
INDI_NEWCALLBACK_DECL(cacaoInterface, m_indiP_loopZero)
pcf::IndiProperty m_indiP_loop
int loopOn()
Turn the loop on.
int setFPSVal(const std::string &fps, const std::string ¶m, const std::string &val)
std::vector< float > m_modeBlockLims
pcf::IndiProperty m_indiP_loopState
bool m_loopProcesses
Status of the loop processes.
float m_maxLim
The current max limit.
INDI_NEWCALLBACK_DECL(cacaoInterface, m_indiP_multCoeff)
pcf::IndiProperty m_fmThreadProp
The property to hold the f.m. thread details.
std::vector< float > m_modeBlockGains
int m_fmThreadPrio
Priority of the filemonitoring thread.
std::vector< float > m_modeBlockMCs
int setGain()
Set loop gain to the value of m_gain_target;.
std::string m_aoCalArchiveTime
virtual void setupConfig()
float m_gain
The current loop gain.
std::string m_loopName
the loop name
~cacaoInterface() noexcept
D'tor, declared and defined for noexcept.
std::string m_aoCalLoadTime
std::string getFPSValStr(const std::string &fps, const std::string ¶m)
int recordLoopGain(bool force=false)
std::vector< int > m_modeBlockStart
std::thread m_fmThread
The file monitoring thread.
virtual int appStartup()
Startup function.
friend class cacaoInterface_test
int getAOCalib()
Get the calibration details.
int setMaxLim()
Set loop max lim to the value of m_maxLim_target;.
virtual int appShutdown()
Shutdown the app.
float m_maxLim_target
The target max limit.
float m_multCoeff
The current multiplicative coefficient (1-leak)
dev::telemeter< cacaoInterface > telemeterT
std::mutex m_modeBlockMutex
int recordTelem(const telem_loopgain *)
pcf::IndiProperty m_indiP_maxLim
INDI_NEWCALLBACK_DECL(cacaoInterface, m_indiP_loopState)
std::string m_loopNumber
The loop number, X in aolX. We keep it a string because that's how it gets used.
int loadConfigImpl(mx::app::appConfigurator &_config)
Implementation of loadConfig logic, separated for testing.
pid_t m_fmThreadID
File monitor thread PID.
std::vector< int > m_modeBlockN
int loopOff()
Turn the loop off.
pcf::IndiProperty m_indiP_loopProcesses
pcf::IndiProperty m_indiP_multCoeff
float m_multCoeff_target
The target multiplicative coefficient (1-leak)
pcf::IndiProperty m_indiP_loopGain
int loopZero()
Zero the loop control channel.
bool m_loopProcesses_stat
What the cacao status file says the state of loop processes is.
virtual void loadConfig()
cacaoInterface()
Default c'tor.
INDI_NEWCALLBACK_DECL(cacaoInterface, m_indiP_loopGain)
#define INDI_NEWCALLBACK_DEFN(class, prop)
Define the callback for a new property request.
#define INDI_NEWCALLBACK(prop)
Get the name of the static callback wrapper for a new property.
@ OPERATING
The device is operating, other than homing.
@ ERROR
The application has encountered an error, from which it is recovering (with or without intervention)
@ READY
The device is ready for operation, but is not operating.
#define INDI_VALIDATE_CALLBACK_PROPS(prop1, prop2)
Standard check for matching INDI properties in a callback.
int addTextElement(pcf::IndiProperty &prop, const std::string &name, const std::string &label="")
Add a standard INDI Text element.
const pcf::IndiProperty & ipRecv
updateIfChanged(m_indiP_angle, "target", m_angle)
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_ERROR
An error has occured which the software will attempt to correct.
A device base class which saves telemetry.
int appShutdown()
Perform telemeter application shutdown.
int loadConfig(appConfigurator &config)
Load the device section from an application configurator.
int appLogic()
Perform telemeter application logic.
int setupConfig(appConfigurator &config)
Setup an application configurator for the device section.
int appStartup()
Starts the telemetry log thread.
int checkRecordTimes(const telT &tel, telTs... tels)
Check the time of the last record for each telemetry type and make an entry if needed.
Software CRITICAL log entry.
Log entry recording the build-time git state.