8#ifndef aoAtmosphere_hpp
9#define aoAtmosphere_hpp
18#include "../../mxlib.hpp"
19#include "../../mxError.hpp"
23#include "../../math/constants.hpp"
25#include "../../app/appConfigurator.hpp"
44template <
typename _realT>
116 realT
r_0(
const realT &lam );
122 void r_0(
const realT &r0,
162 realT
L_0(
const size_t &n );
167 void L_0(
const std::vector<realT> &L0 );
179 realT
l_0(
const size_t &n );
184 void l_0(
const std::vector<realT> &l0 );
220 void alpha(
const std::vector<realT> &alph );
244 void beta(
const std::vector<realT> &bet );
268 void beta_0(
const std::vector<realT> &bet );
283 size_t currentLayer();
285 void currentLayer(
size_t cl );
304 void layer_z(
const std::vector<realT> &layz );
544 realT
f_g( realT lam_sci );
595 template <
typename iosT>
617template <
typename realT>
622template <
typename realT>
625 size_t n = m_L_0.size();
627 if( m_l_0.size() != n )
629 mxError(
"aoAtmosphere", MXE_SIZEERR,
"mismatched layer numbers (inner scale vs. outer scale)" );
633 if( m_layer_z.size() != n )
635 mxError(
"aoAtmosphere", MXE_SIZEERR,
"mismatched layer numbers (layer_z vs. outer scale)" );
639 if( m_layer_Cn2.size() != n )
641 mxError(
"aoAtmosphere", MXE_SIZEERR,
"mismatched layer numbers (layer_Cn2 vs. outer scale)" );
645 if( m_layer_dir.size() != n )
647 mxError(
"aoAtmosphere", MXE_SIZEERR,
"mismatched layer numbers (layer_dir vs. outer scale)" );
651 if( m_layer_v_wind.size() != n )
653 mxError(
"aoAtmosphere", MXE_SIZEERR,
"mismatched layer numbers (layer_v_wind vs. outer scale)" );
660template <
typename realT>
666template <
typename realT>
672template <
typename realT>
687template <
typename realT>
693template <
typename realT>
696 return m_layer_Cn2[n];
699template <
typename realT>
705template <
typename realT>
710 realT layer_norm = 0;
712 for(
size_t i = 0; i < m_layer_Cn2.size(); ++i )
714 layer_norm += cn2[i];
717 for(
size_t i = 0; i < m_layer_Cn2.size(); ++i )
718 m_layer_Cn2[i] = m_layer_Cn2[i] / layer_norm;
726 m_v_wind_updated =
false;
727 m_z_mean_updated =
false;
730template <
typename realT>
736template <
typename realT>
742template <
typename realT>
748template <
typename realT>
754template <
typename realT>
760template <
typename realT>
766template <
typename realT>
769 m_nonKolmogorov = nk;
772template <
typename realT>
775 return m_nonKolmogorov;
778template <
typename realT>
781 if( !m_nonKolmogorov )
791template <
typename realT>
794 m_nonKolmogorov =
true;
798template <
typename realT>
804template <
typename realT>
807 if( !m_nonKolmogorov )
818template <
typename realT>
821 m_nonKolmogorov =
true;
825template <
typename realT>
831template <
typename realT>
834 if( !m_nonKolmogorov )
844template <
typename realT>
847 m_nonKolmogorov =
true;
851template <
typename realT>
857template <
typename realT>
860 if( m_nonKolmogorov )
861 return m_alpha.size();
863 return m_layer_Cn2.size();
866template <
typename realT>
872template <
typename realT>
876 m_z_mean_updated =
false;
879template <
typename realT>
885template <
typename realT>
891template <
typename realT>
897template <
typename realT>
903template <
typename realT>
909template <
typename realT>
912 return m_layer_v_wind[n];
915template <
typename realT>
918 return m_layer_v_wind;
921template <
typename realT>
925 m_v_wind_updated =
false;
928template <
typename realT>
931 return m_layer_dir[n];
934template <
typename realT>
940template <
typename realT>
944 m_v_wind_updated =
false;
947template <
typename realT>
950 if( m_v_wind_updated ==
false )
955template <
typename realT>
958 if( m_v_wind_updated ==
false )
963template <
typename realT>
967 if( m_layer_v_wind.size() == 0 || m_layer_Cn2.size() == 0 )
971 m_v_wind_updated =
true;
980 for(
size_t i = 0; i < m_layer_Cn2.size(); ++i )
984 sin( m_layer_dir[i] );
986 cos( m_layer_dir[i] );
992 m_dir_wind = atan( s / c );
996 m_v_wind_updated =
true;
999template <
typename realT>
1002 if( m_v_wind_updated ==
false )
1005 realT vw_old = m_v_wind;
1010 if( m_layer_v_wind.size() > 0 )
1012 for(
size_t i = 0; i < m_layer_v_wind.size(); ++i )
1014 m_layer_v_wind[i] = m_layer_v_wind[i] * ( vw / vw_old );
1019template <
typename realT>
1022 if( m_z_mean_updated ==
false )
1027template <
typename realT>
1030 if( m_layer_z.size() == 0 || m_layer_Cn2.size() == 0 )
1033 m_z_mean_updated =
true;
1039 for(
size_t i = 0; i < m_layer_Cn2.size(); ++i )
1046 m_z_mean_updated =
true;
1049template <
typename realT>
1052 if( m_z_mean_updated ==
false )
1055 realT zh_old = m_z_mean;
1060 if( m_layer_z.size() > 0 )
1062 for(
size_t i = 0; i < m_layer_z.size(); ++i )
1064 m_layer_z[i] = m_layer_z[i] * ( zm / zh_old );
1069template <
typename realT>
1074 for(
size_t i = 0; i < m_layer_Cn2.size(); ++i )
1076 c += m_layer_Cn2[i] * pow( cos(
math::pi<realT>() * k * k * lam_sci * m_layer_z[i] * secZ ), 2 );
1082template <
typename realT>
1087 for(
size_t i = 0; i < m_layer_Cn2.size(); ++i )
1089 c += m_layer_Cn2[i] * pow( ( cos(
math::pi<realT>() * f * f * lam_sci * m_layer_z[i] ) -
1097template <
typename realT>
1102 for(
size_t i = 0; i < m_layer_Cn2.size(); ++i )
1104 c += m_layer_Cn2[i] * pow( sin(
math::pi<realT>() * k * k * lam_sci * m_layer_z[i] * secZ ), 2 );
1109template <
typename realT>
1114 for(
size_t i = 0; i < m_layer_Cn2.size(); ++i )
1116 c += m_layer_Cn2[i] * pow( ( sin(
math::pi<realT>() * f * f * lam_sci * m_layer_z[i] ) -
1124template <
typename realT>
1127 realT ll2 =
static_cast<realT>( 1 ) / pow( lambda / 1e-6, 2 );
1129 return 1.0 + 8.34213e-5 + 0.0240603 / ( 130.0 - ll2 ) + 0.00015997 / ( 38.9 - ll2 );
1132template <
typename realT>
1136 realT sinZ = sqrt( 1.0 - pow( 1.0 / secZ, 2 ) );
1137 realT tanZ = sinZ * secZ;
1138 realT x0 = ( n_air( lambda_wfs ) - n_air( lambda_i ) ) * m_H * tanZ * secZ;
1141 for(
size_t i = 0; i < m_layer_Cn2.size(); ++i )
1143 x = x0 * ( 1 - exp( ( m_layer_z[i] + m_h_obs ) / m_H ) );
1144 c += m_layer_Cn2[i] * pow( cos(
math::pi<realT>() * k * k * lambda_i * m_layer_z[i] * secZ ), 2 ) *
1151template <
typename realT>
1154 realT r0lam = r_0( lam_sci );
1156 realT fwhm = 0.98 * ( lam_sci / r0lam );
1161template <
typename realT>
1164 realT r0lam = r_0( lam_sci );
1166 realT fwhm = 0.98 * ( lam_sci / r0lam );
1170 fwhm *= sqrt( 1 - 2.183 * pow( r0lam / L_0( 0 ), 0.356 ) );
1175template <
typename realT>
1178 return 0.428 * v_wind() / m_r_0;
1181template <
typename realT>
1187template <
typename realT>
1190 return 0.134 / f_g();
1193template <
typename realT>
1196 return 0.134 / f_g( lam_sci );
1199template <
typename realT>
1206template <
typename realT>
1209 layer_Cn2( { 0.2283, 0.0883, 0.0666, 0.1458, 0.3350, 0.1350 } );
1210 layer_z( { 500, 1000, 2000, 4000, 8000, 16000 } );
1211 layer_v_wind( { 10., 10., 10., 10., 10., 10. } );
1212 layer_dir( { 0.0, 0.0, 0.0, 0.0, 0.0, 0.0 } );
1219template <
typename realT>
1222 layer_Cn2( { 0.42, 0.029, 0.062, 0.16, 0.11, 0.10, 0.12 } );
1223 layer_z( { 250., 500., 1000., 2000., 4000., 8000., 16000. } );
1224 layer_v_wind( { 10.0, 10.0, 20.0, 20.0, 25.0, 30.0, 25.0 } );
1225 layer_dir( { 1.05, 1.05, 1.31, 1.31, 1.75, 1.92, 1.75 } );
1227 r_0( 0.16, 0.5e-6 );
1229 L_0( { 25.0, 25.0, 25.0, 25.0, 25.0, 25.0, 25.0 } );
1230 l_0( { 0.0, 0, 0, 0, 0, 0, 0 } );
1235template <
typename realT>
1239 L_0( std::vector<realT>( { L0 } ) );
1240 l_0( std::vector<realT>( { l0 } ) );
1241 layer_Cn2( std::vector<realT>( { 1 } ) );
1242 layer_z( std::vector<realT>( { lz } ) );
1243 layer_v_wind( std::vector<realT>( { vw } ) );
1244 layer_dir( std::vector<realT>( { dir } ) );
1247template <
typename realT>
1248template <
typename iosT>
1251 ios <<
"# Atmosphere Parameters:\n";
1252 ios <<
"# nonKolmogorov = " << std::boolalpha << nonKolmogorov() <<
'\n';
1253 ios <<
"# n_layers = " << n_layers() <<
'\n';
1255 if( !m_nonKolmogorov )
1257 ios <<
"# r_0 = " << r_0() <<
'\n';
1258 ios <<
"# lam_0 = " << lam_0() <<
'\n';
1259 ios <<
"# tau_0 = " << tau_0( lam_0() ) <<
'\n';
1260 ios <<
"# FWHM = " << fwhm( lam_0() ) <<
'\n';
1261 ios <<
"# layer_Cn2 = ";
1262 for(
size_t i = 0; i < n_layers() - 1; ++i )
1263 ios << layer_Cn2()[i] <<
", ";
1264 ios << layer_Cn2()[n_layers() - 1] <<
'\n';
1266 if( m_nonKolmogorov )
1268 ios <<
"# alpha = ";
1269 for(
size_t i = 0; i < n_layers() - 1; ++i )
1270 ios << alpha()[i] <<
", ";
1271 ios << alpha()[n_layers() - 1] <<
'\n';
1273 for(
size_t i = 0; i < n_layers() - 1; ++i )
1274 ios << beta()[i] <<
", ";
1275 ios << beta()[n_layers() - 1] <<
'\n';
1276 ios <<
"# beta_0 = ";
1277 for(
size_t i = 0; i < n_layers() - 1; ++i )
1278 ios << beta_0()[i] <<
", ";
1279 ios << beta_0()[n_layers() - 1] <<
'\n';
1283 for(
size_t i = 0; i < n_layers() - 1; ++i )
1284 ios << L_0()[i] <<
", ";
1285 ios << L_0()[n_layers() - 1] <<
'\n';
1287 for(
size_t i = 0; i < n_layers() - 1; ++i )
1288 ios << l_0()[i] <<
", ";
1289 ios << l_0()[n_layers() - 1] <<
'\n';
1291 ios <<
"# layer_v_wind = ";
1292 for(
size_t i = 0; i < n_layers() - 1; ++i )
1293 ios << layer_v_wind()[i] <<
", ";
1294 ios << layer_v_wind()[n_layers() - 1] <<
'\n';
1295 ios <<
"# layer_dir = ";
1296 for(
size_t i = 0; i < n_layers() - 1; ++i )
1297 ios << layer_dir()[i] <<
", ";
1298 ios << layer_dir()[n_layers() - 1] <<
'\n';
1299 ios <<
"# mean v_wind = " << v_wind() <<
'\n';
1300 ios <<
"# mean dir_wind = " << dir_wind() <<
'\n';
1302 if( !m_nonKolmogorov )
1304 ios <<
"# layer_z = ";
1305 for(
size_t i = 0; i < n_layers() - 1; ++i )
1306 ios << layer_z()[i] <<
", ";
1307 ios << layer_z()[n_layers() - 1] <<
'\n';
1308 ios <<
"# mean z = " << z_mean() <<
'\n';
1309 ios <<
"# h_obs = " << h_obs() <<
'\n';
1310 ios <<
"# H = " << H() <<
'\n';
1316template <
typename realT>
1319 using namespace mx::app;
1321 config.
add(
"atm.r_0",
"",
"atm.r_0", argType::Required,
"atm",
"r_0",
false,
"real",
"Fried's parameter [m]" );
1322 config.
add(
"atm.lam_0",
1330 "The reference wavlength for r_0 [m]" );
1332 "atm.L_0",
"",
"atm.L_0", argType::Required,
"atm",
"L_0",
false,
"vector<real>",
"Layer outer scales [m]" );
1334 "atm.l_0",
"",
"atm.l_0", argType::Required,
"atm",
"l_0",
false,
"vector<real>",
"Layer inner scales [m]" );
1335 config.
add(
"atm.layer_z",
1343 "layer heights [m]" );
1345 "atm.h_obs",
"",
"atm.h_obs", argType::Required,
"atm",
"h_obs",
false,
"real",
"height of observatory [m]" );
1346 config.
add(
"atm.H",
"",
"atm.H", argType::Required,
"atm",
"H",
false,
"real",
"atmospheric scale heights [m]" );
1347 config.
add(
"atm.layer_Cn2",
1355 "Layer Cn^2. Note that this does not set r_0." );
1356 config.
add(
"atm.layer_v_wind",
1364 "Layer wind speeds [m/s]" );
1365 config.
add(
"atm.layer_dir",
1373 "Layer wind directions [rad]" );
1374 config.
add(
"atm.v_wind",
1382 "Mean windspeed (5/3 momement), rescales layer speeds [m/s]" );
1383 config.
add(
"atm.tau_0",
1391 "AO time constant, sets v_wind and rescales layer speeds. [s]" );
1392 config.
add(
"atm.z_mean",
1400 "Mean layer height (5/3 momemnt), rescales layer heights [m/s]" );
1401 config.
add(
"atm.nonKolmogorov",
1403 "atm.nonKolmogorov",
1409 "Set to use a non-Kolmogorov PSD. See alpha and beta." );
1410 config.
add(
"atm.alpha",
1418 "Non-kolmogorov PSD exponent." );
1419 config.
add(
"atm.beta",
1427 "Non-kolmogorov PSD normalization." );
1428 config.
add(
"atm.beta_0",
1436 "Non-kolmogorov PSD constant." );
1439template <
typename realT>
1448 config( m_lam_0,
"atm.lam_0" );
1451 std::vector<realT> lcn2 = m_layer_Cn2;
1452 config( lcn2,
"atm.layer_Cn2" );
1453 if( config.
isSet(
"atm.layer_Cn2" ) )
1457 config( r0,
"atm.r_0" );
1458 if( config.
isSet(
"atm.r_0" ) )
1461 config( m_L_0,
"atm.L_0" );
1463 config( m_l_0,
"atm.l_0" );
1466 std::vector<realT> layz = m_layer_z;
1467 config( layz,
"atm.layer_z" );
1468 if( config.
isSet(
"atm.layer_z" ) )
1471 config( m_h_obs,
"atm.h_obs" );
1472 config( m_H,
"atm.H" );
1475 std::vector<realT> lvw = m_layer_v_wind;
1476 config( lvw,
"atm.layer_v_wind" );
1477 if( config.
isSet(
"atm.layer_v_wind" ) )
1478 layer_v_wind( lvw );
1481 std::vector<realT> ld = m_layer_dir;
1482 config( ld,
"atm.layer_dir" );
1483 if( config.
isSet(
"atm.layer_dir" ) )
1486 realT vw = m_v_wind;
1487 config( vw,
"atm.v_wind" );
1488 if( config.
isSet(
"atm.v_wind" ) )
1492 config( t0,
"atm.tau_0" );
1493 if( config.
isSet(
"atm.tau_0" ) )
1495 std::cerr <<
"setting tau_0 " << t0 <<
"\n";
1496 tau_0( t0, m_lam_0 );
1498 realT zm = m_z_mean;
1499 config( zm,
"atm.z_mean" );
1500 if( config.
isSet(
"atm.z_mean" ) )
1503 config( m_nonKolmogorov,
"atm.nonKolmogorov" );
1505 std::vector<realT> a = m_alpha;
1506 config( a,
"atm.alpha" );
1507 if( config.
isSet(
"atm.alpha" ) )
1510 std::vector<realT> b = m_beta;
1511 config( b,
"atm.beta" );
1512 if( config.
isSet(
"atm.beta" ) )
1515 std::vector<realT> b0 = m_beta_0;
1516 config( b0,
"atm.beta_0" );
1517 if( config.
isSet(
"atm.beta_0" ) )
Calculate and provide constants related to adaptive optics.
A class to specify atmosphere parameters and perform related calculations.
void beta(const std::vector< realT > &bet)
Set the vector of layer PSD normalizations.
std::vector< realT > m_beta_0
The PSD constant when in non-Kolmogorov mode.
realT r_0(const realT &lam)
Get the value of Fried's parameter r_0 at the specified wavelength.
realT dir_wind()
Get the weighted mean wind direction.
std::vector< realT > alpha()
Get the vector of PSD indices.
realT lam_0()
Get the current value of the reference wavelength.
std::vector< realT > l_0()
Get the vector of inner scales.
void loadLCO()
Load parameters corresponding to the median atmosphere of the GMT site survey at LCO.
void layer_z(const std::vector< realT > &layz)
Set the vector of layer heights.
std::vector< realT > layer_v_wind()
Get the vector of layer windspeeds.
realT v_wind_mean2()
Get the mean-squared wind speed.
realT X(realT k, realT lam_sci, realT secZ)
The fraction of the turbulence PSD in phase after Fresnel propagation.
std::vector< realT > m_beta
The PSD normalization when in non-Kolmogorov mode.
iosT & dumpAtmosphere(iosT &ios)
Output current parameters to a stream.
realT h_obs()
Get the height of the observatory.
realT l_0(const size_t &n)
Get the value of the inner scale for a single layer.
realT m_z_mean
averaged layer height
realT layer_dir(const int n)
Get the wind direction of a single layer.
realT m_v_wind
averaged windspeed
void beta_0(const std::vector< realT > &bet)
Set the vector of layer PSD constants.
realT L_0(const size_t &n)
Get the value of the outer scale for a single layer.
realT m_dir_wind
averaged direction
std::vector< realT > m_alpha
The PSD exponent when in non-Kolmogorov mode.
std::vector< realT > beta()
Get the vector of PSD normalizations.
realT dY(realT k, realT lam_sci, realT lam_wfs)
The differential fraction of the turbulence PSD in amplitude after Fresnel propagation.
realT tau_0()
Get tau_0 at the reference wavelength.
std::vector< realT > L_0()
Get the vector of outer scales.
void layer_v_wind(const std::vector< realT > &spd)
Set the vector of layer windspeeds.
std::vector< realT > layer_dir()
Get the vector of layer wind directions.
void h_obs(realT nh)
Set the height of the observatory.
void loadGuyon2005()
Load the default atmosphere model from Guyon (2005).
void nonKolmogorov(const bool &nk)
Set the value of m_nonKolmogorov.
void layer_dir(const std::vector< realT > &d)
Set the vector of layer wind directions.
realT m_H
The atmospheric scale height, in m.
void L_0(const std::vector< realT > &L0)
Set the vector of layer outer scales.
void loadConfig(app::appConfigurator &config)
Load the configuration of this class from a configurator.
void update_z_mean()
Recalculate m_z_mean.
realT fwhm(realT lam_sci)
_realT realT
The real floating type in which all calculations are performed.
bool m_nonKolmogorov
Flag indicating if non-Kolmogorov PSD parameters are used.
void alpha(const std::vector< realT > &alph)
Set the vector of layer PSD indices.
realT beta_0(const size_t &n)
Return the PSD constant for a single layer.
realT beta(const size_t &n)
Return the PSD normalization for a single layer.
realT fwhm0(realT lam_sci)
realT X_Z(realT k, realT lambda_sci, realT lambda_wfs, realT secZ)
void setupConfig(app::appConfigurator &config)
Setup the configurator to configure this class.
std::vector< realT > m_L_0
The outer scale, in m.
void r_0(const realT &r0, const realT &l0)
Set the value of Fried's parameter and the reference wavelength.
void H(realT nH)
Set the atmospheric scale height.
aoAtmosphere()
Constructor.
std::vector< realT > m_layer_dir
Vector of layer wind directions, in radians.
std::vector< realT > layer_Cn2()
Get the vector of layer strengths.
void l_0(const std::vector< realT > &l0)
Set the vector of layer inner scales.
realT f_g(realT lam_sci)
Get the greenwood frequency at a specified wavelength.
realT m_r_0
Fried's parameter, in m.
void tau_0(realT tau_0, realT lam_sci)
Scale v_wind so that tau_0 has the specified value at the specified wavelength.
realT H()
Get the atmospheric scale height.
bool nonKolmogorov()
Return the value of m_nonKolmogorov.
void setSingleLayer(realT r0, realT lam0, realT L0, realT l0, realT lz, realT vw, realT dir)
Set a single layer model.
realT v_wind()
Get the 5/3 moment weighted mean wind speed.
std::vector< realT > layer_z()
Get the vector layer heights.
void layer_Cn2(const std::vector< realT > &cn2, const realT l0=0)
Set the vector layer strengths, possibly calculating r_0.
realT layer_Cn2(const int n)
Get the strength of a single layer.
realT m_h_obs
Height of the observatory above sea level, in m.
realT alpha(const size_t &n)
Return the PSD index for a single layer.
std::vector< realT > m_l_0
The inner scale of each layer, in m.
realT Y(realT k, realT lam_sci, realT secZ)
The fraction of the turbulence PSD in amplitude after Fresnel propagation.
realT layer_v_wind(const int n)
Get the wind speed of a single layer.
realT f_g()
Get the greenwood frequency at the reference wavelength.
realT z_mean()
Get the weighted mean layer height.
int checkLayers()
Checks if layer vectors have consistent length.
std::vector< realT > m_layer_Cn2
Vector of layer strengths.
std::vector< realT > m_layer_z
Vector of layer heights, in m, above the observatory.
size_t n_layers()
Get the number of layers.
std::vector< realT > beta_0()
Get the vector of PSD constants.
bool m_v_wind_updated
whether or not m_v_wind has been updated after changes
realT r_0()
Get the value of Fried's parameter r_0 at the reference wavelength lam_0.
std::vector< realT > m_layer_v_wind
Vector of layer wind speeds, in m/s.
realT v_wind_mean()
Get the mean wind speed.
void z_mean(const realT &zm)
Set the weighted mean m_z_mean and renormalize the layer heights.
realT tau_0(realT lam_sci)
Get tau_0 at a specified wavelength.
void v_wind(const realT &vw)
Set the weighted mean m_v_wind and renormalize the layer wind speeds.
realT m_lam_0
Wavelength of Fried's parameter, in m.
bool m_z_mean_updated
whether or not m_z_mean has been updated after changes
realT dX(realT k, realT lam_sci, realT lam_wfs)
The differential fraction of the turbulence PSD in phase after Fresnel propagation.
realT layer_z(const size_t n)
Get the height of a single layer.
void update_v_wind()
Recalculate m_v_wind.
constexpr floatT six_fifths()
Return 6/5 in the specified precision.
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.