mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
aoAtmosphere.hpp
Go to the documentation of this file.
1/** \file aoAtmosphere.hpp
2 * \author Jared R. Males (jaredmales@gmail.com)
3 * \brief Provides a class to specify atmosphere parameters.
4 * \ingroup mxAO_files
5 *
6 */
7
8#ifndef aoAtmosphere_hpp
9#define aoAtmosphere_hpp
10
11#include <iostream>
12#include <numeric>
13#include <cmath>
14#include <cstdlib>
15#include <vector>
16#include <algorithm>
17
18#include "../../mxlib.hpp"
19#include "../../mxError.hpp"
20
21#include "aoConstants.hpp"
22
23#include "../../math/constants.hpp"
24
25#include "../../app/appConfigurator.hpp"
26
27namespace mx
28{
29namespace AO
30{
31namespace analysis
32{
33
34/// A class to specify atmosphere parameters and perform related calculations.
35/**
36 * \todo layer outer scales need work
37 * \todo layer PSD params isn't finished. Need to push that to PSD itself, and manage things like wavelength
38 * dependence.
39 *
40 * \tparam realT is the real floating type in which all calculations are performed.
41 *
42 * \ingroup mxAOAnalytic
43 */
44template <typename _realT>
46{
47 public:
48 typedef _realT realT; ///< The real floating type in which all calculations are performed.
49
50 /// Constructor
52
53 protected:
54 realT m_r_0; ///< Fried's parameter, in m
55
56 realT m_lam_0{ 0.5e-6 }; ///< Wavelength of Fried's parameter, in m
57
58 std::vector<realT> m_layer_Cn2; ///< Vector of layer strengths.
59
60 std::vector<realT> m_L_0; ///< The outer scale, in m
61
62 std::vector<realT> m_l_0; ///< The inner scale of each layer, in m
63
64 bool m_nonKolmogorov{ false }; ///< Flag indicating if non-Kolmogorov PSD parameters are used.
65
66 std::vector<realT> m_beta{ 1 }; ///< The PSD normalization when in non-Kolmogorov mode.
67
68 std::vector<realT> m_alpha{ 0 }; ///< The PSD exponent when in non-Kolmogorov mode.
69
70 std::vector<realT> m_beta_0{ 0 }; ///< The PSD constant when in non-Kolmogorov mode.
71
72 std::vector<realT> m_layer_z; ///< Vector of layer heights, in m, above the observatory.
73
74 realT m_h_obs{ 0 }; ///< Height of the observatory above sea level, in m.
75
76 realT m_H{ 8000 }; ///< The atmospheric scale height, in m.
77
78 std::vector<realT> m_layer_v_wind; ///< Vector of layer wind speeds, in m/s.
79
80 std::vector<realT> m_layer_dir; ///< Vector of layer wind directions, in radians.
81
82 bool m_v_wind_updated{ false }; ///< whether or not m_v_wind has been updated after changes
83
84 realT m_v_wind; ///< \f$ C_n^2 \f$ averaged windspeed
85
86 realT m_dir_wind; ///< \f$ C_n^2 \f$ averaged direction
87
88 bool m_z_mean_updated{ false }; ///< whether or not m_z_mean has been updated after changes
89
90 realT m_z_mean; ///< \f$ C_n^2 \f$ averaged layer height
91
92 /// Checks if layer vectors have consistent length.
93 /**
94 * \returns 0 if all layer vectors are the same length
95 * \returns -1 if not, and prints an error.
96 */
98
99 public:
100 /** \name PSD Parameters
101 * @{
102 */
103
104 /// Get the value of Fried's parameter r_0 at the reference wavelength lam_0.
105 /**
106 * \returns the curret value of m_r_0, in m.
107 */
108 realT r_0();
109
110 /// Get the value of Fried's parameter r_0 at the specified wavelength.
111 /**
112 * \note This does not set the value of r_0.
113 *
114 * \returns the current value of m_r_0 * pow(lam, 6/5), in m.
115 */
116 realT r_0( const realT &lam /**< [in] the wavelength, in m, at which to calculate r_0*/ );
117
118 /// Set the value of Fried's parameter and the reference wavelength.
119 /** If the provided reference wavelength is <=0, then 0.5 microns is used.
120 *
121 */
122 void r_0( const realT &r0, ///< [in] is the new value of r_0, m
123 const realT &l0 ///< [in] is the new value of lam_0 (in m), if 0 then 0.5e-6 is the default.
124 );
125
126 /// Get the current value of the reference wavelength.
127 /** This is the wavelength at which r_0 is specified.
128 *
129 * \returns the current value of m_lam_0, in m.
130 */
131 realT lam_0();
132
133 /// Get the strength of a single layer.
134 /**
135 * \returns the value of m_layer_Cn2[n].
136 */
137 realT layer_Cn2( const int n /**< [in] specifies the layer. */ );
138
139 /// Get the vector of layer strengths.
140 /**
141 * \returns a copy of the vector of layer strengths: m_layer_Cn2.
142 */
143 std::vector<realT> layer_Cn2();
144
145 /// Set the vector layer strengths, possibly calculating r_0.
146 /**
147 * If a reference wavelength is specified (l0 > 0), then r_0 is set from the layer strengths
148 * according to
149 *
150 * Regardless of what units the strengths are specified in, they are stored normalized so that
151 * \f$ \sum_n C_n^2 = 1 \f$.
152 *
153 */
154 void layer_Cn2( const std::vector<realT> &cn2, ///< [in] is a vector containing the layer strengths
155 const realT l0 = 0 ///< [in] [optional] if l0 > 0, then r_0 is set from the layer strengths.
156 );
157
158 /// Get the value of the outer scale for a single layer.
159 /**
160 * \returns the current value of m_L_0[n], in m.
161 */
162 realT L_0( const size_t &n );
163
164 /// Set the vector of layer outer scales.
165 /**
166 */
167 void L_0( const std::vector<realT> &L0 /**< [in] is the new vector of outer scales, in m */ );
168
169 /// Get the vector of outer scales.
170 /**
171 * \returns a copy of the vector of layer outer scales, in m
172 */
173 std::vector<realT> L_0();
174
175 /// Get the value of the inner scale for a single layer.
176 /**
177 * \returns the current value of m_l_0[n], in m.
178 */
179 realT l_0( const size_t &n );
180
181 /// Set the vector of layer inner scales.
182 /**
183 */
184 void l_0( const std::vector<realT> &l0 /**< [in] is the new vector of inner scales, in m*/ );
185
186 /// Get the vector of inner scales.
187 /**
188 * \returns a copy of the vector of layer inner scales, in m
189 */
190 std::vector<realT> l_0();
191
192 /// Set the value of m_nonKolmogorov
193 /** This flag indicates if non-Kolmogorov turbulence is being modeled.
194 */
195 void nonKolmogorov( const bool &nk /**< [in] the value of m_nonKolmogorov*/ );
196
197 /// Return the value of m_nonKolmogorov
198 /** This flag indicates if non-Kolmogorov turbulence is being modeled.
199 *
200 * \returns the current value of m_nonKolmogorov
201 */
203
204 /// Return the PSD index for a single layer.
205 /** Satifies the requirements of psdParamsT.
206 *
207 * If m_nonKolmogorov is false, this returns
208 * \[
209 \alpha = \frac{11}{3}
210 \]
211 * Otherwise it returns the current value of m_alpha[n].
212 *
213 * \returns the PSD index.
214 */
215 realT alpha( const size_t &n );
216
217 /// Set the vector of layer PSD indices.
218 /**
219 */
220 void alpha( const std::vector<realT> &alph /**< [in] is the new vector of PSD indices*/ );
221
222 /// Get the vector of PSD indices.
223 /**
224 * \returns a copy of the vector of PSD indices
225 */
226 std::vector<realT> alpha();
227
228 /// Return the PSD normalization for a single layer.
229 /** Satifies the requirements of psdParamsT.
230 *
231 * If m_nonKolmogorov is false, this returns
232 * \[
233 \beta = \frac{0.0218}{r_0^{5/3}}
234 \]
235 * Otherwise it returns the current value of m_beta[n].
236 *
237 * \returns the PSD normalization.
238 */
239 realT beta( const size_t &n );
240
241 /// Set the vector of layer PSD normalizations.
242 /**
243 */
244 void beta( const std::vector<realT> &bet /**< [in] is the new vector of PSD normalizations*/ );
245
246 /// Get the vector of PSD normalizations.
247 /**
248 * \returns a copy of the vector of PSD normalizations
249 */
250 std::vector<realT> beta();
251
252 /// Return the PSD constant for a single layer.
253 /** Satifies the requirements of psdParamsT.
254 *
255 * If m_nonKolmogorov is false, this returns
256 * \[
257 \beta_0 = 0
258 \]
259 * Otherwise it returns the current value of m_beta_0[n].
260 *
261 * \returns the PSD constant.
262 */
263 realT beta_0( const size_t &n );
264
265 /// Set the vector of layer PSD constants.
266 /**
267 */
268 void beta_0( const std::vector<realT> &bet /**< [in] is the new vector of PSD constants*/ );
269
270 /// Get the vector of PSD constants.
271 /**
272 * \returns a copy of the vector of PSD constants
273 */
274 std::vector<realT> beta_0();
275
276 /// Get the number of layers
277 /**
278 * \returns the size of the m_layer_Cn2 vector if m_nonKolmogrov==false
279 * \returns the size of the m_alpha vector if m_nonKolmogrov==true
280 */
281 size_t n_layers();
282
283 size_t currentLayer();
284
285 void currentLayer( size_t cl );
286
287 /// @}
288
289 /// Get the height of a single layer.
290 /**
291 * \returns the height of layer n, in m.
292 */
293 realT layer_z( const size_t n /**< [in] specifies the layer. */ );
294
295 /// Get the vector layer heights.
296 /**
297 * \returns a copy of the vector of layer heights:m_layer_z, in m.
298 */
299 std::vector<realT> layer_z();
300
301 /// Set the vector of layer heights.
302 /**
303 */
304 void layer_z( const std::vector<realT> &layz /**< [in] new vector of layer heights, in m. */ );
305
306 /// Get the height of the observatory.
307 /**
308 * \return the current value of m_h_obs, in m.
309 */
310 realT h_obs();
311
312 /// Set the height of the observatory.
313 /**
314 */
315 void h_obs( realT nh /**< [in] the new height of the observatory [m] */ );
316
317 /// Get the atmospheric scale height.
318 /**
319 * \returns the current value of m_H, in m.
320 */
321 realT H();
322
323 /// Set the atmospheric scale height.
324 /**
325 */
326 void H( realT nH /**< [in] the new value of m_H [m] */ );
327
328 /// Get the wind speed of a single layer.
329 /**
330 * \returns the value of m_layer_v_wind[n].
331 */
332 realT layer_v_wind( const int n /**< [in] specifies the layer. */ );
333
334 /// Get the vector of layer windspeeds
335 /**
336 * \returns a copy of the vector of layer windspeeds: m_layer_v_wind.
337 */
338 std::vector<realT> layer_v_wind();
339
340 /// Set the vector of layer windspeeds.
341 /**
342 * \param
343 */
344 void layer_v_wind( const std::vector<realT> &spd /**< [in] the new vector, which is copied to m_layer_v_wind. */ );
345
346 /// Get the wind direction of a single layer.
347 /**
348 * \returns the value of m_layer_dir[n], which is the wind direction in that layer in radians.
349 */
350 realT layer_dir( const int n /**< [in] specifies the layer. */ );
351
352 /// Get the vector of layer wind directions
353 /**
354 * \returns a copy of the vector of layer wind directions: m_layer_dir.
355 */
356 std::vector<realT> layer_dir();
357
358 /// Set the vector of layer wind directions.
359 /**
360 * \param
361 */
362 void layer_dir( const std::vector<realT>
363 &d /**< [in] the new vector of wind directions in radians, which is copied to m_layer_dir. */ );
364
365 /// Get the 5/3 moment weighted mean wind speed
366 /** Returns the weighted mean wind speed according to the 5/3's turbulence moment. This is defined as
367 *
368 \f[
369 \bar{v} = \left[\sum_i C_N^2(z_i) v_i^{5/3} \right]^{3/5}
370 \f]
371 * See Hardy (1998) Section 3.3.6. \cite hardy_1998.
372 *
373 * This is only re-calculated if either m_layer_Cn2 or m_layer_v_wind is changed, otherwise this just
374 * returns the value of m_v_wind.
375 *
376 * \returns the current value of m_v_wind.
377 */
378 realT v_wind();
379
380 /// Get the mean wind speed
381 /** Returns the weighted mean wind speed according to the 5/3's turbulence moment. This is defined as
382 *
383 \f[
384 \bar{v} = \sum_i C_N^2(z_i) v_i
385 \f]
386 * See Hardy (1998) Section 3.3.6. \cite hardy_1998.
387 *
388 * This is only re-calculated if either m_layer_Cn2 or m_layer_v_wind is changed, otherwise this just
389 * returns the value of m_v_wind.
390 *
391 * \returns the current value of m_v_wind.
392 */
394
395 /// Get the mean-squared wind speed
397
398 realT v_max()
399 {
400 auto max = std::max_element( std::begin( m_layer_v_wind ), std::end( m_layer_v_wind ) );
401
402 return *max;
403 }
404
405 /// Get the weighted mean wind direction
406 /** Returns the weighted mean wind direction according to the 5/3's turbulence moment. This is defined as
407 *
408 \f[
409 \bar{\theta} = \left[\sum_i C_N^2(z_i) \theta_i^{5/3} \right]^{3/5}
410 \f]
411 * See Hardy (1998) Section 3.3.6. \cite hardy_1998.
412 *
413 * This is only re-calculated if either m_layer_Cn2 or m_layer_v_wind is changed, otherwise this just
414 * returns the value of m_dir_wind.
415 *
416 * \returns the current value of m_dir_wind.
417 */
418 realT dir_wind();
419
420 protected:
421 /// Recalculate m_v_wind
422 /** Called by v_wind() whenever m_v_wind_updated is true.
423 *
424 * \todo handle dir_wind averaging across 0/2pi
425 */
427
428 public:
429 /// Set the weighted mean m_v_wind and renormalize the layer wind speeds
430 /** Calling this function changes the values of m_layer_v_wind so that the
431 * Layer averaged 5/3 \f$C_n^2\f$ moment of wind speed is the new value specified by vw.
432 *
433 */
434 void v_wind( const realT &vw /**< [in] the new value of m_v_wind. */ );
435
436 /// Get the weighted mean layer height
437 /** Returns the weighted layer height according to the 5/3's turbulence moment. This is defined as
438 \f[
439 \bar{z} = \left[\sum_i C_N^2(z_i) z_i^{5/3} \right]^{3/5}
440 \f]
441 * See Hardy (1998) Section 3.3.6 and 3.7.2. \cite hardy_1998.
442 *
443 * This is only re-calculated if either m_layer_Cn2 orm_layer_z is changed, otherwise this just
444 * returns the value of m_z_mean.
445 *
446 * \returns the current value of m_z_mean.
447 */
448 realT z_mean();
449
450 protected:
451 /// Recalculate m_z_mean
452 /** Called by z_mean() whenever m_z_mean_updated is true.
453 */
455
456 public:
457 /// Set the weighted mean m_z_mean and renormalize the layer heights
458 /** Calling this function changes the values ofm_layer_z so that the
459 * Layer averaged 5/3 \f$C_n^2\f$ moment of the height is the new value specified.
460 *
461 */
462 void z_mean( const realT &zm /**< [in] is the new value of m_v_wind. */ );
463
464 /// The fraction of the turbulence PSD in phase after Fresnel propagation.
465 /** See Equation (14) of Guyon (2005) \cite guyon_2005.
466 *
467 * \returns the value of the X function.
468 */
469 realT X( realT k, ///< [in] the spatial frequency, in inverse meters.
470 realT lam_sci, ///< [in] is the science observation wavelength.
471 realT secZ ///< [in] is the secant of the zenith distance.
472 );
473
474 /// The differential fraction of the turbulence PSD in phase after Fresnel propagation.
475 /** See Equation (25) of Guyon (2005) \cite guyon_2005.
476 *
477 * \returns the value of the dX function.
478 */
479 realT dX( realT k, ///< [in] the spatial frequency, in inverse meters.
480 realT lam_sci, ///< [in] is the science observation wavelength.
481 realT lam_wfs ///< [in] is the wavefront sensor wavelength.
482 );
483
484 /// The fraction of the turbulence PSD in amplitude after Fresnel propagation.
485 /** See Equation (15) of Guyon (2005) \cite guyon_2005.
486 *
487 * \returns the value of the Y function.
488 */
489 realT Y( realT k, ///< [in] the spatial frequency, in inverse meters.
490 realT lam_sci, ///< [in] is the science observation wavelength.
491 realT secZ ///< [in] is the secant of the zenith distance.
492 );
493
494 /// The differential fraction of the turbulence PSD in amplitude after Fresnel propagation.
495 /** See Equation (27) of Guyon (2005) \cite guyon_2005.
496 *
497 * \returns the value of the dY function.
498 */
499 realT dY( realT k, ///< [in] the spatial frequency, in inverse meters.
500 realT lam_sci, ///< [in] is the science observation wavelength.
501 realT lam_wfs ///< [in] is the wavefront sensor wavelength.
502 );
503
504 realT n_air( realT lam /**< [in]The wavelength*/ );
505
506 realT X_Z( realT k, ///< [in] the spatial frequency, in inverse meters
507 realT lambda_sci, ///< [in] is the science observation wavelength.
508 realT lambda_wfs, ///< [in] is the wavefront sensor wavelength.
509 realT secZ ///< [in] is the secant of the zenith distance.
510 );
511
512 /// Calculate the full-width at half-maximum of a seeing limited image for this atmosphere for a small telescope
513 /// (ignoring L_0)
514 /** Calculate the FWHM of a seeing limited image with the current parameters according to Floyd et al. (2010) \cite
515 floyd_2010 \f[ \epsilon_0 = 0.98\frac{\lambda_{sci}}{r_0(\lambda_sci)}. \f]
516 *
517 * \returns the value of the FWHM for the current atmosphere parameters.
518 */
519 realT fwhm0( realT lam_sci /**< [in] the wavelength of the science observation. */ );
520
521 /// Calculate the full-width at half-maximum of a seeing limited image for this atmosphere for a large telescope
522 /// (including L_0)
523 /** Calculate the FWHM of a seeing limited image with the current parameters according to Floyd et al. (2010) \cite
524 floyd_2010 \f[ \epsilon_0 = 0.98\frac{\lambda_{sci}}{r_0(\lambda_sci)}. \f]
525 * If there is an outer scale (_L_0 > 0), then a correction is applied according to Tokovinin (2002) \cite
526 tokovinin_2002 \f[ \left( \frac{\epsilon_{vK}}{\epsilon_0}\right)^2 = 1 - 2.183\left(
527 \frac{r_0(\lambda_{sci}}{L_0}\right)^{0.356} \f]
528 *
529 *
530 * \returns the value of the FWHM (\f$ \epsilon_{0/vK} \f$) for the current atmosphere parameters.
531 */
532 realT fwhm( realT lam_sci /**< [in] the wavelength of the science observation. */ );
533
534 /// Get the greenwood frequency at the reference wavelength
535 /**
536 * \todo derive full value of the constant
537 */
538 realT f_g();
539
540 /// Get the greenwood frequency at a specified wavelength
541 /**
542 *
543 */
544 realT f_g( realT lam_sci /**< [in] the wavelength of the science observation. */ );
545
546 /// Get tau_0 at the reference wavelength
547 /**
548 * \todo derive full value of the constant
549 */
550 realT tau_0();
551
552 /// Get tau_0 at a specified wavelength.
553 /**
554 *
555 */
556 realT tau_0( realT lam_sci /**< [in] the wavelength of the science observation. */ );
557
558 /// Scale v_wind so that tau_0 has the specified value at the specified wavelength.
559 /** Does not modify r_0.
560 */
561 void tau_0( realT tau_0, ///< [in] the desired tau_0
562 realT lam_sci ///< [in] the wavelength of the science observation.
563 );
564
565 /// Load the default atmosphere model from Guyon (2005).
566 /** Sets the parameters from Table 4 of Guyon (2005) \cite guyon_2005.
567 */
569
570 /// Load parameters corresponding to the median atmosphere of the GMT site survey at LCO.
571 /**
572 */
573 void loadLCO();
574
575 /// Set a single layer model.
576 /** Sets all layer vectors to size=1 and populates their fields based on these arguments.
577 *
578 */
579 void setSingleLayer( realT r0, ///< [in] is the new value of r_0
580 realT lam0, ///< [in] is the new value of lam_0, if 0 then 0.5 microns is the default.
581 realT L0, ///< [in] the new outer scale
582 realT l0, ///< [in] the new inner scale
583 realT lz, ///< [in] the layer height
584 realT vw, ///< [in] the layer wind-speed
585 realT dir ///< [in] the layer wind direction.
586 );
587
588 /// Output current parameters to a stream
589 /** Prints a formatted list of all current parameters.
590 *
591 * \tparam iosT is a std::ostream-like type.
592 *
593 * \todo update for new vector components (L_0, etc.)
594 */
595 template <typename iosT>
596 iosT &dumpAtmosphere( iosT &ios /**< [in] a std::ostream-like stream. */ );
597
598 /// \name mx::application support
599 /** @{
600 */
601
602 /// Setup the configurator to configure this class
603 /**
604 * \test Loading aoAtmosphere config settings \ref tests_ao_analysis_aoAtmosphere_config "[test doc]"
605 */
606 void setupConfig( app::appConfigurator &config /**< [in] the app::configurator object*/ );
607
608 /// Load the configuration of this class from a configurator
609 /**
610 * \test Loading aoAtmosphere config settings \ref tests_ao_analysis_aoAtmosphere_config "[test doc]"
611 */
612 void loadConfig( app::appConfigurator &config /**< [in] the app::configurator object*/ );
613
614 /// @}
615};
616
617template <typename realT>
621
622template <typename realT>
624{
625 size_t n = m_L_0.size();
626
627 if( m_l_0.size() != n )
628 {
629 mxError( "aoAtmosphere", MXE_SIZEERR, "mismatched layer numbers (inner scale vs. outer scale)" );
630 return -1;
631 }
632
633 if( m_layer_z.size() != n )
634 {
635 mxError( "aoAtmosphere", MXE_SIZEERR, "mismatched layer numbers (layer_z vs. outer scale)" );
636 return -1;
637 }
638
639 if( m_layer_Cn2.size() != n )
640 {
641 mxError( "aoAtmosphere", MXE_SIZEERR, "mismatched layer numbers (layer_Cn2 vs. outer scale)" );
642 return -1;
643 }
644
645 if( m_layer_dir.size() != n )
646 {
647 mxError( "aoAtmosphere", MXE_SIZEERR, "mismatched layer numbers (layer_dir vs. outer scale)" );
648 return -1;
649 }
650
651 if( m_layer_v_wind.size() != n )
652 {
653 mxError( "aoAtmosphere", MXE_SIZEERR, "mismatched layer numbers (layer_v_wind vs. outer scale)" );
654 return -1;
655 }
656
657 return 0;
658}
659
660template <typename realT>
662{
663 return m_r_0;
664}
665
666template <typename realT>
668{
669 return m_r_0 * pow( lam / m_lam_0, math::six_fifths<realT>() );
670}
671
672template <typename realT>
673void aoAtmosphere<realT>::r_0( const realT &r0, const realT &l0 )
674{
675 m_r_0 = r0;
676
677 if( l0 > 0 )
678 {
679 m_lam_0 = l0;
680 }
681 else
682 {
683 m_lam_0 = 0.5e-6;
684 }
685}
686
687template <typename realT>
689{
690 return m_lam_0;
691}
692
693template <typename realT>
695{
696 return m_layer_Cn2[n];
697}
698
699template <typename realT>
701{
702 return m_layer_Cn2;
703}
704
705template <typename realT>
706void aoAtmosphere<realT>::layer_Cn2( const std::vector<realT> &cn2, const realT l0 )
707{
708 m_layer_Cn2 = cn2;
709
710 realT layer_norm = 0;
711
712 for( size_t i = 0; i < m_layer_Cn2.size(); ++i )
713 {
714 layer_norm += cn2[i];
715 }
716
717 for( size_t i = 0; i < m_layer_Cn2.size(); ++i )
718 m_layer_Cn2[i] = m_layer_Cn2[i] / layer_norm;
719
720 if( l0 > 0 )
721 {
722 m_r_0 = 1.0 / pow( layer_norm * 5.520e13, math::three_fifths<realT>() );
723 m_lam_0 = l0;
724 }
725
726 m_v_wind_updated = false;
727 m_z_mean_updated = false;
728}
729
730template <typename realT>
732{
733 return m_L_0[n];
734}
735
736template <typename realT>
737void aoAtmosphere<realT>::L_0( const std::vector<realT> &L_0 )
738{
739 m_L_0 = L_0;
740}
741
742template <typename realT>
743std::vector<realT> aoAtmosphere<realT>::L_0()
744{
745 return m_L_0;
746}
747
748template <typename realT>
750{
751 return m_l_0[n];
752}
753
754template <typename realT>
755void aoAtmosphere<realT>::l_0( const std::vector<realT> &l_0 )
756{
757 m_l_0 = l_0;
758}
759
760template <typename realT>
761std::vector<realT> aoAtmosphere<realT>::l_0()
762{
763 return m_l_0;
764}
765
766template <typename realT>
768{
769 m_nonKolmogorov = nk;
770}
771
772template <typename realT>
774{
775 return m_nonKolmogorov;
776}
777
778template <typename realT>
780{
781 if( !m_nonKolmogorov )
782 {
784 }
785 else
786 {
787 return m_alpha[n];
788 }
789}
790
791template <typename realT>
792void aoAtmosphere<realT>::alpha( const std::vector<realT> &alph )
793{
794 m_nonKolmogorov = true;
795 m_alpha = alph;
796}
797
798template <typename realT>
799std::vector<realT> aoAtmosphere<realT>::alpha()
800{
801 return m_alpha;
802}
803
804template <typename realT>
806{
807 if( !m_nonKolmogorov )
808 {
809 return constants::a_PSD<realT>() * pow( m_r_0, -math::five_thirds<realT>() );
810 ;
811 }
812 else
813 {
814 return m_beta[n];
815 }
816}
817
818template <typename realT>
819void aoAtmosphere<realT>::beta( const std::vector<realT> &bet )
820{
821 m_nonKolmogorov = true;
822 m_beta = bet;
823}
824
825template <typename realT>
826std::vector<realT> aoAtmosphere<realT>::beta()
827{
828 return m_beta;
829}
830
831template <typename realT>
833{
834 if( !m_nonKolmogorov )
835 {
836 return 0;
837 }
838 else
839 {
840 return m_beta_0[n];
841 }
842}
843
844template <typename realT>
845void aoAtmosphere<realT>::beta_0( const std::vector<realT> &bet )
846{
847 m_nonKolmogorov = true;
848 m_beta_0 = bet;
849}
850
851template <typename realT>
852std::vector<realT> aoAtmosphere<realT>::beta_0()
853{
854 return m_beta_0;
855}
856
857template <typename realT>
859{
860 if( m_nonKolmogorov )
861 return m_alpha.size();
862 else
863 return m_layer_Cn2.size();
864}
865
866template <typename realT>
868{
869 return m_layer_z[n];
870}
871
872template <typename realT>
873void aoAtmosphere<realT>::layer_z( const std::vector<realT> &z )
874{
875 m_layer_z = z;
876 m_z_mean_updated = false;
877}
878
879template <typename realT>
881{
882 return m_layer_z;
883}
884
885template <typename realT>
887{
888 return m_h_obs;
889}
890
891template <typename realT>
893{
894 m_h_obs = nh;
895}
896
897template <typename realT>
899{
900 return m_H;
901}
902
903template <typename realT>
905{
906 m_H = nH;
907}
908
909template <typename realT>
911{
912 return m_layer_v_wind[n];
913}
914
915template <typename realT>
917{
918 return m_layer_v_wind;
919}
920
921template <typename realT>
922void aoAtmosphere<realT>::layer_v_wind( const std::vector<realT> &v )
923{
924 m_layer_v_wind = v;
925 m_v_wind_updated = false;
926}
927
928template <typename realT>
930{
931 return m_layer_dir[n];
932}
933
934template <typename realT>
936{
937 return m_layer_dir;
938}
939
940template <typename realT>
941void aoAtmosphere<realT>::layer_dir( const std::vector<realT> &dir )
942{
943 m_layer_dir = dir;
944 m_v_wind_updated = false;
945}
946
947template <typename realT>
949{
950 if( m_v_wind_updated == false )
951 update_v_wind();
952 return m_v_wind;
953}
954
955template <typename realT>
957{
958 if( m_v_wind_updated == false )
959 update_v_wind();
960 return m_dir_wind;
961}
962
963template <typename realT>
965{
966
967 if( m_layer_v_wind.size() == 0 || m_layer_Cn2.size() == 0 )
968 {
969 m_v_wind = 0;
970 m_dir_wind = 0;
971 m_v_wind_updated = true;
972 return;
973 }
974
975 m_v_wind = 0;
976
977 realT s = 0;
978 realT c = 0;
979
980 for( size_t i = 0; i < m_layer_Cn2.size(); ++i )
981 {
982 m_v_wind += m_layer_Cn2[i] * pow( m_layer_v_wind[i], math::five_thirds<realT>() );
983 s += pow( m_layer_v_wind[i], math::five_thirds<realT>() ) *
984 sin( m_layer_dir[i] ); // pow( sin(m_layer_dir[i]), math::five_thirds<realT>() );
985 c += pow( m_layer_v_wind[i], math::five_thirds<realT>() ) *
986 cos( m_layer_dir[i] ); // pow( cos(m_layer_dir[i]), math::five_thirds<realT>() );
987 }
988
989 m_v_wind = pow( m_v_wind, math::three_fifths<realT>() );
990
991 // m_dir_wind = atan(pow(s, math::three_fifths<realT>()) / pow(c, math::three_fifths<realT>()));
992 m_dir_wind = atan( s / c );
993 if( m_dir_wind < 0 )
994 m_dir_wind += math::pi<realT>();
995
996 m_v_wind_updated = true;
997}
998
999template <typename realT>
1001{
1002 if( m_v_wind_updated == false )
1003 update_v_wind();
1004
1005 realT vw_old = m_v_wind;
1006
1007 m_v_wind = vw;
1008
1009 // Now update the layers if needed
1010 if( m_layer_v_wind.size() > 0 )
1011 {
1012 for( size_t i = 0; i < m_layer_v_wind.size(); ++i )
1013 {
1014 m_layer_v_wind[i] = m_layer_v_wind[i] * ( vw / vw_old );
1015 }
1016 }
1017}
1018
1019template <typename realT>
1021{
1022 if( m_z_mean_updated == false )
1023 update_z_mean();
1024 return m_z_mean;
1025}
1026
1027template <typename realT>
1029{
1030 if( m_layer_z.size() == 0 || m_layer_Cn2.size() == 0 )
1031 {
1032 m_z_mean = 0;
1033 m_z_mean_updated = true;
1034 return;
1035 }
1036
1037 m_z_mean = 0;
1038
1039 for( size_t i = 0; i < m_layer_Cn2.size(); ++i )
1040 {
1041 m_z_mean += m_layer_Cn2[i] * pow( m_layer_z[i], math::five_thirds<realT>() );
1042 }
1043
1044 m_z_mean = pow( m_z_mean, math::three_fifths<realT>() );
1045
1046 m_z_mean_updated = true;
1047}
1048
1049template <typename realT>
1051{
1052 if( m_z_mean_updated == false )
1053 update_z_mean();
1054
1055 realT zh_old = m_z_mean;
1056
1057 m_z_mean = zm;
1058
1059 // Now update the layers if needed
1060 if( m_layer_z.size() > 0 )
1061 {
1062 for( size_t i = 0; i < m_layer_z.size(); ++i )
1063 {
1064 m_layer_z[i] = m_layer_z[i] * ( zm / zh_old );
1065 }
1066 }
1067}
1068
1069template <typename realT>
1071{
1072 realT c = 0;
1073
1074 for( size_t i = 0; i < m_layer_Cn2.size(); ++i )
1075 {
1076 c += m_layer_Cn2[i] * pow( cos( math::pi<realT>() * k * k * lam_sci * m_layer_z[i] * secZ ), 2 );
1077 }
1078
1079 return c;
1080}
1081
1082template <typename realT>
1084{
1085 realT c = 0;
1086
1087 for( size_t i = 0; i < m_layer_Cn2.size(); ++i )
1088 {
1089 c += m_layer_Cn2[i] * pow( ( cos( math::pi<realT>() * f * f * lam_sci * m_layer_z[i] ) -
1090 cos( math::pi<realT>() * f * f * lam_wfs * m_layer_z[i] ) ),
1091 2 );
1092 }
1093
1094 return c;
1095}
1096
1097template <typename realT>
1099{
1100 realT c = 0;
1101
1102 for( size_t i = 0; i < m_layer_Cn2.size(); ++i )
1103 {
1104 c += m_layer_Cn2[i] * pow( sin( math::pi<realT>() * k * k * lam_sci * m_layer_z[i] * secZ ), 2 );
1105 }
1106 return c;
1107}
1108
1109template <typename realT>
1111{
1112 realT c = 0;
1113
1114 for( size_t i = 0; i < m_layer_Cn2.size(); ++i )
1115 {
1116 c += m_layer_Cn2[i] * pow( ( sin( math::pi<realT>() * f * f * lam_sci * m_layer_z[i] ) -
1117 sin( math::pi<realT>() * f * f * lam_wfs * m_layer_z[i] ) ),
1118 2 );
1119 }
1120
1121 return c;
1122}
1123
1124template <typename realT>
1126{
1127 realT ll2 = static_cast<realT>( 1 ) / pow( lambda / 1e-6, 2 );
1128
1129 return 1.0 + 8.34213e-5 + 0.0240603 / ( 130.0 - ll2 ) + 0.00015997 / ( 38.9 - ll2 );
1130}
1131
1132template <typename realT>
1133realT aoAtmosphere<realT>::X_Z( realT k, realT lambda_i, realT lambda_wfs, realT secZ )
1134{
1135 realT c = 0;
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;
1139 realT x;
1140
1141 for( size_t i = 0; i < m_layer_Cn2.size(); ++i )
1142 {
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 ) *
1145 pow( sin( math::pi<realT>() * x * k * cos( 0. * 3.14 / 180. ) ), 2 );
1146 }
1147
1148 return 4 * c;
1149}
1150
1151template <typename realT>
1153{
1154 realT r0lam = r_0( lam_sci );
1155
1156 realT fwhm = 0.98 * ( lam_sci / r0lam );
1157
1158 return fwhm;
1159}
1160
1161template <typename realT>
1163{
1164 realT r0lam = r_0( lam_sci );
1165
1166 realT fwhm = 0.98 * ( lam_sci / r0lam );
1167
1168 ///\todo this needs to handle layers with different L_0
1169 if( L_0( 0 ) > 0 )
1170 fwhm *= sqrt( 1 - 2.183 * pow( r0lam / L_0( 0 ), 0.356 ) );
1171
1172 return fwhm;
1173}
1174
1175template <typename realT>
1177{
1178 return 0.428 * v_wind() / m_r_0;
1179}
1180
1181template <typename realT>
1183{
1184 return 0.428 * pow( m_lam_0 / lam_sci, math::six_fifths<realT>() ) * v_wind() / m_r_0;
1185}
1186
1187template <typename realT>
1189{
1190 return 0.134 / f_g();
1191}
1192
1193template <typename realT>
1195{
1196 return 0.134 / f_g( lam_sci );
1197}
1198
1199template <typename realT>
1201{
1202 realT vw = ( 0.134 / tau_0 ) / 0.428 * m_r_0 * pow( lam_sci / m_lam_0, math::six_fifths<realT>() );
1203 v_wind( vw );
1204}
1205
1206template <typename realT>
1208{
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 } );
1213
1214 r_0( 0.2, 0.5e-6 );
1215
1216 h_obs( 4200 );
1217}
1218
1219template <typename realT>
1221{
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 } );
1226
1227 r_0( 0.16, 0.5e-6 );
1228
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 } );
1231
1232 h_obs( 2400 );
1233}
1234
1235template <typename realT>
1237{
1238 r_0( r0, lam0 );
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 } ) );
1245}
1246
1247template <typename realT>
1248template <typename iosT>
1250{
1251 ios << "# Atmosphere Parameters:\n";
1252 ios << "# nonKolmogorov = " << std::boolalpha << nonKolmogorov() << '\n';
1253 ios << "# n_layers = " << n_layers() << '\n';
1254
1255 if( !m_nonKolmogorov )
1256 {
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';
1265 }
1266 if( m_nonKolmogorov )
1267 {
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';
1272 ios << "# beta = ";
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';
1280 }
1281
1282 ios << "# L_0 = ";
1283 for( size_t i = 0; i < n_layers() - 1; ++i )
1284 ios << L_0()[i] << ", ";
1285 ios << L_0()[n_layers() - 1] << '\n';
1286 ios << "# l_0 = ";
1287 for( size_t i = 0; i < n_layers() - 1; ++i )
1288 ios << l_0()[i] << ", ";
1289 ios << l_0()[n_layers() - 1] << '\n';
1290
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';
1301
1302 if( !m_nonKolmogorov )
1303 {
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';
1311 }
1312
1313 return ios;
1314}
1315
1316template <typename realT>
1318{
1319 using namespace mx::app;
1320
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",
1323 "",
1324 "atm.lam_0",
1325 argType::Required,
1326 "atm",
1327 "lam_0",
1328 false,
1329 "real",
1330 "The reference wavlength for r_0 [m]" );
1331 config.add(
1332 "atm.L_0", "", "atm.L_0", argType::Required, "atm", "L_0", false, "vector<real>", "Layer outer scales [m]" );
1333 config.add(
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",
1336 "",
1337 "atm.layer_z",
1338 argType::Required,
1339 "atm",
1340 "layer_z",
1341 false,
1342 "vector<real>",
1343 "layer heights [m]" );
1344 config.add(
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",
1348 "",
1349 "atm.layer_Cn2",
1350 argType::Required,
1351 "atm",
1352 "layer_Cn2",
1353 false,
1354 "vector<real>",
1355 "Layer Cn^2. Note that this does not set r_0." );
1356 config.add( "atm.layer_v_wind",
1357 "",
1358 "atm.layer_v_wind",
1359 argType::Required,
1360 "atm",
1361 "layer_v_wind",
1362 false,
1363 "vector<real>",
1364 "Layer wind speeds [m/s]" );
1365 config.add( "atm.layer_dir",
1366 "",
1367 "atm.layer_dir",
1368 argType::Required,
1369 "atm",
1370 "layer_dir",
1371 false,
1372 "vector<real>",
1373 "Layer wind directions [rad]" );
1374 config.add( "atm.v_wind",
1375 "",
1376 "atm.v_wind",
1377 argType::Required,
1378 "atm",
1379 "v_wind",
1380 false,
1381 "real",
1382 "Mean windspeed (5/3 momement), rescales layer speeds [m/s]" );
1383 config.add( "atm.tau_0",
1384 "",
1385 "atm.tau_0",
1386 argType::Required,
1387 "atm",
1388 "tau_0",
1389 false,
1390 "real",
1391 "AO time constant, sets v_wind and rescales layer speeds. [s]" );
1392 config.add( "atm.z_mean",
1393 "",
1394 "atm.z_mean",
1395 argType::Required,
1396 "atm",
1397 "z_mean",
1398 false,
1399 "real",
1400 "Mean layer height (5/3 momemnt), rescales layer heights [m/s]" );
1401 config.add( "atm.nonKolmogorov",
1402 "",
1403 "atm.nonKolmogorov",
1404 argType::Required,
1405 "atm",
1406 "nonKolmogorov",
1407 false,
1408 "bool",
1409 "Set to use a non-Kolmogorov PSD. See alpha and beta." );
1410 config.add( "atm.alpha",
1411 "",
1412 "atm.alpha",
1413 argType::Required,
1414 "atm",
1415 "alpha",
1416 false,
1417 "vector<real>",
1418 "Non-kolmogorov PSD exponent." );
1419 config.add( "atm.beta",
1420 "",
1421 "atm.beta",
1422 argType::Required,
1423 "atm",
1424 "beta",
1425 false,
1426 "vector<real>",
1427 "Non-kolmogorov PSD normalization." );
1428 config.add( "atm.beta_0",
1429 "",
1430 "atm.beta_0",
1431 argType::Required,
1432 "atm",
1433 "beta_0",
1434 false,
1435 "vector<real>",
1436 "Non-kolmogorov PSD constant." );
1437}
1438
1439template <typename realT>
1441{
1442 // Here "has side effecs" means that the set function does more than simply copy the value.
1443
1444 // The order of lam_0, Cn2, and r_0 is so that r_0 overrides the value set with Cn2 if lam_0 != 0.
1445 // lam_0 comes first because it calibrates r0 and Cn2
1446
1447 // lam_0
1448 config( m_lam_0, "atm.lam_0" );
1449
1450 // layer_Cn2
1451 std::vector<realT> lcn2 = m_layer_Cn2;
1452 config( lcn2, "atm.layer_Cn2" );
1453 if( config.isSet( "atm.layer_Cn2" ) )
1454 layer_Cn2( lcn2 );
1455
1456 realT r0 = r_0();
1457 config( r0, "atm.r_0" );
1458 if( config.isSet( "atm.r_0" ) )
1459 r_0( r0, m_lam_0 );
1460
1461 config( m_L_0, "atm.L_0" );
1462
1463 config( m_l_0, "atm.l_0" );
1464
1465 // Has side effects:
1466 std::vector<realT> layz = m_layer_z;
1467 config( layz, "atm.layer_z" ); // Do this no matter what to record source
1468 if( config.isSet( "atm.layer_z" ) )
1469 layer_z( layz ); // but only call this if changed
1470
1471 config( m_h_obs, "atm.h_obs" );
1472 config( m_H, "atm.H" );
1473
1474 // Has side effects:
1475 std::vector<realT> lvw = m_layer_v_wind;
1476 config( lvw, "atm.layer_v_wind" ); // Do this no matter what to record source
1477 if( config.isSet( "atm.layer_v_wind" ) )
1478 layer_v_wind( lvw ); // but only call this if changed
1479
1480 // Has side effects:
1481 std::vector<realT> ld = m_layer_dir;
1482 config( ld, "atm.layer_dir" ); // Do this no matter what to record source
1483 if( config.isSet( "atm.layer_dir" ) )
1484 layer_dir( ld ); // but only call this if changed
1485
1486 realT vw = m_v_wind;
1487 config( vw, "atm.v_wind" ); // Do this no matter what to record source
1488 if( config.isSet( "atm.v_wind" ) )
1489 v_wind( vw ); // but only call this if changed
1490
1491 realT t0 = tau_0();
1492 config( t0, "atm.tau_0" ); // Do this no matter what to record source
1493 if( config.isSet( "atm.tau_0" ) )
1494 {
1495 std::cerr << "setting tau_0 " << t0 << "\n";
1496 tau_0( t0, m_lam_0 ); // but only call this if changed
1497 }
1498 realT zm = m_z_mean;
1499 config( zm, "atm.z_mean" ); // Do this no matter what to record source
1500 if( config.isSet( "atm.z_mean" ) )
1501 z_mean( zm ); // but only call this if changed
1502
1503 config( m_nonKolmogorov, "atm.nonKolmogorov" );
1504
1505 std::vector<realT> a = m_alpha;
1506 config( a, "atm.alpha" );
1507 if( config.isSet( "atm.alpha" ) )
1508 alpha( a ); // this sets m_nonKolmogorov
1509
1510 std::vector<realT> b = m_beta;
1511 config( b, "atm.beta" );
1512 if( config.isSet( "atm.beta" ) )
1513 beta( b ); // this sets m_nonKolmogorov
1514
1515 std::vector<realT> b0 = m_beta_0;
1516 config( b0, "atm.beta_0" );
1517 if( config.isSet( "atm.beta_0" ) )
1518 beta_0( b0 ); // this sets m_nonKolmogorov
1519}
1520
1521extern template class aoAtmosphere<float>;
1522
1523extern template class aoAtmosphere<double>;
1524
1525extern template class aoAtmosphere<long double>;
1526
1527#ifdef HASQUAD
1528extern template class aoAtmosphere<__float128>;
1529#endif
1530
1531} // namespace analysis
1532} // namespace AO
1533} // namespace mx
1534
1535#endif // aoAtmosphere_hpp
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 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 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.
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.
The mxlib c++ namespace.
Definition mxError.hpp:106
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.