@@ -26,6 +26,7 @@ use crate::propagators::TrajectoryEventSnafu;
26
26
use crate :: time:: { Duration , Epoch , Unit } ;
27
27
use crate :: State ;
28
28
use anise:: almanac:: Almanac ;
29
+ use anise:: errors:: MathError ;
29
30
use rayon:: iter:: ParallelBridge ;
30
31
use rayon:: prelude:: ParallelIterator ;
31
32
use snafu:: ResultExt ;
@@ -222,7 +223,10 @@ where
222
223
// Report status every minute
223
224
let cur_epoch = self . state . epoch ( ) ;
224
225
let dur_to_go = ( stop_time - cur_epoch) . floor ( Unit :: Second * 1 ) ;
225
- info ! ( "\t ... current epoch {}, remaing {}" , cur_epoch, dur_to_go) ;
226
+ info ! (
227
+ "\t ... current epoch {}, remaining {} (step size = {})" ,
228
+ cur_epoch, dur_to_go, self . details. step
229
+ ) ;
226
230
prev_tick = Instant :: now ( ) ;
227
231
}
228
232
}
@@ -385,7 +389,7 @@ where
385
389
// Reset the number of attempts used (we don't reset the error because it's set before it's read)
386
390
self . details . attempts = 1 ;
387
391
// Convert the step size to seconds -- it's mutable because we may change it below
388
- let mut step_size = self . step_size . to_seconds ( ) ;
392
+ let mut step_size_s = self . step_size . to_seconds ( ) ;
389
393
loop {
390
394
let ki = self
391
395
. prop
@@ -411,8 +415,8 @@ where
411
415
. prop
412
416
. dynamics
413
417
. eom (
414
- ci * step_size ,
415
- & ( state_vec + step_size * wi) ,
418
+ ci * step_size_s ,
419
+ & ( state_vec + step_size_s * wi) ,
416
420
state_ctx,
417
421
self . almanac . clone ( ) ,
418
422
)
@@ -429,9 +433,9 @@ where
429
433
let b_i = self . prop . method . b_coeffs ( ) [ i] ;
430
434
if !self . fixed_step {
431
435
let b_i_star = self . prop . method . b_coeffs ( ) [ i + self . prop . method . stages ( ) ] ;
432
- error_est += step_size * ( b_i - b_i_star) * ki;
436
+ error_est += step_size_s * ( b_i - b_i_star) * ki;
433
437
}
434
- next_state += step_size * b_i * ki;
438
+ next_state += step_size_s * b_i * ki;
435
439
}
436
440
437
441
if self . fixed_step {
@@ -445,46 +449,66 @@ where
445
449
. opts
446
450
. error_ctrl
447
451
. estimate ( & error_est, & next_state, state_vec) ;
452
+
448
453
if self . details . error <= self . prop . opts . tolerance
449
- || step_size <= self . prop . opts . min_step . to_seconds ( )
454
+ || step_size_s <= self . prop . opts . min_step . to_seconds ( )
450
455
|| self . details . attempts >= self . prop . opts . attempts
451
456
{
457
+ if next_state. iter ( ) . any ( |x| x. is_nan ( ) ) {
458
+ return Err ( PropagationError :: PropMathError {
459
+ source : MathError :: DomainError {
460
+ value : f64:: NAN ,
461
+ msg : "try another integration method, or decrease step size; part of state vector is" ,
462
+ } ,
463
+ } ) ;
464
+ }
452
465
if self . details . attempts >= self . prop . opts . attempts {
453
466
warn ! (
454
467
"Could not further decrease step size: maximum number of attempts reached ({})" ,
455
468
self . details. attempts
456
469
) ;
457
470
}
458
471
459
- self . details . step = step_size * Unit :: Second ;
472
+ self . details . step = step_size_s * Unit :: Second ;
460
473
if self . details . error < self . prop . opts . tolerance {
461
474
// Let's increase the step size for the next iteration.
462
475
// Error is less than tolerance, let's attempt to increase the step for the next iteration.
463
476
let proposed_step = 0.9
464
- * step_size
477
+ * step_size_s
465
478
* ( self . prop . opts . tolerance / self . details . error )
466
479
. powf ( 1.0 / f64:: from ( self . prop . method . order ( ) ) ) ;
467
- step_size = if proposed_step > self . prop . opts . max_step . to_seconds ( ) {
480
+
481
+ step_size_s = if proposed_step > self . prop . opts . max_step . to_seconds ( ) {
468
482
self . prop . opts . max_step . to_seconds ( )
469
483
} else {
470
484
proposed_step
471
485
} ;
472
486
}
473
487
// In all cases, let's update the step size to whatever was the adapted step size
474
- self . step_size = step_size * Unit :: Second ;
488
+ self . step_size = step_size_s * Unit :: Second ;
489
+ if self . step_size . abs ( ) < self . prop . opts . min_step {
490
+ // Custom signum in case the step size becomes zero.
491
+ let signum = if self . step_size . is_negative ( ) {
492
+ -1.0
493
+ } else {
494
+ 1.0
495
+ } ;
496
+ self . step_size = self . prop . opts . min_step * signum;
497
+ }
475
498
return Ok ( ( self . details . step , next_state) ) ;
476
499
} else {
477
500
// Error is too high and we aren't using the smallest step, and we haven't hit the max number of attempts.
478
501
// So let's adapt the step size.
479
502
self . details . attempts += 1 ;
480
- let proposed_step = 0.9
481
- * step_size
503
+ let proposed_step_s = 0.9
504
+ * step_size_s
482
505
* ( self . prop . opts . tolerance / self . details . error )
483
506
. powf ( 1.0 / f64:: from ( self . prop . method . order ( ) - 1 ) ) ;
484
- step_size = if proposed_step < self . prop . opts . min_step . to_seconds ( ) {
507
+
508
+ step_size_s = if proposed_step_s < self . prop . opts . min_step . to_seconds ( ) {
485
509
self . prop . opts . min_step . to_seconds ( )
486
510
} else {
487
- proposed_step
511
+ proposed_step_s
488
512
} ;
489
513
// Note that we don't set self.step_size, that will be updated right before we return
490
514
}
0 commit comments