29#ifndef mx_AO_sim_turbLayer_hpp
30#define mx_AO_sim_turbLayer_hpp
36#include "../../improc/imageTransforms.hpp"
45template <
typename _aoSystemT>
54template <
typename _aoSystemT>
57 typedef _aoSystemT aoSystemT;
58 typedef typename aoSystemT::realT realT;
59 typedef Eigen::Array<realT, -1, -1> imageT;
80 imageT m_shiftPhaseWP;
82 imageT m_shiftPhaseWork;
101 void shift( realT dt );
122template <
typename aoSystemT>
130 if( m_parent ==
nullptr )
133 err::paramnotset,
"mx::AO::sim::turbLayer::setLayer",
"parent is not set (m_parent is nullptr)" );
136 if( m_parent->aosys() ==
nullptr )
139 "mx::AO::sim::turbLayer::setLayer",
140 "parent ao system is not set (m_parent->aosys() is nullptr)" );
143 realT vwind = m_parent->aosys()->atm.layer_v_wind( m_layerNo );
144 realT dwind = m_parent->aosys()->atm.layer_dir( m_layerNo );
145 realT pupD = m_parent->aosys()->D();
146 uint32_t wfSz = m_parent->wfSz();
148 m_dx = vwind * cos( dwind ) / ( pupD / wfSz );
149 m_dy = vwind * sin( dwind ) / ( pupD / wfSz );
154template <
typename aoSystemT>
155uint32_t turbLayer<aoSystemT>::layerNo()
160template <
typename aoSystemT>
161uint32_t turbLayer<aoSystemT>::scrnSz()
166template <
typename aoSystemT>
167void turbLayer<aoSystemT>::alloc()
169 if( m_parent ==
nullptr )
172 err::paramnotset,
"mx::AO::sim::turbLayer::alloc",
"parent is not set (m_parent is nullptr)" );
175 if( m_parent->aosys() ==
nullptr )
177 mxThrowException( err::paramnotset,
178 "mx::AO::sim::turbLayer::alloc",
179 "parent ao system is not set (m_parent->aosys() is nullptr)" );
182 m_phase.resize( m_scrnSz, m_scrnSz );
184 uint32_t wfSz = m_parent->wfSz();
186 uint32_t buffSz = m_parent->buffSz();
188 m_shiftPhaseWP.resize( wfSz + 2 * buffSz, wfSz + 2 * buffSz );
190 m_shiftPhase.resize( wfSz + 2 * buffSz, wfSz + 2 * buffSz );
192 m_shiftPhaseWork.resize( wfSz + 2 * buffSz, wfSz + 2 * buffSz );
198 m_last_wdx = m_scrnSz + 1;
199 m_last_wdy = m_scrnSz + 1;
202 realT r0 = m_parent->aosys()->atm.r_0( m_parent->aosys()->lam_sci() );
203 realT L0 = m_parent->aosys()->atm.L_0( m_layerNo );
204 realT l0 = m_parent->aosys()->atm.l_0( m_layerNo );
206 m_psd.resize( m_scrnSz, m_scrnSz );
208 m_freq.resize( m_scrnSz, m_scrnSz );
209 sigproc::frequencyGrid<imageT>( m_freq, m_parent->aosys()->D() / wfSz );
211 realT beta = 0.0218 / pow( r0, 5. / 3. );
212 realT sqrt_alpha = 0.5 * 11. / 3.;
216 L02 = 1.0 / ( L0 * L0 );
220#pragma omp parallel for
221 for(
size_t jj = 0; jj < m_scrnSz; ++jj )
223 for(
size_t ii = 0; ii < m_scrnSz; ++ii )
226 if( m_freq( ii, jj ) == 0 && L02 == 0 )
232 p = beta / pow( pow( m_freq( ii, jj ), 2 ) + L02, sqrt_alpha );
234 p *= exp( -1 * pow( m_freq( ii, jj ) * l0, 2 ) );
239 if( m_parent->aosys()->psd.subPiston() )
244 if( m_parent->aosys()->psd.subTipTilt() )
250 m_psd( ii, jj ) = sqrt( p * ( 1.0 - Ppiston - Ptiptilt ) );
254 if( m_parent->shLevel() > 0 )
257 if( m_parent->outerSubHarmonics() )
263 m_psd( 0, 1 ) *= 0.5 * 0.5 * onoff;
264 m_psd( 1, 0 ) *= 0.5 * 0.5 * onoff;
265 m_psd( m_psd.rows() - 1, 0 ) *= 0.5 * 0.5 * onoff;
266 m_psd( 0, m_psd.cols() - 1 ) *= 0.5 * 0.5 * onoff;
268 m_psd( 1, 1 ) *= 0.75 * 0.75 * onoff;
269 m_psd( m_psd.rows() - 1, m_psd.cols() - 1 ) *= 0.75 * 0.75 * onoff;
270 m_psd( m_psd.rows() - 1, 1 ) *= 0.75 * 0.75 * onoff;
271 m_psd( 1, m_psd.cols() - 1 ) *= 0.75 * 0.75 * onoff;
275template <
typename aoSystemT>
278 m_freq.resize( 0, 0 );
279 m_psd.resize( 0, 0 );
282template <
typename aoSystemT>
288 ddx = m_x0 + m_dx * dt;
289 ddy = m_y0 + m_dy * dt;
291 wdx = (int)trunc( ddx );
294 wdy = (int)trunc( ddy );
301 if( wdx != m_last_wdx || wdy != m_last_wdy )
314template <
typename aoSystemT>
320template <
typename aoSystemT>
326 ddx = uniVar * ( m_scrnSz );
327 ddy = uniVar * ( m_scrnSz );
329 wdx = (int)trunc( ddx );
331 wdy = (int)trunc( ddy );
342 if( ddx != 0 || ddy != 0 )
348 m_shiftPhase = m_shiftPhaseWP;
mxException for parameters which aren't set
A random number type, which functions like any other arithmetic type.
T2 jincN(const T1 &v, const T2 &x)
The JincN function.
T jinc(const T &x)
The Jinc function.
constexpr floatT six_fifths()
Return 6/5 in the specified precision.
A turbulent atmosphere simulator.
Simulation of a single turbulent layer.
void genDealloc()
Deallocate memory necessary for phase screen generation.
void initRandom()
Seed the uniform deviation. Call this if you intend to use shiftRandom.
mx::math::uniDistT< realT > uniVar
Uniform deviate, used in shiftRandom.
void shift(realT dt)
Shift to a timestep.
void shiftRandom(bool nofract=false)
Shift by a random amount using the uniform distribution.