7#ifndef modalGainOpt_hpp
8#define modalGainOpt_hpp
10#include <mx/ao/analysis/clGainOpt.hpp>
11#include <mx/ao/analysis/clAOLinearPredictor.hpp>
13#include "../../libMagAOX/libMagAOX.hpp"
14#include "../../magaox_git_version.h"
64 return "gainFactShmim";
77 return "multFactShmim";
90 return "pcGainFactShmim";
103 return "pcMultFactShmim";
116 return "numpccoeffShmim";
129 return "acoeffShmim";
142 return "bcoeffShmim";
155 return "gainCalShmim";
168 return "gainCalFactShmim";
173 return "gainCalFact";
239 typedef std::chrono::time_point<std::chrono::steady_clock>
timePointT;
268 std::vector<mx::AO::analysis::clGainOpt<float>>
m_goptSI;
269 std::vector<mx::AO::analysis::clGainOpt<float>>
m_goptLP;
270 std::vector<mx::AO::analysis::clAOLinearPredictor<float>>
m_linPred;
582 config.add(
"loop.number",
590 "The number of the loop. Used to set shmim names, as in aolN_mgainfact." );
592 config.add(
"loop.name",
600 "The name of the loop control INDI device name." );
602 config.add(
"loop.psdDev",
610 "The INDI device name of the PSD calculator. Defaults to aolN_modevalPSDs where N is loop.number." );
612 config.add(
"loop.autoUpdate",
620 "Flag controlling whether the gains are auto updated. Also settable via INDI." );
622 config.add(
"loop.gainGain",
630 "The gain to use for closed-loop gain updates. Default is 0.1" );
900 static_cast<void>(
dummy );
1103 static_cast<void>(
dummy );
1121 if( !
lock.owns_lock() )
1146 static_cast<void>(
dummy );
1153 static_cast<void>(
dummy );
1162 float *
f =
static_cast<float *
>(
curr_src );
1173 for(
size_t n = 0;
n <
sz; ++
n )
1191 for(
size_t n = 0;
n <
sz; ++
n )
1203 std::cerr <<
"got freq: " <<
sz <<
'\n';
1204 std::cerr <<
" fps: " <<
m_fps <<
'\n';
1212 static_cast<void>(
dummy );
1224 static_cast<void>(
dummy );
1242 float *
g =
static_cast<float *
>(
curr_src );
1269 std::cerr <<
"got gains: " <<
m_gainFacts.size() <<
"\n";
1277 static_cast<void>(
dummy );
1289 static_cast<void>(
dummy );
1307 float *
m =
static_cast<float *
>(
curr_src );
1337 std::cerr <<
"got mcs: " <<
m_multFacts.size() <<
"\n";
1345 static_cast<void>(
dummy );
1357 static_cast<void>(
dummy );
1375 float *
g =
static_cast<float *
>(
curr_src );
1402 std::cerr <<
"got pc gains: " <<
m_pcGainFacts.size() <<
"\n";
1410 static_cast<void>(
dummy );
1422 static_cast<void>(
dummy );
1440 float *
m =
static_cast<float *
>(
curr_src );
1470 std::cerr <<
"got mcs: " <<
m_multFacts.size() <<
"\n";
1478 static_cast<void>(
dummy );
1490 static_cast<void>(
dummy );
1528 mx::improc::eigenMap<int>
Npc(
reinterpret_cast<int *
>(
curr_src ), w, 2 );
1559 std::cerr <<
"got num pc coeffs: " <<
m_Na.size() <<
"\n";
1568 static_cast<void>(
dummy );
1575 static_cast<void>(
dummy );
1593 m_as.resize( w - 1, h );
1643 std::cerr <<
"got a coeffs: " << w <<
' ' << h <<
' ' <<
m_NaCurrent.size() <<
"\n";
1651 static_cast<void>(
dummy );
1658 static_cast<void>(
dummy );
1677 m_bs.resize( w - 1, h );
1727 std::cerr <<
"got b coeffs: " <<
m_NbCurrent.size() <<
"\n";
1735 static_cast<void>(
dummy );
1742 static_cast<void>(
dummy );
1765 float *
g =
static_cast<float *
>(
curr_src );
1789 std::cerr <<
"got gainCals: " <<
m_gainCals.size() <<
"\n";
1797 static_cast<void>(
dummy );
1804 static_cast<void>(
dummy );
1827 float *
g =
static_cast<float *
>(
curr_src );
1851 std::cerr <<
"got gainCalsFacts: " <<
m_gainCalFacts.size() <<
"\n";
1859 static_cast<void>(
dummy );
1866 static_cast<void>(
dummy );
1889 float *
t =
static_cast<float *
>(
curr_src );
1914 std::cerr <<
"got taus: " <<
m_taus.size() <<
"\n";
1933 std::vector<bool>
logged( 50,
false );
2199 std::cerr <<
"updating gopt structures\n";
2223 std::vector<float>
ta(
m_as.rows() );
2224 for(
size_t m = 0;
m <
ta.size(); ++
m )
2230 std::vector<float>
tb(
m_bs.rows() );
2231 for(
size_t m = 0;
m <
tb.size(); ++
m )
2254 std::cerr <<
"done\n";
2264#pragma omp parallel for num_threads( 15 )
2386 std::cerr <<
"Optimization took " <<
dt.count() <<
" seconds\n";
2529 if(
ipRecv.find(
"toggle" ) )
2531 if(
ipRecv[
"toggle"].getSwitchState() == pcf::IndiElement::On )
2537 m_autoUpdate =
true;
2545 m_autoUpdate =
false;
2556 if(
ipRecv.find(
"request" ) )
2558 if(
ipRecv[
"request"].getSwitchState() == pcf::IndiElement::On )
2560 m_updateOnce =
true;
2564 m_updateOnce =
false;
2575 if(
ipRecv.find(
"request" ) )
2577 if(
ipRecv[
"request"].getSwitchState() == pcf::IndiElement::On )
2595 if( indiTargetUpdate( m_indiP_gainGain, target,
ipRecv,
true ) < 0 )
2597 log<software_error>( { __FILE__, __LINE__ } );
2601 m_gainGain = target;
2610 if(
ipRecv.find(
"current" ) )
2612 float psdTime =
ipRecv[
"current"].get<
float>();
2614 if( psdTime != m_psdTime )
2617 std::lock_guard<std::mutex>
lock( m_goptMutex );
2620 m_psdTime = psdTime;
2625 std::cerr <<
"Got psdTime: " << m_psdTime <<
'\n';
2635 if(
ipRecv.find(
"current" ) )
2637 float psdAvgTime =
ipRecv[
"current"].get<
float>();
2639 if( psdAvgTime != m_psdAvgTime )
2642 std::lock_guard<std::mutex>
lock( m_goptMutex );
2645 m_psdAvgTime = psdAvgTime;
2650 std::cerr <<
"Got psdAvgTime: " << m_psdAvgTime <<
'\n';
2661 if(
ipRecv.find(
"toggle" ) )
2665 if(
ipRecv[
"toggle"].getSwitchState() == pcf::IndiElement::On )
2674 if( state != m_loop )
2677 std::lock_guard<std::mutex>
lock( m_goptMutex );
2684 std::cerr <<
"Got loop: " << m_loop <<
'\n';
2695 if(
ipRecv.find(
"current" ) )
2697 float gain =
ipRecv[
"current"].get<
float>();
2699 if( gain != m_gain && !m_pcOn )
2702 std::lock_guard<std::mutex>
lock( m_goptMutex );
2714 std::cerr <<
"Got gain: " << m_gain <<
'\n';
2729 if(
ipRecv.find(
"current" ) )
2731 float mc =
ipRecv[
"current"].get<
float>();
2733 if( mc != m_mult && !m_pcOn )
2736 std::lock_guard<std::mutex>
lock( m_goptMutex );
2746 m_goptUpdated =
true;
2748 std::cerr <<
"Got mc: " << m_mult <<
'\n';
2763 if(
ipRecv.find(
"current" ) )
2765 float gain =
ipRecv[
"current"].get<
float>();
2767 if( gain != m_pcGain && m_pcOn )
2770 std::lock_guard<std::mutex>
lock( m_goptMutex );
2782 std::cerr <<
"Got pc gain: " << m_pcGain <<
'\n';
2797 if(
ipRecv.find(
"current" ) )
2799 float mc =
ipRecv[
"current"].get<
float>();
2801 if( mc != m_pcMult && m_pcOn )
2804 std::lock_guard<std::mutex>
lock( m_goptMutex );
2814 m_pcgoptUpdated =
true;
2816 std::cerr <<
"Got pc mc: " << m_pcMult <<
'\n';
2831 if(
ipRecv.find(
"toggle" ) )
2835 if(
ipRecv[
"toggle"].getSwitchState() == pcf::IndiElement::On )
2844 if( state != m_pcOn )
2847 std::lock_guard<std::mutex>
lock( m_goptMutex );
2854 std::cerr <<
"Got pcOn: " << std::boolalpha << m_pcOn <<
'\n';
#define XWCAPP_THREAD_CHECK(thrdSt, thrdName)
Error handling wrapper for checking on thread status with tryjoin.
#define XWCAPP_THREAD_START(thrdSt, thrdInit, thrdId, thrdProp, thrdPrio, thrdCpuset, thrdName, thrdStart)
Error handling wrapper for the threadStart function of the XWCApp.
#define XWCAPP_THREAD_STOP(thrdSt)
Error handlng wrapper for stopping a thread.
The base-class for MagAO-X applications.
stateCodes::stateCodeT state()
Get the current state code.
int m_shutdown
Flag to signal it's time to shutdown. When not 0, the main loop exits.
int shutdown()
Get the value of the shutdown flag.
void updateSwitchIfChanged(pcf::IndiProperty &p, const std::string &el, const pcf::IndiElement::SwitchStateType &newVal, pcf::IndiProperty::PropertyStateType ipState=pcf::IndiProperty::Ok)
Update an INDI switch element value if it has changed.
static int log(const typename logT::messageT &msg, logPrioT level=logPrio::LOG_DEFAULT)
Make a log entry.
uint32_t m_width
The width of the images in the stream.
uint32_t m_height
The height of the images in the stream.
IMAGE m_imageStream
The ImageStreamIO shared memory buffer.
std::string m_shmimName
The name of the shared memory image, is used in /tmp/<shmimName>.im.shm. Derived classes should set a...
uint8_t m_dataType
The ImageStreamIO type code.
bool m_getExistingFirst
If set to true by derivedT, any existing image will be grabbed and sent to processImage before waitin...
The MagAO-X PSD-based gain optimizer.
std::vector< mx::AO::analysis::clGainOpt< float > > m_goptCurrent
Each mode gets its own gain optimizer.
pcf::IndiProperty m_indiP_gainGain
std::string m_clXferSIShmimName
INDI_NEWCALLBACK_DECL(modalGainOpt, m_indiP_updateOnce)
std::string m_clXferCurrentShmimName
dev::shmimMonitor< modalGainOpt, numpccoeffShmimT > numpccoeffShmimMonitorT
mx::improc::eigenImage< float > m_clXferCurrent
~modalGainOpt() noexcept
D'tor, declared and defined for noexcept.
IMAGE * m_clXferLPStream
The ImageStreamIO shared memory buffer to publish the LP ETF.
std::vector< int > m_regCounter
Counters to track when this mode was last regularized.
std::vector< float > m_gmaxLP
The previously calculated maximum gains for LP.
std::vector< float > m_pcGainFacts
std::vector< mx::AO::analysis::clGainOpt< float > > m_goptLP
std::string m_optGainSIShmimName
std::chrono::time_point< std::chrono::steady_clock > timePointT
dev::shmimMonitor< modalGainOpt, pcGainFactShmimT > pcGainFactShmimMonitorT
pid_t m_goptThreadID
gain optimization thread PID.
INDI_NEWCALLBACK_DECL(modalGainOpt, m_indiP_autoUpdate)
dev::shmimMonitor< modalGainOpt, pcMultFactShmimT > pcMultFactShmimMonitorT
pcf::IndiProperty m_indiP_fps
std::vector< float > m_taus
dev::shmimMonitor< modalGainOpt, gainCalShmimT > gainCalShmimMonitorT
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_fps)
float m_psdOverlapFraction
int allocate(const psdShmimT &)
INDI_NEWCALLBACK_DECL(modalGainOpt, m_indiP_dump)
std::vector< float > m_freq
pcf::IndiProperty m_indiP_siGain
virtual int appLogic()
Implementation of the FSM for modalGainOpt.
std::vector< float > m_modeVarSI
pcf::IndiProperty m_indiP_autoUpdate
int m_loopNum
The number of the loop. Used to set shmim names, as in aolN_mgainfact.
std::vector< float > m_gainCalFacts
pcf::IndiProperty m_indiP_dump
std::vector< float > m_gainFacts
dev::shmimMonitor< modalGainOpt, bcoeffShmimT > bcoeffShmimMonitorT
int m_goptThreadPrio
Priority of the gain optimization thread.
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_pcOn)
std::vector< float > m_optGainLP
sem_t m_goptSemaphore
Semaphore used to synchronize the psdShmim thread and the gopt thread.
IMAGE * m_optGainLPStream
The ImageStreamIO shared memory buffer to publish the LP optimal gains.
std::string m_goptThreadCpuset
The cpuset to use for the gain optimization thread.
pcf::IndiProperty m_indiP_loop
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_siGain)
mx::improc::eigenImage< float > m_clPSDs
pcf::IndiProperty m_indiP_siMult
pcf::IndiProperty m_indiP_pcOn
IMAGE * m_olPSDStream
The ImageStreamIO shared memory buffer to publish the open loop PSDs.
IMAGE * m_clXferSIStream
The ImageStreamIO shared memory buffer to publish the SI ETF.
std::vector< std::vector< float > > m_olPSDs
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_loop)
std::string m_clXferLPShmimName
std::string m_olPSDShmimName
bool m_goptUpdated
Tracks if a parameter has updated requiring updates to the m_gopt entries.
mx::improc::eigenImage< float > m_clXferLP
dev::shmimMonitor< modalGainOpt, freqShmimT > freqShmimMonitorT
virtual void loadConfig()
std::string m_loopName
The name of the loop control INDI device name.
std::chrono::duration< double > durationT
dev::shmimMonitor< modalGainOpt, gainFactShmimT > gainFactShmimMonitorT
std::vector< int > m_NbCurrent
bool m_updateOnce
Flag to trigger a single update with gain.
pcf::IndiProperty m_indiP_updateOnce
dev::shmimMonitor< modalGainOpt, tauShmimT > tauShmimMonitorT
std::vector< mx::AO::analysis::clAOLinearPredictor< float > > m_linPred
pcf::IndiProperty m_goptThreadProp
The property to hold the gain optimization thread details.
static void goptThreadStart(modalGainOpt *p)
Gain Optimization thread starter function.
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_psdAvgTime)
bool m_dump
Flag to trigger a single update with no gain.
mx::improc::eigenImage< float > m_clXferSI
pcf::IndiProperty m_indiP_psdAvgTime
std::vector< float > m_modeVarLP
IMAGE * m_optGainStream
The ImageStreamIO shared memory buffer to publish the optimal gains.
std::vector< float > m_optGainSI
IMAGE * m_optGainSIStream
The ImageStreamIO shared memory buffer to publish the SI optimal gains.
modalGainOpt()
Default c'tor.
IMAGE * m_clXferCurrentStream
The ImageStreamIO shared memory buffer to publish the SI ETF.
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_psdTime)
int loadConfigImpl(mx::app::appConfigurator &_config)
Implementation of loadConfig logic, separated for testing.
std::vector< float > m_gainCals
virtual int appStartup()
Startup function.
std::thread m_goptThread
The gain optimization thread.
void goptThreadExec()
Gain optimization thread function.
dev::shmimMonitor< modalGainOpt, psdShmimT > psdShmimMonitorT
virtual void setupConfig()
bool m_updating
Flag used to indicate to the goptThread that it should stop calculations ASAP.
virtual int appShutdown()
Shutdown the app.
bool m_goptThreadInit
Initialization flag for the gain optimization thread.
int processImage(void *curr_src, const psdShmimT &)
std::string m_optGainShmimName
std::vector< float > m_pcMultFacts
std::vector< int > m_NaCurrent
std::mutex m_goptMutex
Mutex for synchronizing updates.
dev::shmimMonitor< modalGainOpt, multFactShmimT > multFactShmimMonitorT
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_siMult)
bool m_autoUpdate
Flag controlling whether gains are automatically updated.
dev::shmimMonitor< modalGainOpt, gainCalFactShmimT > gainCalFactShmimMonitorT
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_pcMult)
std::string m_optGainLPShmimName
std::vector< float > m_multFacts
dev::shmimMonitor< modalGainOpt, acoeffShmimT > acoeffShmimMonitorT
std::vector< float > m_regScale
The regularizatio scale factors for each mode.
bool m_pcgoptUpdated
Tracks if a parameter has updated requiring updates to the m_gopt entries.
std::vector< std::vector< float > > m_nPSDs
float m_gainGain
The gain to use for closed-loop gain updates. Default is 0.1.
pcf::IndiProperty m_indiP_pcGain
std::vector< mx::AO::analysis::clGainOpt< float > > m_goptSI
INDI_SETCALLBACK_DECL(modalGainOpt, m_indiP_pcGain)
INDI_NEWCALLBACK_DECL(modalGainOpt, m_indiP_gainGain)
pcf::IndiProperty m_indiP_psdTime
int m_nRegCycles
How often to regularize each mode.
pcf::IndiProperty m_indiP_pcMult
friend class modalGainOpt_test
#define INDI_NEWCALLBACK_DEFN(class, prop)
Define the callback for a new property request.
#define CREATE_REG_INDI_NEW_TOGGLESWITCH(prop, name)
Create and register a NEW INDI property as a standard toggle switch, using the standard callback name...
#define CREATE_REG_INDI_NEW_REQUESTSWITCH(prop, name)
Create and register a NEW INDI property as a standard request switch, using the standard callback nam...
#define CREATE_REG_INDI_NEW_NUMBERF(prop, name, min, max, step, format, label, group)
Create and register a NEW INDI property as a standard number as float, using the standard callback na...
#define INDI_SETCALLBACK_DEFN(class, prop)
Define the callback for a set property request.
#define REG_INDI_SETPROP(prop, devName, propName)
Register a SET INDI property with the class, using the standard callback name.
@ OPERATING
The device is operating, other than homing.
#define INDI_VALIDATE_CALLBACK_PROPS(prop1, prop2)
Standard check for matching INDI properties in a callback.
const pcf::IndiProperty & ipRecv
std::unique_lock< std::mutex > lock(m_indiMutex)
static constexpr logPrioT LOG_NOTICE
A normal but significant condition.
#define XWC_SEM_WAIT_TS_RETVOID(ts, sec, nsec)
Add the wait time to a timespec for a sem_timedwait call, with no value returned on error.
#define SHMIMMONITORT_APP_STARTUP(SHMIMMONITORT)
Call shmimMonitorT::appStartup with error checking for a typedef-ed shmimMonitor.
#define SHMIMMONITORT_UPDATE_INDI(SHMIMMONITORT)
Call shmimMonitorT::updateINDI with error checking for a typedef-ed shmimMonitor.
#define SHMIMMONITORT_SETUP_CONFIG(SHMIMMONITORT, cfig)
Call shmimMonitorT::setupConfig with error checking for a typedef-ed shmimMonitor.
#define SHMIMMONITORT_APP_LOGIC(SHMIMMONITORT)
Call shmimMonitorT::appLogic with error checking for a typedef-ed shmimMonitor.
#define SHMIMMONITORT_APP_SHUTDOWN(SHMIMMONITORT)
Call shmimMonitorT::appShutodwn with error checking for a typedef-ed shmimMonitor.
#define SHMIMMONITORT_LOAD_CONFIG(SHMIMMONITORT, cfig)
Call shmimMonitorT::loadConfig with error checking for a typedef-ed shmimMonitor.
static std::string configSection()
static std::string indiPrefix()
static std::string indiPrefix()
static std::string configSection()
static std::string indiPrefix()
static std::string configSection()
static std::string configSection()
static std::string indiPrefix()
static std::string indiPrefix()
static std::string configSection()
static std::string configSection()
static std::string indiPrefix()
static std::string configSection()
static std::string indiPrefix()
static std::string indiPrefix()
static std::string configSection()
static std::string configSection()
static std::string indiPrefix()
static std::string indiPrefix()
static std::string configSection()
static std::string configSection()
static std::string indiPrefix()
static std::string indiPrefix()
static std::string configSection()
Software CRITICAL log entry.