30#include "../../math/constants.hpp"
31#include "../../math/roots.hpp"
32#include "../../mxError.hpp"
33#include "../../mxException.hpp"
41#define FITTING_ERROR_NO 0
42#define FITTING_ERROR_ZERO 1
43#define FITTING_ERROR_X 2
44#define FITTING_ERROR_Y 3
63template <
typename _realT,
class _inputSpectT,
typename iosT = std::ostream>
69 typedef _inputSpectT inputSpectT;
185 void d_min(
const std::vector<realT> &nd );
198 realT
d_min(
size_t idx );
204 std::vector<realT>
d_min();
209 void optd(
bool od );
233 template <
typename wfsT>
241 template<
typename wfsT>
242 void wfsBeta(
const wfsT * w );
294 void npix_wfs(
const std::vector<realT> &npix );
318 void ron_wfs(
const std::vector<realT> &nron );
342 void Fbg(
const std::vector<realT> &fbg );
355 realT
Fbg(
size_t idx );
361 std::vector<realT>
Fbg();
366 void minTauWFS(
const std::vector<realT> &ntau );
410 void tauWFS( realT ntau );
454 void zeta( realT nz );
536 void F0( realT nF0 );
559 realT
Fg( realT mag );
876 template <
typename varFuncT>
894 template <
typename imageT,
typename CfuncT>
895 void C_Map( imageT &im,
905 realT
C0var( realT m,
919 bool normStrehl =
true
926 template <
typename imageT>
927 void C0Map( imageT &map,
935 realT
C1var( realT m,
947 bool normStrehl =
true
957 template <
typename imageT>
958 void C1Map( imageT &map,
966 realT
C2var( realT m,
979 bool normStrehl =
true
989 template <
typename imageT>
990 void C2Map( imageT &map,
998 realT
C3var( realT m,
1011 bool normStrehl =
true
1021 template <
typename imageT>
1022 void C3Map( imageT &map,
1030 realT
C4var( realT m,
1043 bool normStrehl =
true
1053 template <
typename imageT>
1054 void C4Map( imageT &map,
1064 realT
C5var( realT m,
1077 bool normStrehl =
true
1087 template <
typename imageT>
1088 void C5Map( imageT &map,
1096 realT
C6var( realT m,
1109 bool normStrehl =
true
1119 template <
typename imageT>
1120 void C6Map( imageT &map,
1128 realT
C7var( realT m,
1140 bool normStrehl =
true
1150 template <
typename imageT>
1151 void C7Map( imageT &map,
1175template <
typename realT,
class inputSpectT,
typename iosT>
1178 wfsBeta<wfs<realT, iosT>>( nullptr );
1181template <
typename realT,
class inputSpectT,
typename iosT>
1184 if( m_wfsBeta && m_ownWfsBeta )
1190template <
typename realT,
class inputSpectT,
typename iosT>
1193 atm.loadGuyon2005();
1202 npix_wfs( std::vector<realT>( { 12868 } ) );
1203 ron_wfs( std::vector<realT>( { 0.0 } ) );
1204 Fbg( std::vector<realT>( { 0.0 } ) );
1207 minTauWFS( (realT)( 1. / 1e9 ) );
1210 m_specsChanged =
true;
1211 m_dminChanged =
true;
1214template <
typename realT,
class inputSpectT,
typename iosT>
1219 lam_wfs( 0.791e-6 );
1220 lam_sci( 0.656e-6 );
1222 npix_wfs( std::vector<realT>( { 9024 } ) );
1223 ron_wfs( std::vector<realT>( { 0.57 } ) );
1224 Fbg( std::vector<realT>( { 0.22 } ) );
1226 d_min( std::vector<realT>( { 6.5 / 48.0 } ) );
1227 minTauWFS( std::vector<realT>( { 1. / 3622. } ) );
1228 tauWFS( 1. / 3622. );
1232 m_specsChanged =
true;
1233 m_dminChanged =
true;
1236template <
typename realT,
class inputSpectT,
typename iosT>
1242 F0( 7.6e10 * 368.0 / ( 0.25 *
math::pi<realT>() * 6.5 * 6.5 * ( 1. - 0.29 * 0.29 ) ) );
1246 m_specsChanged =
true;
1247 m_dminChanged =
true;
1250template <
typename realT,
class inputSpectT,
typename iosT>
1256 m_specsChanged =
true;
1257 m_dminChanged =
true;
1260template <
typename realT,
class inputSpectT,
typename iosT>
1266template <
typename realT,
class inputSpectT,
typename iosT>
1269 m_d_min.assign( nd.begin(), nd.end() );
1270 m_specsChanged =
true;
1271 m_dminChanged =
true;
1274template <
typename realT,
class inputSpectT,
typename iosT>
1277 if( m_d_min.size() < idx + 1 )
1279 mxThrowException(
err::sizeerr,
"aoSystem::d_min",
"idx larger than m_d_min" );
1284 m_specsChanged =
true;
1285 m_dminChanged =
true;
1288template <
typename realT,
class inputSpectT,
typename iosT>
1291 if( m_d_min.size() < idx + 1 )
1293 mxThrowException(
err::sizeerr,
"aoSystem::d_min",
"idx larger than m_d_min" );
1296 return m_d_min[idx];
1299template <
typename realT,
class inputSpectT,
typename iosT>
1305template <
typename realT,
class inputSpectT,
typename iosT>
1309 m_specsChanged =
true;
1310 m_dminChanged =
true;
1313template <
typename realT,
class inputSpectT,
typename iosT>
1319template <
typename realT,
class inputSpectT,
typename iosT>
1323 m_specsChanged =
true;
1324 m_dminChanged =
true;
1327template <
typename realT,
class inputSpectT,
typename iosT>
1330 return m_optd_delta;
1333template <
typename realT,
class inputSpectT,
typename iosT>
1334template <
typename wfsT>
1337 if( m_wfsBeta && m_ownWfsBeta )
1340 m_wfsBeta =
nullptr;
1344 m_ownWfsBeta =
false;
1347template <
typename realT,
class inputSpectT,
typename iosT>
1348template <
typename wfsT>
1351 if( m_wfsBeta && m_ownWfsBeta )
1354 m_wfsBeta =
nullptr;
1360 m_ownWfsBeta =
false;
1364 m_wfsBeta =
new wfsT;
1365 m_ownWfsBeta =
true;
1369template <
typename realT,
class inputSpectT,
typename iosT>
1375template <
typename realT,
class inputSpectT,
typename iosT>
1378 if( m_wfsBeta == 0 )
1379 mxThrowException(
err::paramnotset,
"aoSystem::beta_p",
"The WFS is not assigned." );
1381 return m_wfsBeta->beta_p( m, n, m_D, d_opt(), atm.r_0( m_lam_sci ) );
1384template <
typename realT,
class inputSpectT,
typename iosT>
1387 if( m_wfsBeta == 0 )
1388 mxThrowException(
err::paramnotset,
"aoSystem::beta_r",
"The WFS is not assigned." );
1390 return m_wfsBeta->beta_r( m, n, m_D, d_opt(), atm.r_0( m_lam_sci ) );
1393template <
typename realT,
class inputSpectT,
typename iosT>
1397 m_specsChanged =
true;
1398 m_dminChanged =
true;
1401template <
typename realT,
class inputSpectT,
typename iosT>
1404 return m_opticalGain;
1407template <
typename realT,
class inputSpectT,
typename iosT>
1411 m_specsChanged =
true;
1412 m_dminChanged =
true;
1415template <
typename realT,
class inputSpectT,
typename iosT>
1421template <
typename realT,
class inputSpectT,
typename iosT>
1424 m_npix_wfs.resize( npix.size() );
1425 for(
size_t n = 0; n < npix.size(); ++n )
1427 m_npix_wfs[n] = npix[n];
1430 m_specsChanged =
true;
1431 m_dminChanged =
true;
1434template <
typename realT,
class inputSpectT,
typename iosT>
1437 if( m_npix_wfs.size() < idx + 1 )
1439 mxThrowException(
err::sizeerr,
"aoSystem::npix_wfs",
"idx larger than m_npix_wfs" );
1442 m_npix_wfs[idx] = npix;
1444 m_specsChanged =
true;
1445 m_dminChanged =
true;
1448template <
typename realT,
class inputSpectT,
typename iosT>
1451 if( m_npix_wfs.size() < idx + 1 )
1453 mxThrowException(
err::sizeerr,
"aoSystem::npix_wfs",
"idx larger than m_npix_wfs" );
1456 return m_npix_wfs[idx];
1459template <
typename realT,
class inputSpectT,
typename iosT>
1465template <
typename realT,
class inputSpectT,
typename iosT>
1468 m_ron_wfs.resize( nron.size() );
1469 for(
size_t n = 0; n < nron.size(); ++n )
1471 m_ron_wfs[n] = nron[n];
1474 m_specsChanged =
true;
1475 m_dminChanged =
true;
1478template <
typename realT,
class inputSpectT,
typename iosT>
1481 if( m_ron_wfs.size() < idx + 1 )
1483 mxThrowException(
err::sizeerr,
"aoSystem::ron_wfs",
"idx larger than m_ron_wfs" );
1486 m_ron_wfs[idx] = nron;
1488 m_specsChanged =
true;
1489 m_dminChanged =
true;
1492template <
typename realT,
class inputSpectT,
typename iosT>
1495 if( m_ron_wfs.size() < idx + 1 )
1497 mxThrowException(
err::sizeerr,
"aoSystem::ron_wfs",
"idx larger than m_ron_wfs" );
1500 return m_ron_wfs[idx];
1503template <
typename realT,
class inputSpectT,
typename iosT>
1509template <
typename realT,
class inputSpectT,
typename iosT>
1512 m_Fbg.resize( fbg.size() );
1513 for(
size_t n = 0; n < fbg.size(); ++n )
1518 m_specsChanged =
true;
1519 m_dminChanged =
true;
1522template <
typename realT,
class inputSpectT,
typename iosT>
1525 if( m_Fbg.size() < idx + 1 )
1527 mxThrowException(
err::sizeerr,
"aoSyste,::Fbg",
"idx larger than m_Fbg" );
1532 m_specsChanged =
true;
1533 m_dminChanged =
true;
1536template <
typename realT,
class inputSpectT,
typename iosT>
1539 if( m_Fbg.size() < idx + 1 )
1541 mxThrowException(
err::sizeerr,
"aoSyste,::Fbg",
"idx larger than m_Fbg" );
1547template <
typename realT,
class inputSpectT,
typename iosT>
1553template <
typename realT,
class inputSpectT,
typename iosT>
1556 m_minTauWFS.resize( ntau.size() );
1557 for(
size_t n = 0; n < ntau.size(); ++n )
1559 m_minTauWFS[n] = ntau[n];
1562 m_specsChanged =
true;
1563 m_dminChanged =
true;
1566template <
typename realT,
class inputSpectT,
typename iosT>
1569 if( m_minTauWFS.size() < idx + 1 )
1571 mxThrowException(
err::sizeerr,
"aoSystem::minTauWFS",
"idx larger than m_ntau_wfs" );
1574 m_minTauWFS[idx] = ntau;
1576 m_specsChanged =
true;
1577 m_dminChanged =
true;
1580template <
typename realT,
class inputSpectT,
typename iosT>
1583 if( m_minTauWFS.size() < idx + 1 )
1585 mxThrowException(
err::sizeerr,
"aoSystem::minTauWFS",
"idx larger than m_ntau_wfs" );
1588 return m_minTauWFS[idx];
1591template <
typename realT,
class inputSpectT,
typename iosT>
1597template <
typename realT,
class inputSpectT,
typename iosT>
1600 if( bnp != m_bin_npix )
1603 m_specsChanged =
true;
1604 m_dminChanged =
true;
1608template <
typename realT,
class inputSpectT,
typename iosT>
1614template <
typename realT,
class inputSpectT,
typename iosT>
1621template <
typename realT,
class inputSpectT,
typename iosT>
1625 m_specsChanged =
true;
1626 m_dminChanged =
true;
1629template <
typename realT,
class inputSpectT,
typename iosT>
1635template <
typename realT,
class inputSpectT,
typename iosT>
1639 m_specsChanged =
true;
1640 m_dminChanged =
true;
1643template <
typename realT,
class inputSpectT,
typename iosT>
1649template <
typename realT,
class inputSpectT,
typename iosT>
1653 m_specsChanged =
true;
1654 m_dminChanged =
true;
1657template <
typename realT,
class inputSpectT,
typename iosT>
1663template <
typename realT,
class inputSpectT,
typename iosT>
1667 m_specsChanged =
true;
1668 m_dminChanged =
true;
1671template <
typename realT,
class inputSpectT,
typename iosT>
1677template <
typename realT,
class inputSpectT,
typename iosT>
1681 m_secZeta = 1 / cos( m_zeta );
1683 m_specsChanged =
true;
1684 m_dminChanged =
true;
1687template <
typename realT,
class inputSpectT,
typename iosT>
1693template <
typename realT,
class inputSpectT,
typename iosT>
1699template <
typename realT,
class inputSpectT,
typename iosT>
1707template <
typename realT,
class inputSpectT,
typename iosT>
1710 return m_fit_mn_max;
1713template <
typename realT,
class inputSpectT,
typename iosT>
1716 m_circularLimit = cl;
1717 m_specsChanged =
true;
1718 m_dminChanged =
true;
1721template <
typename realT,
class inputSpectT,
typename iosT>
1724 return m_circularLimit;
1727template <
typename realT,
class inputSpectT,
typename iosT>
1730 m_spatialFilter_ku = fabs( kx );
1731 m_specsChanged =
true;
1732 m_dminChanged =
true;
1735template <
typename realT,
class inputSpectT,
typename iosT>
1738 return m_spatialFilter_ku;
1741template <
typename realT,
class inputSpectT,
typename iosT>
1744 m_spatialFilter_kv = fabs( ky );
1745 m_specsChanged =
true;
1746 m_dminChanged =
true;
1749template <
typename realT,
class inputSpectT,
typename iosT>
1752 return m_spatialFilter_kv;
1755template <
typename realT,
class inputSpectT,
typename iosT>
1759 m_specsChanged =
true;
1760 m_dminChanged =
true;
1763template <
typename realT,
class inputSpectT,
typename iosT>
1769template <
typename realT,
class inputSpectT,
typename iosT>
1772 m_ncp_alpha = alpha;
1773 m_specsChanged =
true;
1774 m_dminChanged =
true;
1777template <
typename realT,
class inputSpectT,
typename iosT>
1783template <
typename realT,
class inputSpectT,
typename iosT>
1787 m_specsChanged =
true;
1788 m_dminChanged =
true;
1791template <
typename realT,
class inputSpectT,
typename iosT>
1797template <
typename realT,
class inputSpectT,
typename iosT>
1801 m_specsChanged =
true;
1802 m_dminChanged =
true;
1805template <
typename realT,
class inputSpectT,
typename iosT>
1811template <
typename realT,
class inputSpectT,
typename iosT>
1814 return m_F0 * pow( 10.0, -0.4 * mag );
1817template <
typename realT,
class inputSpectT,
typename iosT>
1820 return Fg( m_starMag );
1823template <
typename realT,
class inputSpectT,
typename iosT>
1826 if( m_specsChanged || m_dminChanged )
1829 return m_controlledModes;
1832template <
typename realT,
class inputSpectT,
typename iosT>
1835 Nph = Fg() * tau_wfs;
1837 double binfact = 1.0;
1841 if( m_npix_wfs.size() == 1 )
1843 binfact = 1. / pow( (realT)b + 1, 2 );
1849 return pow( Nph, 2 ) / ( m_npix_wfs[binidx] * binfact * ( m_Fbg[binidx] * tau_wfs + pow( m_ron_wfs[binidx], 2 ) ) );
1852template <
typename realT,
class inputSpectT,
typename iosT>
1855 if( m == 0 and n == 0 )
1861 tau_wfs = optimumTauWFS( m, n, d, b );
1865 if( m_wfsBeta == 0 )
1866 mxThrowException(
err::paramnotset,
"aoSystem::measurementError",
"The WFS is not assigned." );
1868 realT beta_p = m_wfsBeta->beta_p( m, n, m_D, d, atm.r_0( m_lam_wfs ) ) / sqrt( m_opticalGain );
1869 realT beta_r = m_wfsBeta->beta_r( m, n, m_D, d, atm.r_0( m_lam_wfs ) );
1871 realT snr2 = signal2Noise2( Nph, tau_wfs, d, b );
1873 return ( pow( beta_r, 2 ) / snr2 + pow( beta_p, 2 ) / Nph ) * pow( m_lam_wfs / m_lam_sci, 2 );
1876template <
typename realT,
class inputSpectT,
typename iosT>
1879 if( m_specsChanged || m_dminChanged )
1881 return m_wfeMeasurement;
1884template <
typename realT,
class inputSpectT,
typename iosT>
1887 if( m == 0 and n == 0 )
1890 realT k = sqrt( m * m + n * n ) / m_D;
1895 tau_wfs = optimumTauWFS( m, n, d, b );
1899 realT tau = tau_wfs + m_deltaTau;
1902 return psd( atm, 0, k, m_secZeta ) / pow( m_D, 2 ) * pow( atm.lam_0() / m_lam_sci, 2 ) *
1903 sqrt( atm.X( k, m_lam_sci, m_secZeta ) ) * pow(
math::two_pi<realT>() * atm.v_wind() * k, 2 ) *
1907template <
typename realT,
class inputSpectT,
typename iosT>
1910 if( m_specsChanged || m_dminChanged )
1912 return m_wfeTimeDelay;
1915template <
typename realT,
class inputSpectT,
typename iosT>
1918 realT k = sqrt( m * m + n * n ) / m_D;
1921 return psd( atm, 0, k, m_secZeta ) / pow( m_D, 2 ) * pow( atm.lam_0() / m_lam_sci, 2 );
1924template <
typename realT,
class inputSpectT,
typename iosT>
1927 if( m_specsChanged || m_dminChanged )
1929 return m_wfeFitting;
1932template <
typename realT,
class inputSpectT,
typename iosT>
1935 return C4var( m, n );
1938template <
typename realT,
class inputSpectT,
typename iosT>
1941 if( m_specsChanged || m_dminChanged )
1943 return m_wfeChromScintOPD;
1946template <
typename realT,
class inputSpectT,
typename iosT>
1949 return C5var( m, n );
1952 int mn_max = floor(0.5*m_D/d_opt());
1956 for(
int m = -mn_max; m <= mn_max; ++m)
1958 for(
int n = -mn_max; n <= mn_max; ++n)
1960 if(n == 0 && m == 0)
continue;
1970template <
typename realT,
class inputSpectT,
typename iosT>
1973 return C6var( m, n );
1976template <
typename realT,
class inputSpectT,
typename iosT>
1979 if( m_specsChanged || m_dminChanged )
1981 return m_wfeChromIndex;
1984template <
typename realT,
class inputSpectT,
typename iosT>
1987 return C7var( m, n );
1990template <
typename realT,
class inputSpectT,
typename iosT>
1993 if( m_specsChanged || m_dminChanged )
1995 return m_wfeAnisoOPD;
1998template <
typename realT,
class inputSpectT,
typename iosT>
2004 int mn_max = floor(0.5*m_D/d_opt());
2008 for(
int m = -mn_max; m <= mn_max; ++m)
2010 for(
int n = -mn_max; n <= mn_max; ++n)
2012 if(n == 0 && m == 0)
continue;
2014 if( m_circularLimit )
2016 if( m*m + n*n > mn_max*mn_max )
continue;
2027template <
typename realT,
class inputSpectT,
typename iosT>
2036 mxError(
"aoSystem::optimumTauWFS", MXE_PARAMNOTSET,
"Diameter (D) not set." );
2042 mxError(
"aoSystem::optimumTauWFS", MXE_PARAMNOTSET,
"0-mag photon flux (F0) not set." );
2046 double binfact = 1.0;
2050 if( m_npix_wfs.size() > 1 )
2056 binfact = 1.0 / pow( (realT)( bbin + 1 ), 2 );
2060 realT k = sqrt( m * m + n * n ) / m_D;
2064 if( m_wfsBeta == 0 )
2065 mxThrowException(
err::paramnotset,
"aoSystem::beta_p",
"The WFS is not assigned." );
2067 realT beta_p = m_wfsBeta->beta_p( m, n, m_D, dact, atm.r_0( m_lam_wfs ) ) / sqrt( m_opticalGain );
2068 realT beta_r = m_wfsBeta->beta_r( m, n, m_D, dact, atm.r_0( m_lam_wfs ) ) / sqrt( m_opticalGain );
2071 realT a, b, c, d, e;
2074 realT Atmp = 2 * pow( atm.lam_0(), 2 ) * psd( atm, 0, k, m_secZeta ) / pow( m_D, 2 ) *
2075 ( atm.X( k, m_lam_wfs, m_secZeta ) ) * pow(
math::two_pi<realT>() * atm.v_wind() * k, 2 );
2078 b = Atmp * m_deltaTau;
2080 d = -1 * ( pow( m_lam_wfs, 2 ) / F ) *
2081 ( ( m_npix_wfs[binidx] * binfact * m_Fbg[binidx] / F ) * pow( beta_r, 2 ) + pow( beta_p, 2 ) );
2082 e = -2 * pow( m_lam_wfs, 2 ) * ( m_npix_wfs[binidx] * binfact ) * pow( m_ron_wfs[binidx], 2 ) * pow( beta_r, 2 ) /
2085 std::vector<std::complex<realT>> x;
2093 for(
int i = 0; i < 4; i++ )
2095 if( real( x[i] ) > 0 && imag( x[i] ) == 0 && real( x[i] ) > tauopt )
2096 tauopt = real( x[i] );
2099 if( tauopt < m_minTauWFS[binidx] )
2100 tauopt = m_minTauWFS[binidx];
2105template <
typename realT,
class inputSpectT,
typename iosT>
2109 return optimumTauWFS( m, n, m_d_opt, m_bin_opt );
2112template <
typename realT,
class inputSpectT,
typename iosT>
2115 if( !m_dminChanged )
2120 if( m_npix_wfs.size() < 1 )
2122 mxThrowException(
err::sizeerr,
"aoSystem::d_opt",
"npix_wfs must have at least one entry" );
2125 if( m_d_min.size() != m_npix_wfs.size() )
2127 mxThrowException(
err::sizeerr,
"aoSystem::d_opt",
"d_min must be the same size as npix_wfs" );
2130 if( m_ron_wfs.size() != m_npix_wfs.size() )
2132 mxThrowException(
err::sizeerr,
"aoSystem::d_opt",
"ron_wfs must be the same size as npix_wfs" );
2135 if( m_Fbg.size() != m_npix_wfs.size() )
2137 mxThrowException(
err::sizeerr,
"aoSystem::d_opt",
"F_bg must be the same size as npix_wfs" );
2140 if( m_minTauWFS.size() != m_npix_wfs.size() )
2142 mxThrowException(
err::sizeerr,
"aoSystem::d_opt",
"minTauWFS must be the same size as npix_wfs" );
2146 if( m_d_min.size() == 1 && m_d_min[0] == 0 )
2150 m_dminChanged =
false;
2156 m_d_opt = m_d_min[0];
2158 m_dminChanged =
false;
2167 if( m_npix_wfs.size() > 1 )
2170 best_d = m_d_min[0];
2172 realT bestStrehlOverall = 0;
2174 for(
int b = 0; b < m_npix_wfs.size(); ++b )
2176 realT d = m_d_min[b];
2177 realT s = strehl( d, b );
2178 realT bestStrehl = s;
2181 while( d < m_D / 2 )
2183 d += m_d_min[b] / m_optd_delta;
2186 if( s < bestStrehl )
2189 d -= m_d_min[b] / m_optd_delta;
2197 if( bestStrehl >= bestStrehlOverall )
2199 bestStrehlOverall = bestStrehl;
2205 m_bin_opt = best_idx;
2209 realT d = m_d_min[0];
2212 realT s = strehl( d, m_bin_opt );
2213 realT bestStrehl = s;
2215 while( d < m_D / 2 )
2217 d += m_d_min[0] / m_optd_delta;
2218 m_bin_opt = d / m_d_min[0] - 1;
2222 s = strehl( d, m_bin_opt );
2224 if( s < bestStrehl )
2227 d -= m_d_min[0] / m_optd_delta;
2228 m_bin_opt = d / m_d_min[0] - 1;
2241 realT d = m_d_min[0];
2244 realT s = strehl( d, m_bin_opt );
2245 realT bestStrehl = s;
2248 while( d < m_D / 2 )
2250 d += m_d_min[0] / m_optd_delta;
2252 s = strehl( d, m_bin_opt );
2253 if( s < bestStrehl )
2255 d -= m_d_min[0] / m_optd_delta;
2265 m_dminChanged =
false;
2270template <
typename realT,
class inputSpectT,
typename iosT>
2273 if( m == 0 and n == 0 )
2276 realT k = sqrt( m * m + n * n ) / m_D;
2278 realT kmax = m_fit_mn_max / m_D;
2279 realT kmin = 1. / m_D;
2282 if( m_ncp_alpha != 2 )
2285 ( pow( kmin, -m_ncp_alpha + 2 ) - pow( kmax, -m_ncp_alpha + 2 ) );
2291 return beta * ncpError() * pow( k, -m_ncp_alpha ) * pow( kmin, 2 );
2294template <
typename realT,
class inputSpectT,
typename iosT>
2300template <
typename realT,
class inputSpectT,
typename iosT>
2304 int mn_max = floor( 0.5 * m_D / d );
2309 for(
int m = -m_fit_mn_max; m <= m_fit_mn_max; ++m )
2311 for(
int n = -m_fit_mn_max; n <= m_fit_mn_max; ++n )
2314 if( m_circularLimit )
2316 if( m * m + n * n <= mn_max * mn_max )
2321 if( abs( m ) <= mn_max && abs( n ) <= mn_max )
2327 realT wfeMeasurement = measurementError( m, n, d, b );
2328 realT wfeTimeDelay = timeDelayError( m, n, d, b );
2329 realT wfeChromScintOPD = chromScintOPDError( m, n );
2330 realT wfeChromIndex = chromIndexError( m, n );
2331 realT wfeAnisoOPD = dispAnisoOPDError( m, n );
2333 realT wfeFitting = fittingError( m, n );
2335 if( wfeFitting < wfeMeasurement + wfeTimeDelay + wfeChromScintOPD + wfeChromIndex + wfeAnisoOPD )
2337 wfeVar += wfeFitting;
2341 wfeVar += wfeMeasurement + wfeTimeDelay + wfeChromScintOPD + wfeChromIndex + wfeAnisoOPD;
2346 wfeVar += fittingError( m, n );
2351 wfeVar += ncpError();
2352 return exp( -1 * wfeVar );
2355template <
typename realT,
class inputSpectT,
typename iosT>
2361 int mn_max = floor( 0.5 * m_D / d );
2363 m_wfeMeasurement = 0;
2365 m_wfeChromScintOPD = 0;
2366 m_wfeChromIndex = 0;
2373 m_controlledModes.resize( 2 * m_fit_mn_max + 1, 2 * m_fit_mn_max + 1 );
2374 m_controlledModes.setZero();
2377 for(
int m = -m_fit_mn_max; m <= m_fit_mn_max; ++m )
2379 for(
int n = -m_fit_mn_max; n <= m_fit_mn_max; ++n )
2381 if( m == 0 && n == 0 )
2387 if( m_circularLimit )
2389 if( m * m + n * n <= mn_max * mn_max )
2394 if( abs( m ) <= mn_max && abs( n ) <= mn_max )
2400 realT wfeMeasurement = measurementError( m, n, d, b );
2401 realT wfeTimeDelay = timeDelayError( m, n, d, b );
2402 realT wfeChromScintOPD = chromScintOPDError( m, n );
2403 realT wfeChromIndex = chromIndexError( m, n );
2404 realT wfeAnisoOPD = dispAnisoOPDError( m, n );
2406 realT wfeFitting = fittingError( m, n );
2408 if( wfeFitting < wfeMeasurement + wfeTimeDelay + wfeChromScintOPD + wfeChromIndex + wfeAnisoOPD )
2410 m_wfeFitting += wfeFitting;
2414 m_wfeMeasurement += wfeMeasurement;
2415 m_wfeTimeDelay += wfeTimeDelay;
2416 m_wfeChromScintOPD += wfeChromScintOPD;
2417 m_wfeChromIndex += wfeChromIndex;
2418 m_wfeAnisoOPD += wfeAnisoOPD;
2419 m_controlledModes( m_fit_mn_max + m, m_fit_mn_max + n ) = 1;
2424 m_wfeFitting += fittingError( m, n );
2429 m_wfeNCP = ncpError();
2431 m_wfeVar = m_wfeMeasurement + m_wfeTimeDelay + m_wfeFitting + m_wfeChromScintOPD + m_wfeChromIndex + m_wfeAnisoOPD +
2434 m_strehl = exp( -1 * m_wfeVar );
2436 m_specsChanged =
false;
2439template <
typename realT,
class inputSpectT,
typename iosT>
2442 if( m_specsChanged || m_dminChanged )
2448template <
typename realT,
class inputSpectT,
typename iosT>
2451 if( m_specsChanged || m_dminChanged )
2457template <
typename realT,
class inputSpectT,
typename iosT>
2458template <
typename varFuncT>
2461 if( m == 0 && n == 0 )
2471 if( doFittingError != FITTING_ERROR_NO )
2473 int mn_max = m_D / ( 2. * d_opt() );
2475 if( m_controlledModes( m_fit_mn_max + m, m_fit_mn_max + n ) == false )
2477 if( doFittingError == FITTING_ERROR_ZERO )
2480 realT fe = fittingError( m, n );
2482 realT k = sqrt( m * m + n * n ) / m_D;
2484 if( doFittingError == FITTING_ERROR_X )
2485 fe *= ( atm.X( k, m_lam_sci, m_secZeta ) );
2486 else if( doFittingError == FITTING_ERROR_Y )
2487 fe *= ( atm.Y( k, m_lam_sci, m_secZeta ) );
2490 std::cerr <<
"Unknown doFittingError\n";
2499 realT var = ( this->*varFunc )( m, n );
2504template <
typename realT,
class inputSpectT,
typename iosT>
2505template <
typename imageT,
typename CfuncT>
2508 int dim1 = im.rows();
2509 int dim2 = im.cols();
2511 int mc = 0.5 * ( dim1 - 1 );
2512 int nc = 0.5 * ( dim2 - 1 );
2514 for(
int i = 0; i < dim1; ++i )
2518 for(
int j = 0; j < dim2; ++j )
2522 im( i, j ) = ( this->*Cfunc )( m, n, normStrehl );
2527template <
typename realT,
class inputSpectT,
typename iosT>
2530 realT k = sqrt( m * m + n * n ) / m_D;
2532 return psd( atm, 0, k, m_secZeta ) / pow( m_D, 2 ) * pow( atm.lam_0() / m_lam_sci, 2 ) *
2533 ( atm.X( k, m_lam_sci, m_secZeta ) );
2536template <
typename realT,
class inputSpectT,
typename iosT>
2543template <
typename realT,
class inputSpectT,
typename iosT>
2544template <
typename imageT>
2550template <
typename realT,
class inputSpectT,
typename iosT>
2553 realT k = sqrt( m * m + n * n ) / m_D;
2556 return psd( atm, 0, k, m_secZeta ) / pow( m_D, 2 ) * pow( atm.lam_0() / m_lam_sci, 2 ) *
2557 ( atm.Y( k, m_lam_sci, m_secZeta ) );
2560template <
typename realT,
class inputSpectT,
typename iosT>
2566template <
typename realT,
class inputSpectT,
typename iosT>
2567template <
typename imageT>
2573template <
typename realT,
class inputSpectT,
typename iosT>
2577 return measurementError( m, n, d_opt(), m_bin_opt ) + timeDelayError( m, n, d_opt(), m_bin_opt );
2580template <
typename realT,
class inputSpectT,
typename iosT>
2586template <
typename realT,
class inputSpectT,
typename iosT>
2587template <
typename imageT>
2593template <
typename realT,
class inputSpectT,
typename iosT>
2599template <
typename realT,
class inputSpectT,
typename iosT>
2605template <
typename realT,
class inputSpectT,
typename iosT>
2606template <
typename imageT>
2612template <
typename realT,
class inputSpectT,
typename iosT>
2615 realT k = sqrt( m * m + n * n ) / m_D;
2619 return psd( atm, 0, k, m_secZeta ) / pow( m_D, 2 ) * pow( atm.lam_0() / m_lam_sci, 2 ) *
2620 atm.dX( k, m_lam_sci, m_lam_wfs );
2623template <
typename realT,
class inputSpectT,
typename iosT>
2629template <
typename realT,
class inputSpectT,
typename iosT>
2630template <
typename imageT>
2636template <
typename realT,
class inputSpectT,
typename iosT>
2639 realT k = sqrt( m * m + n * n ) / m_D;
2643 return psd( atm, 0, k, m_secZeta ) / pow( m_D, 2 ) * pow( atm.lam_0() / m_lam_sci, 2 ) *
2644 atm.dY( k, m_lam_sci, m_lam_wfs );
2647template <
typename realT,
class inputSpectT,
typename iosT>
2653template <
typename realT,
class inputSpectT,
typename iosT>
2654template <
typename imageT>
2660template <
typename realT,
class inputSpectT,
typename iosT>
2663 realT ni = atm.n_air( m_lam_sci );
2664 realT nw = atm.n_air( m_lam_wfs );
2666 return C0var( m, n ) * pow( ( ni - nw ) / ( ni - 1 ), 2 );
2669template <
typename realT,
class inputSpectT,
typename iosT>
2675template <
typename realT,
class inputSpectT,
typename iosT>
2676template <
typename imageT>
2682template <
typename realT,
class inputSpectT,
typename iosT>
2685 realT k = sqrt( m * m + n * n ) / m_D;
2688 return psd( atm, 0, k, m_secZeta ) / pow( m_D, 2 ) * pow( atm.lam_0() / m_lam_sci, 2 ) *
2689 atm.X_Z( k, m_lam_wfs, m_lam_sci, m_secZeta );
2692template <
typename realT,
class inputSpectT,
typename iosT>
2698template <
typename realT,
class inputSpectT,
typename iosT>
2699template <
typename imageT>
2705template <
typename realT,
class inputSpectT,
typename iosT>
2708 ios <<
"# AO Params:\n";
2709 ios <<
"# D = " << D() <<
'\n';
2711 if( m_d_min.size() > 0 )
2713 ios <<
"# d_min = " << d_min( (
size_t)0 );
2714 for(
size_t n = 1; n < m_d_min.size(); ++n )
2715 ios <<
',' << d_min( n );
2719 ios <<
"# d_min = null\n";
2721 ios <<
"# optd = " << std::boolalpha << m_optd <<
'\n';
2722 ios <<
"# d_opt_delta = " << optd_delta() <<
'\n';
2723 ios <<
"# lam_sci = " << lam_sci() <<
'\n';
2724 ios <<
"# F0 = " << F0() <<
'\n';
2725 ios <<
"# starMag = " << starMag() <<
'\n';
2726 ios <<
"# lam_sci = " << lam_sci() <<
'\n';
2727 ios <<
"# zeta = " << zeta() <<
'\n';
2728 ios <<
"# lam_wfs = " << lam_wfs() <<
'\n';
2730 if( npix_wfs().size() > 0 )
2732 ios <<
"# npix_wfs = " << npix_wfs( (
size_t)0 );
2733 for(
size_t n = 1; n < npix_wfs().size(); ++n )
2734 ios <<
',' << npix_wfs( n );
2738 ios <<
"# npix_wfs = null\n";
2740 if( ron_wfs().size() > 0 )
2742 ios <<
"# ron_wfs = " << ron_wfs( (
size_t)0 );
2743 for(
size_t n = 1; n < ron_wfs().size(); ++n )
2744 ios <<
',' << ron_wfs( n );
2748 ios <<
"# ron_wfs = null\n";
2750 if( Fbg().size() > 0 )
2752 ios <<
"# Fbg = " << Fbg( (
size_t)0 );
2753 for(
size_t n = 1; n < Fbg().size(); ++n )
2754 ios <<
',' << Fbg( n );
2758 ios <<
"# Fbg = null\n";
2760 if( minTauWFS().size() > 0 )
2762 ios <<
"# minTauWFS = " << minTauWFS( (
size_t)0 );
2763 for(
size_t n = 1; n < minTauWFS().size(); ++n )
2764 ios <<
',' << minTauWFS( n );
2768 ios <<
"# minTauWFS = null\n";
2770 ios <<
"# bin_npix = " << std::boolalpha << m_bin_npix <<
'\n';
2771 ios <<
"# tauWFS = " << tauWFS() <<
'\n';
2772 ios <<
"# optTau = " << std::boolalpha << m_optTau <<
'\n';
2773 ios <<
"# deltaTau = " << deltaTau() <<
'\n';
2774 ios <<
"# fit_mn_max = " << m_fit_mn_max <<
'\n';
2775 ios <<
"# spatialFilter_ku = " << m_spatialFilter_ku <<
'\n';
2776 ios <<
"# spatialFilter_kv = " << m_spatialFilter_kv <<
'\n';
2777 ios <<
"# ncp_wfe = " << m_ncp_wfe <<
'\n';
2778 ios <<
"# ncp_alpha = " << m_ncp_alpha <<
'\n';
2780 if( m_wfsBeta == 0 )
2781 mxThrowException(
err::paramnotset,
"aoSystem::beta_p",
"The WFS is not assigned." );
2782 m_wfsBeta->dumpWFS( ios );
2784 atm.dumpAtmosphere( ios );
2791template <
typename realT,
class inputSpectT,
typename iosT>
2794 using namespace mx::app;
2797 config.
add(
"aosys.wfs",
2805 "The WFS type: idealWFS, unmodPyWFS, asympModPyWFS, shwfs, calculatedWFS" );
2806 config.
add(
"aosys.wfs_beta_p",
2814 "The beta_p file path for calcualtedWFS" );
2815 config.
add(
"aosys.wfs_beta_r",
2823 "The beta_r file path for calcualtedWFS" );
2824 config.
add(
"aosys.wfs_sensitivity",
2826 "aosys.wfs_sensitivity",
2832 "Flag indicating that beta_p/beta_r are sensitivities (inverse) [default false]" );
2834 "aosys.D",
"",
"aosys.D", argType::Required,
"aosys",
"D",
false,
"real",
"The telescope diameter [m]" );
2835 config.
add(
"aosys.d_min",
2843 "The minimum actuator spacing [m]" );
2844 config.
add(
"aosys.optd",
2852 "Whether or not the actuator spacing is optimized" );
2853 config.
add(
"aosys.optd_delta",
2861 "The fractional change from d_min used in optimization. Set to 1 (default) for integer binnings, > 1 "
2862 "for finer sampling." );
2863 config.
add(
"aosys.F0",
2871 "Zero-mag photon flux, [photons/sec]" );
2872 config.
add(
"aosys.lam_wfs",
2880 "WFS wavelength [m]" );
2881 config.
add(
"aosys.npix_wfs",
2889 "The number of pixels in the WFS" );
2890 config.
add(
"aosys.ron_wfs",
2898 "WFS readout noise [photons/read]" );
2899 config.
add(
"aosys.bin_npix",
2907 "Whether or not WFS pixels are re-binned along with actuator spacing optimization" );
2908 config.
add(
"aosys.Fbg",
2916 "Background counts, [counts/pix/sec]" );
2917 config.
add(
"aosys.tauWFS",
2925 "WFS integration time [s]" );
2926 config.
add(
"aosys.minTauWFS",
2934 "Minimum WFS integration time [s]" );
2935 config.
add(
"aosys.deltaTau",
2944 config.
add(
"aosys.optTau",
2952 "Whether or not the integration time is optimized" );
2953 config.
add(
"aosys.lam_sci",
2961 "Science wavelength [m]" );
2963 "aosys.zeta",
"",
"aosys.zeta", argType::Required,
"aosys",
"zeta",
false,
"real",
"Zenith distance [rad]" );
2964 config.
add(
"aosys.fit_mn_max",
2972 "Maximum spatial frequency index to use for analysis" );
2973 config.
add(
"aosys.circularLimit",
2975 "aosys.circularLimit",
2981 " Flag to indicate that the spatial frequency limit is circular, not square." );
2982 config.
add(
"aosys.spatialFilter_ku",
2984 "aosys.spatialFilter_ku",
2990 "Spatial filter cutoff frequency in u [m^-1]" );
2991 config.
add(
"aosys.spatialFilter_kv",
2993 "aosys.spatialFilter_kv",
2999 "Spatial filter cutoff frequency in v [m^-1]" );
3000 config.
add(
"aosys.ncp_wfe",
3008 "NCP WFE between 1 lambda/D and fit_mn_max [rad^2]" );
3009 config.
add(
"aosys.ncp_alpha",
3017 "PSD index for NCP WFE" );
3019 "aosys.starMag",
"",
"aosys.starMag", argType::Required,
"aosys",
"starMag",
false,
"real",
"Star magnitude" );
3020 config.
add(
"aosys.starMags",
3028 "A vector of star magnitudes" );
3030 atm.setupConfig( config );
3031 psd.setupConfig( config );
3034template <
typename realT,
class inputSpectT,
typename iosT>
3038 if( config.
isSet(
"aosys.wfs" ) )
3041 config( wfsStr,
"aosys.wfs" );
3043 if( wfsStr ==
"ideal" )
3045 wfsBeta<wfs<realT, iosT>>( nullptr );
3047 else if( wfsStr ==
"unmodPyWFS" )
3049 wfsBeta<pywfsUnmod<realT, iosT>>( nullptr );
3051 else if( wfsStr ==
"asympModPyWFS" )
3053 wfsBeta<pywfsModAsymptotic<realT>>( nullptr );
3055 else if( wfsStr ==
"SHWFS" )
3057 wfsBeta<shwfs<realT, iosT>>( nullptr );
3059 else if( wfsStr ==
"calculatedWFS" )
3061 wfsBeta<calculatedWFS<realT, iosT>>( nullptr );
3064 config( cwfs->m_beta_p_file,
"aosys.wfs_beta_p" );
3065 config( cwfs->m_beta_r_file,
"aosys.wfs_beta_r" );
3066 bool sens = cwfs->m_sensitivity;
3067 config( sens,
"aosys.wfs_sensitivity" );
3068 if( config.isSet(
"aosys.wfs_sensitivity" ) )
3070 cwfs->m_sensitivity = sens;
3075 mxThrowException(
err::invalidconfig,
"aoSystem::loadConfig",
"unknown WFS " + wfsStr +
" specified" );
3084 config( nD,
"aosys.D" );
3085 if( config.
isSet(
"aosys.D" ) )
3089 std::vector<realT> nd_min = d_min();
3090 config( nd_min,
"aosys.d_min" );
3091 if( config.
isSet(
"aosys.d_min" ) )
3094 bool noptd = optd();
3095 config( noptd,
"aosys.optd" );
3096 if( config.
isSet(
"aosys.optd" ) )
3099 realT noptd_delta = optd_delta();
3100 config( noptd_delta,
"aosys.optd_delta" );
3101 if( config.
isSet(
"aosys.optd_delta" ) )
3102 optd_delta( noptd_delta );
3104 realT nlam_wfs = lam_wfs();
3105 config( nlam_wfs,
"aosys.lam_wfs" );
3106 if( config.
isSet(
"aosys.lam_wfs" ) )
3107 lam_wfs( nlam_wfs );
3110 std::vector<realT> nnpix_wfs = npix_wfs();
3111 config( nnpix_wfs,
"aosys.npix_wfs" );
3112 if( config.
isSet(
"aosys.npix_wfs" ) )
3113 npix_wfs( nnpix_wfs );
3116 std::vector<realT> nron_wfs = ron_wfs();
3117 config( nron_wfs,
"aosys.ron_wfs" );
3118 if( config.
isSet(
"aosys.ron_wfs" ) )
3119 ron_wfs( nron_wfs );
3122 std::vector<realT> nFbg = Fbg();
3123 config( nFbg,
"aosys.Fbg" );
3124 if( config.
isSet(
"aosys.Fbg" ) )
3128 std::vector<realT> nminTauWFS = minTauWFS();
3129 config( nminTauWFS,
"aosys.minTauWFS" );
3130 if( config.
isSet(
"aosys.minTauWFS" ) )
3131 minTauWFS( nminTauWFS );
3134 bool nbin_npix = bin_npix();
3135 config( nbin_npix,
"aosys.bin_npix" );
3136 if( config.
isSet(
"aosys.bin_npix" ) )
3137 bin_npix( nbin_npix );
3140 realT ntauWFS = tauWFS();
3141 config( ntauWFS,
"aosys.tauWFS" );
3142 if( config.
isSet(
"aosys.tauWFS" ) )
3146 realT ndeltaTau = deltaTau();
3147 config( ndeltaTau,
"aosys.deltaTau" );
3148 if( config.
isSet(
"aosys.deltaTau" ) )
3149 deltaTau( ndeltaTau );
3152 bool noptTau = optTau();
3153 config( noptTau,
"aosys.optTau" );
3154 if( config.
isSet(
"aosys.optTau" ) )
3158 realT nlam_sci = lam_sci();
3159 config( nlam_sci,
"aosys.lam_sci" );
3160 if( config.
isSet(
"aosys.lam_sci" ) )
3161 lam_sci( nlam_sci );
3164 realT nzeta = zeta();
3165 config( nzeta,
"aosys.zeta" );
3166 if( config.
isSet(
"aosys.zeta" ) )
3170 realT fmnm = fit_mn_max();
3171 config( fmnm,
"aosys.fit_mn_max" );
3172 if( config.
isSet(
"aosys.fit_mn_max" ) )
3176 bool cl = circularLimit();
3177 config( cl,
"aosys.circularLimit" );
3178 if( config.
isSet(
"aosys.circularLimit" ) )
3179 circularLimit( cl );
3182 realT ku = spatialFilter_ku();
3183 config( ku,
"aosys.spatialFilter_ku" );
3184 if( config.
isSet(
"aosys.spatialFilter_ku" ) )
3185 spatialFilter_ku( ku );
3188 realT kv = spatialFilter_kv();
3189 config( kv,
"aosys.spatialFilter_kv" );
3190 if( config.
isSet(
"aosys.spatialFilter_kv" ) )
3191 spatialFilter_kv( kv );
3194 realT nwfe = ncp_wfe();
3195 config( nwfe,
"aosys.ncp_wfe" );
3196 if( config.
isSet(
"aosys.ncp_wfe" ) )
3200 realT na = ncp_alpha();
3201 config( na,
"aosys.ncp_alpha" );
3202 if( config.
isSet(
"aosys.ncp_alpha" ) )
3207 config( nF0,
"aosys.F0" );
3208 if( config.
isSet(
"aosys.F0" ) )
3212 realT smag = starMag();
3213 config( smag,
"aosys.starMag" );
3214 if( config.
isSet(
"aosys.starMag" ) )
3217 atm.loadConfig( config );
3218 psd.loadConfig( config );
Provides a class to specify atmosphere parameters.
Calculate and provide constants related to adaptive optics.
Spatial power spectra used in adaptive optics.
Definitions of various analytic wavefront sensors.
A class to specify atmosphere parameters and perform related calculations.
Describes an analytic adaptive optics (AO) system.
bool circularLimit()
Get the value of the circularLimit flag.
realT m_ncp_wfe
Static WFE [m rms].
void loadMagAOX()
Load parameters corresponding to the MagAO-X system.
bool bin_npix()
Get the value of the pixel binngin flag.
void loadGuyon2005()
Load the default parameters from Guyon, 2005 .
bool optd()
Get the value of m_optd.
void C1Map(imageT &map, bool normStrehl)
Calculate a 2D map of contrast C1.
realT m_wfeNCP
Total WFE due to NCP errors [rad^2 at m_lam_sci].
realT F0()
Get the value of the 0 magnitude photon rate.
realT C5(realT m, realT n, bool normStrehl=true)
Calculate the contrast due to scintilation-amplitude chromaticity.
realT chromScintOPDErrorTotal()
Calculate the wavefront error due to scintillation chromaticity in the OPD over all spatial frequenci...
void C5Map(imageT &map, bool normStrehl)
Calculate a 2D map of contrast C5.
realT C1(realT m, realT n, bool normStrehl=true)
Calculate the contrast due to uncorrected amplitude, C1.
realT C0var(realT m, realT n)
Calculate the residual variance due to uncorrected phase at a spatial frequency.
realT C_(int m, int n, bool normStrehl, varFuncT varFunc, int doFittingError)
Worker function for raw contrast fuctions.
realT m_wfeVar
The WFE variance, in meters^2. Never use this directly, instead use wfeVar().
realT d_opt()
Calculate the optimum actuator spacing.
std::vector< realT > d_min()
Get the minimum subaperture sampling for all WFS modes.
realT ncpError()
Calculate the total NCP variance in rad^2.
realT m_opticalGain
The optical gain of the WFS, 0-1. Treated as a sensitivity reduction in measurement noise....
realT spatialFilter_kv()
Get the value of spatialFilter_kv.
bool m_specsChanged
Flag to indicate that a specification has changed.
bool m_circularLimit
Flag to indicate that the spatial frequency limit is circular, not square.
realT zeta()
Get the zenith angle.
realT Fg()
Get the photon rate at the current Star magnitude.
std::vector< realT > minTauWFS()
Get the values of the minimum WFS exposure time.
std::vector< realT > m_npix_wfs
Number of WFS pixels. One per WFS mode.
void calcStrehl()
Calculate the component WFE, total WFE, and Strehl ratio.
realT lam_sci()
Get the science wavelength.
realT m_wfeFitting
Total WFE due to fitting error [rad^2 at m_lam_sci].
realT m_wfeChromScintOPD
Total WFE due to the chromaticity of scintillation OPD [rad^2 at lam_sc].
realT m_strehl
Strehl ratio, a calculated quantity. Never use this directdy, instead use strehl().
realT deltaTau()
Get the value of m_deltaTau.
realT wfeVar()
Get the current value of the total WFE variance.
realT ncp_wfe()
Get the value of the non-common path WFE.
realT m_spatialFilter_ku
The spatial filter cutoff in u, [m^-1].
realT tauWFS()
Get the value of the minimum WFS exposure time.
std::vector< realT > m_ron_wfs
WFS readout noise [electrons/pix]. One per WFS mode.
bool m_ownWfsBeta
Flag indicating if the WFS beta_p pointer is owned by this instance.
realT m_secZeta
Secant of the Zenith angle (calculated)
realT m_starMag
The magnitude of the star.
realT C4(realT m, realT n, bool normStrehl=true)
Calculate the contrast due to scintilation-OPD chromaticity.
realT timeDelayError(realT m, realT n, realT d, int b)
Calculate the time delay at a spatial frequency at the optimum exposure time and the specified actuat...
realT chromIndexErrorTotal()
Calculate the wavefront error due to chromaticity in the index of refraction at a specific spatial fr...
realT chromIndexError(int m, int n)
Calculate the wavefront error due to chromaticity in the index of refraction at a given spatial frequ...
realT m_d_opt
Current optimum AO system actuator pitch [m].
realT C4var(realT m, realT n)
Calculate the residual variance due to scintilation-OPD chromaticity.
realT m_ncp_alpha
Power-law exponent for the NCP aberations. Default is 2.
realT beta_r(realT m, realT n)
Get the value of beta_r for a spatial frequency.
realT spatialFilter_ku()
Get the value of spatialFilter_ku.
realT m_spatialFilter_kv
The spatial filter cutoff in v, [m^-1].
void C7Map(imageT &map, bool normStrehl)
Calculate a 2D map of contrast C7.
realT C7var(realT m, realT n)
Calculate the residual variance due to dispersive anisoplanatism.
realT fittingErrorTotal()
Calculate the total fitting error over all uncorrected spatial frequencies.
bool m_bin_npix
Flag controlling whether or not to bin WFS pixels according to the actuator spacing.
void C0Map(imageT &map, bool normStrehl)
Calculate a 2D map of contrast C0.
bool optTau()
Get the value of m_optTau.
realT m_tauWFS
Actual WFS exposure time [sec].
realT beta_p(realT m, realT n)
Get the value of beta_p for a spatial frequency.
std::vector< realT > m_minTauWFS
Minimum WFS exposure time [sec]. One per WFS mode.
realT C5var(realT m, realT n)
Calculate the residual variance due to to scintilation-amplitude chromaticity.
wfs< realT, iosT > * m_wfsBeta
The WFS beta_p class.
realT secZeta()
Get the zecant of the zenith angle.
realT measurementError(realT m, realT n, realT d, int b)
Calculate the measurement noise at a spatial frequency and specified actuator spacing.
realT dispAnisoOPDErrorTotal()
Calculate the wavefront error due to dispersive anisoplanatism in the OPD over all specific spatial f...
realT m_lam_wfs
WFS wavelength [m].
void C3Map(imageT &map, bool normStrehl)
Calculate a 2D map of contrast C3.
realT m_F0
0 mag flux from star at WFS [photons/sec]
realT C2var(realT m, realT n)
Calculate the residual variance due to measurement and time delay errors in phase/OPD at a spatial fr...
bool m_dminChanged
Flag to indicate that d_min has changed.
std::vector< realT > Fbg()
Get the value of the background flux for all WFS modes.
realT timeDelayErrorTotal()
Calculate the time delay error over all corrected spatial frequencies.
realT dispAnisoOPDError(int m, int n)
Calculate the wavefront error due to dispersive anisoplanatism in the OPD at a given spatial frequenc...
realT m_zeta
Zenith angle [radians].
realT signal2Noise2(realT &Nph, realT tau_wfs, realT d, int b)
Calculate the terms of the signal to noise ratio squared (S/N)^2 for the WFS measurement.
realT C3var(realT m, realT n)
Calculate the residual variance due to measurement and time delay errors in amplitude at a spatial fr...
realT starMag()
Get the value of the Star's magnitude.
void loadGMagAOX()
Load parameters corresponding to the G-MagAO-X system.
realT m_D
Telescope diameter [m].
realT chromScintAmpError()
Calculate the wavefront error due to scintillation chromaticity in amplitude over all spatial frequen...
std::vector< realT > ron_wfs()
Get the value of the WFS readout noise for all WFS modes.
std::vector< realT > npix_wfs()
Get the number of pixels in the WFS for all WFS modes.
realT C7(realT m, realT n, bool normStrehl=true)
Calculate the contrast due to dispersive anisoplanatism.
realT measurementErrorTotal()
Calculate the total measurement error over all corrected spatial frequencies.
void setupConfig(app::appConfigurator &config)
Setup the configurator to configure this class.
realT strehl()
Get the current Strehl ratio.
realT C0(realT m, realT n, bool normStrehl=true)
Calculate the contrast due to uncorrected phase, C0.
realT C1var(realT m, realT n)
Calculate the residual variance due to uncorrected amplitude at a spatial frequency.
realT ncp_alpha()
Get the value of the non-common path WFE PSD index.
const Eigen::Array< int, -1, -1 > & controlledModes()
Access the array of controlledModes.
realT m_wfeTimeDelay
Total WFE due to time delay [rad^2 at m_lam_sci].
realT chromScintOPDError(int m, int n)
Calculate the wavefront error due to scintillation chromaticity in the OPD at a spatial frequency.
void C_Map(imageT &im, CfuncT Cfunc, bool normStrehl)
Worker function for the contrast-map functions.
int m_fit_mn_max
Maximum spatial frequency index to use for fitting error calculation.
iosT & dumpAOSystem(iosT &ios)
Output current parameters to a stream.
std::vector< realT > m_Fbg
Background flux, [photons/sec/pixel]. One per WFS mode.
void C4Map(imageT &map, bool normStrehl)
Calculate a 2D map of contrast C4.
realT C6var(realT m, realT n)
Calculate the residual variance due to chromaticity of the index of refraction of air.
realT C2(realT m, realT n, bool normStrehl=true)
Calculate the contrast due to measurement and time delay errors in phase/OPD at a spatial frequency.
realT m_wfeMeasurement
Total WFE due to measurement a error [rad^2 at m_lam_sci].
void C2Map(imageT &map, bool normStrehl)
Calculate a 2D map of contrast C2().
realT dispAnisoAmpError()
Calculate the wavefront error due to dispersive anisoplanatism in the amplitude at a specific spatial...
Eigen::Array< int, -1, -1 > m_controlledModes
Map of which modes are under control. Set by calcStrehl.
std::vector< realT > m_d_min
Minimum AO system actuator pitch [m]. One per WFS mode.
void loadConfig(app::appConfigurator &config)
Load the configuration of this class from a configurator.
realT C6(realT m, realT n, bool normStrehl=true)
Calculate the contrast due to chromaticity of the index of refraction of air.
int fit_mn_max()
Get the value of m_fit_mn_max.
realT opticalGain()
Get the value of the optical gain.
int bin_opt()
Get the value of the optimum binning factor/index.
realT m_wfeChromIndex
Total WFE due to the chromaticity of the index of refraction [rad^2 at lam_Sci].
realT m_lam_sci
Science wavelength [m].
realT optimumTauWFS(realT m, realT n, realT d, int b)
Calculate the optimum exposure time for a given spatial frequency at a specified actuator spacing.
realT m_deltaTau
Loop latency [sec].
void C6Map(imageT &map, bool normStrehl)
Calculate a 2D map of contrast C6.
realT lam_wfs()
Get the value of the WFS wavelength.
realT D()
Get the value of the primary mirror diamter.
realT m_wfeAnisoOPD
Total WFE due to dispersive anisoplanatism OPD.
realT C3(realT m, realT n, bool normStrehl=true)
Calculate the contrast due to measurement and time delay errors in amplitude at a spatial frequency.
realT fittingError(realT m, realT n)
Calculate the fitting error at a specific spatial frequency.
wfs< realT, iosT > * wfsBeta()
Get the WFS Beta pointer.
realT optd_delta()
Get the value of the fractional change in actuator spacing for optimization..
mxException for invalid config settings
mxException for parameters which aren't set
mxException for a size error
void quarticRoots(std::vector< std::complex< realT > > &x, realT a, realT b, realT c, realT d, realT e)
Find the roots of the general quartic equation.
constexpr floatT six_fifths()
Return 6/5 in the specified precision.
iosT & dumpGitStatus(iosT &ios)
Dump the current git status of the library to a stream.
The calculated WFS uses sensitivities provided by FITS files.
The ideal wavefront sensor sensitivity function.
Class to manage a set of configurable values, and read their values from config/ini files and the com...
void add(const configTarget &tgt)
Add a configTarget.
bool isSet(const std::string &name, std::unordered_map< std::string, configTarget > &targets)
Check if a target has been set by the configuration.