mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
leakyIntegrator.hpp
Go to the documentation of this file.
1/** \file leakyIntegrator.hpp
2 * \author Jared R. Males (jaredmales@gmail.com)
3 * \brief Declares and defines the leaky integrator controller class for AO control.
4 * \ingroup mxAO_sim_files
5 *
6 */
7
8#ifndef __leakyIntegrator_hpp__
9#define __leakyIntegrator_hpp__
10
11namespace mx
12{
13namespace AO
14{
15namespace sim
16{
17
18template <typename _realT>
19struct wfMeasurement
20{
21 typedef _realT realT;
22
23 realT iterNo;
24
25 typedef Eigen::Array<realT, Eigen::Dynamic, Eigen::Dynamic> commandT;
26
27 commandT measurement;
28};
29
30/// Implements the leaky integrator controller.
31/**
32 * \tparam _realT is the floating point type for all calculations.
33 */
34template <typename _realT>
36{
37
38 public:
39 /// The real data type
40 typedef _realT realT;
41
42 /// The wavefront data type
44
45 /// The command type
46 typedef wfMeasurement<realT> commandT;
47
48 /// The image type, used here as a general storage array
49 typedef Eigen::Array<realT, Eigen::Dynamic, Eigen::Dynamic> imageT;
50
51 /// Default c'tor.
53
54 protected:
55 int _nModes; ///< The number of modes being filtered.
56
57 bool _openLoop; ///< If true, then commands are not integrated.
58
59 int _closingDelay; ///< If > 0, then the gains are ramped linearly up to this value. Default = 0.
60
61 int _lowOrders; ///< If > 0, then this sets the maximum mode number which is filtered. All remaining modes are set
62 ///< to 0. Default = 0.
63
64 imageT _gains; ///< Column-vector of gains
65
66 imageT _leaks; ///< Column-vector of leaks
67
68 imageT _commands; ///< Column-vector past commands
69
70 public:
71 int _lowOrderDelay;
72
73 /// Allocate and initialize all state.
74 /**
75 * \returns 0 on success, a negative integer otherwise.
76 */
77 int initialize( int nModes /**< [in] the number of modes to be filtered */ );
78
79 /// Get the number of modes
80 /** nModes is only set by calling initialize.
81 *
82 * \returns the current value of _nModes
83 */
84 int nModes();
85
86 /// Set the _openLoop flag.
87 /** If _openLoop is true, then commands are not filtered.
88 *
89 * \returns 0 on success, a negative integer on error.
90 */
91 int openLoop( bool ol /**< The new value of _openLoop */ );
92
93 /// Get the value of the _openLoop flag.
94 /**
95 * \returns the current value of _openLoop.
96 */
97 bool openLoop();
98
99 /// Set _closingDelay.
100 /** If _closingDelay > 0, then the gains are ramped linearly up to this value.
101 *
102 * \returns 0 on success, a negative integer on error.
103 */
104 int closingDelay( int cd /**< The new value of _closingDelay */ );
105
106 /// Get the value of the _closingDelay.
107 /**
108 * \returns the current value of _closingDelay.
109 */
110 int closingDelay();
111
112 /// Set _lowOrders.
113 /** If _lowOrders > 0, then this sets the maximum mode number which is filtered. All remaining modes are set to 0.
114 *
115 * \returns 0 on success, a negative integer on error.
116 */
117 int lowOrders( int lo /**< The new value of _lowOrders */ );
118
119 /// Get the value of the _lowOrders.
120 /**
121 * \returns the current value of _lowOrders.
122 */
123 int lowOrders();
124
125 /// Get the gain for a single mode.
126 /** \returns the gain value if mode exists.
127 */
128 realT gain( int i /**< [in] the mode number*/ );
129
130 /// Set the gain for a single mode.
131 /**
132 * \returns 0 on success, negative number on error.
133 */
134 int gain( int i, ///< [in] the mode number
135 realT g ///< [in] the new gain value
136 );
137
138 /// Set the gain for all modes to a single value
139 /**
140 * \returns 0 on success, negative number on error.
141 */
142 int gains( realT g /**< [in] the new gain value*/ );
143
144 /// Set the gains for all modes, using a vector to specify each gain
145 /** The vector must be exactly as long as _nModes.
146 *
147 * \returns 0 on success, negative number on error.
148 */
149 int gains( const std::vector<realT> &vgains /**< [in] vector of gains.*/ );
150
151 /// Set the gains for all modes, using a file to specify each gain
152 /** The file format is a simple ASCII single column, with 1 gain per line.
153 * Must be exactly as long as _nModes.
154 *
155 * \returns 0 on success, negative number on error.
156 */
157 int gains( const std::string &ogainf /**< [in] the name of the file, full path */ );
158
159 /// Set the leak for a single mode.
160 /**
161 * \returns 0 on success, negative number on error.
162 */
163 int leak( int i, ///< [in] the mode number
164 realT l ///< [in] the new leak value for this mode
165 );
166
167 /// Set a leak for all modes.
168 /**
169 * \returns 0 on success, negative number on error.
170 */
171 int leaks( realT l /**< [in] the new leak value to set for all modes */ );
172
173 /// Allocate the provided command structures
174 /** Used by the calling system to allocate the commands being passed between components.
175 *
176 * \returns 0 on success, negative number on error.
177 */
178 int initMeasurements( commandT &filtAmps, ///< The structure to contain the filtered commands
179 commandT &rawAmps ///< The structure to contain the raw commands
180 );
181
182 /// Apply the leaky integrator
183 /**
184 * \returns 0 on success, negative number on error.
185 */
186 int filterCommands( commandT &filtAmps, ///< [out] the filtered commands
187 commandT &rawAmps, ///< [in] the raw commands
188 int iterNo ///< [in] The current iteration number
189 );
190};
191
192template <typename realT>
194{
195 _nModes = 0;
196
197 _openLoop = false;
198
199 _closingDelay = 0;
200
201 _lowOrders = 0;
202 _lowOrderDelay = 0;
203}
204
205template <typename realT>
207{
208 _nModes = nModes;
209
210 _gains.resize( 1, nModes );
211 _leaks.resize( 1, nModes );
212 _commands.resize( 1, nModes );
213
214 _gains.setZero();
215
216 _leaks.setZero();
217
218 _commands.setZero();
219
220 return 0;
221}
222
223template <typename realT>
225{
226 return _nModes;
227}
228
229template <typename realT>
231{
232 _openLoop = ol;
233 return 0;
234}
235
236template <typename realT>
238{
239 return _openLoop;
240}
241
242template <typename realT>
244{
245 _closingDelay = cd;
246
247 return 0;
248}
249
250template <typename realT>
252{
253 return _closingDelay;
254}
255
256template <typename realT>
258{
259 _lowOrders = lo;
260
261 return 0;
262}
263
264template <typename realT>
266{
267 return _lowOrders;
268}
269
270template <typename realT>
272{
273 if( i < 0 || i >= _nModes )
274 {
275 mxError( "leakyIntegrator::gain", MXE_INVALIDARG, "mode index out of range" );
276 return 0; ///\retval 0 if the mode doesn't exist.
277 }
278
279 return _gains( i );
280}
281
282template <typename realT>
284{
285 if( i < 0 || i >= _nModes )
286 {
287 mxError( "leakyIntegrator::gain", MXE_INVALIDARG, "mode index out of range" );
288 return 0; ///\retval 0 if the mode doesn't exist.
289 }
290
291 _gains( 0, i ) = g;
292
293 return 0;
294}
295
296template <typename realT>
298{
299 for( int i = 0; i < _gains.cols(); ++i )
300 {
301 _gains( 0, i ) = g;
302 }
303
304 return 0;
305}
306
307template <typename realT>
308int leakyIntegrator<realT>::gains( const std::vector<realT> &gains )
309{
310
311 if( gains.size() != _gains.cols() )
312 {
313 mxError( "leakyIntegrator::gain", MXE_SIZEERR, "input gain vector not same size as number of modes" );
314 return -1; ///\retval -1 on vector size mismatch
315 }
316
317 for( int i = 0; i < _gains.cols(); ++i )
318 {
319 _gains( 0, i ) = gains[i];
320 }
321
322 return 0;
323}
324
325template <typename realT>
326int leakyIntegrator<realT>::gains( const std::string &ogainf )
327{
328 std::ifstream fin;
329 fin.open( ogainf );
330
331 if( !fin.good() )
332 {
333 mxError( "leakyIntegrator::gains", MXE_FILEOERR, "could not open gan file" );
334 return -1; /// \retval -1 if file open fails
335 }
336
337 std::string tmpstr;
338 realT g;
339 for( int i = 0; i < _gains.cols(); ++i )
340 {
341 fin >> tmpstr;
342 g = mx::convertFromString<realT>( tmpstr );
343 gain( i, g );
344 }
345
346 return 0;
347}
348
349template <typename realT>
351{
352 _leaks( 0, i ) = l;
353}
354
355template <typename realT>
357{
358 for( int i = 0; i < _leaks.cols(); ++i )
359 _leaks( 0, i ) = l;
360
361 return 0;
362}
363
364template <class realT>
366{
367 filtAmps.measurement.resize( 1, _nModes );
368 filtAmps.measurement.setZero();
369
370 rawAmps.measurement.resize( 1, _nModes );
371 rawAmps.measurement.setZero();
372
373 return 0;
374}
375
376template <class realT>
377int leakyIntegrator<realT>::filterCommands( commandT &filtAmps, commandT &rawAmps, int iterNo )
378{
379 filtAmps.iterNo = rawAmps.iterNo;
380
381 if( _openLoop )
382 {
383 filtAmps.measurement.setZero();
384 return 0;
385 }
386
387 int topN = rawAmps.measurement.cols();
388
389 realT gainF = 1.0;
390 realT HOgainF = 0.0;
391
392 // leaks(0.01);
393
394 if( iterNo < _closingDelay )
395 gainF = ( (realT)iterNo ) / _closingDelay;
396
397 // if( iterNo < 0.55*_closingDelay) leaks(0.1);
398
399 if( _lowOrders > 0 )
400 {
401 if( iterNo >= _closingDelay + _lowOrderDelay )
402 {
403 _lowOrders = 0;
404 }
405 else
406 {
407 topN = _lowOrders;
408
409 if( iterNo >= _closingDelay )
410 {
411 HOgainF = ( (realT)( iterNo - _closingDelay ) ) / ( _lowOrderDelay );
412 }
413 }
414 }
415
416 for( int i = 0; i < topN; ++i )
417 {
418 if( std::isnan( rawAmps.measurement( 0, i ) ) || !std::isfinite( rawAmps.measurement( 0, i ) ) )
419 rawAmps.measurement( 0, i ) = 0.0;
420
421 _commands( 0, i ) =
422 ( 1.0 - _leaks( 0, i ) ) * _commands( 0, i ) + gainF * _gains( 0, i ) * rawAmps.measurement( 0, i );
423
424 filtAmps.measurement( 0, i ) = _commands( 0, i );
425 }
426
427 for( int i = topN; i < rawAmps.measurement.cols(); ++i )
428 {
429 if( std::isnan( rawAmps.measurement( 0, i ) ) || !std::isfinite( rawAmps.measurement( 0, i ) ) )
430 rawAmps.measurement( 0, i ) = 0.0;
431
432 _commands( 0, i ) =
433 ( 1.0 - _leaks( 0, i ) ) * _commands( 0, i ) + HOgainF * _gains( 0, i ) * rawAmps.measurement( 0, i );
434
435 filtAmps.measurement( 0, i ) = _commands( 0, i );
436
437 // filtAmps.measurement(0,i) = 0;
438 }
439
440 return 0;
441}
442
443} // namespace sim
444} // namespace AO
445} // namespace mx
446
447#endif //__leakyIntegrator_hpp__
Implements the leaky integrator controller.
imageT _gains
Column-vector of gains.
Eigen::Array< realT, Eigen::Dynamic, Eigen::Dynamic > imageT
The image type, used here as a general storage array.
_realT realT
The real data type.
bool openLoop()
Get the value of the _openLoop flag.
wfMeasurement< realT > commandT
The command type.
int closingDelay()
Get the value of the _closingDelay.
wavefront< realT > wavefrontT
The wavefront data type.
int leaks(realT l)
Set a leak for all modes.
imageT _leaks
Column-vector of leaks.
int filterCommands(commandT &filtAmps, commandT &rawAmps, int iterNo)
Apply the leaky integrator.
int initMeasurements(commandT &filtAmps, commandT &rawAmps)
Allocate the provided command structures.
int initialize(int nModes)
Allocate and initialize all state.
realT gain(int i)
Get the gain for a single mode.
int gains(realT g)
Set the gain for all modes to a single value.
int leak(int i, realT l)
Set the leak for a single mode.
int _nModes
The number of modes being filtered.
bool _openLoop
If true, then commands are not integrated.
imageT _commands
Column-vector past commands.
int _closingDelay
If > 0, then the gains are ramped linearly up to this value. Default = 0.
int lowOrders()
Get the value of the _lowOrders.
int nModes()
Get the number of modes.
The mxlib c++ namespace.
Definition mxError.hpp:106
Structure containing the phase and amplitude of a wavefront.
Definition wavefront.hpp:24