Skip to content

Commit 37b179d

Browse files
Merge pull request #378 from nyx-space/fix/gh-377
Bound the proposed integrator steps
2 parents 6906f70 + 96b29b3 commit 37b179d

File tree

3 files changed

+43
-16
lines changed

3 files changed

+43
-16
lines changed

src/propagators/instance.rs

+39-15
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@ use crate::propagators::TrajectoryEventSnafu;
2626
use crate::time::{Duration, Epoch, Unit};
2727
use crate::State;
2828
use anise::almanac::Almanac;
29+
use anise::errors::MathError;
2930
use rayon::iter::ParallelBridge;
3031
use rayon::prelude::ParallelIterator;
3132
use snafu::ResultExt;
@@ -222,7 +223,10 @@ where
222223
// Report status every minute
223224
let cur_epoch = self.state.epoch();
224225
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+
);
226230
prev_tick = Instant::now();
227231
}
228232
}
@@ -385,7 +389,7 @@ where
385389
// Reset the number of attempts used (we don't reset the error because it's set before it's read)
386390
self.details.attempts = 1;
387391
// 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();
389393
loop {
390394
let ki = self
391395
.prop
@@ -411,8 +415,8 @@ where
411415
.prop
412416
.dynamics
413417
.eom(
414-
ci * step_size,
415-
&(state_vec + step_size * wi),
418+
ci * step_size_s,
419+
&(state_vec + step_size_s * wi),
416420
state_ctx,
417421
self.almanac.clone(),
418422
)
@@ -429,9 +433,9 @@ where
429433
let b_i = self.prop.method.b_coeffs()[i];
430434
if !self.fixed_step {
431435
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;
433437
}
434-
next_state += step_size * b_i * ki;
438+
next_state += step_size_s * b_i * ki;
435439
}
436440

437441
if self.fixed_step {
@@ -445,46 +449,66 @@ where
445449
.opts
446450
.error_ctrl
447451
.estimate(&error_est, &next_state, state_vec);
452+
448453
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()
450455
|| self.details.attempts >= self.prop.opts.attempts
451456
{
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+
}
452465
if self.details.attempts >= self.prop.opts.attempts {
453466
warn!(
454467
"Could not further decrease step size: maximum number of attempts reached ({})",
455468
self.details.attempts
456469
);
457470
}
458471

459-
self.details.step = step_size * Unit::Second;
472+
self.details.step = step_size_s * Unit::Second;
460473
if self.details.error < self.prop.opts.tolerance {
461474
// Let's increase the step size for the next iteration.
462475
// Error is less than tolerance, let's attempt to increase the step for the next iteration.
463476
let proposed_step = 0.9
464-
* step_size
477+
* step_size_s
465478
* (self.prop.opts.tolerance / self.details.error)
466479
.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() {
468482
self.prop.opts.max_step.to_seconds()
469483
} else {
470484
proposed_step
471485
};
472486
}
473487
// 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+
}
475498
return Ok((self.details.step, next_state));
476499
} else {
477500
// Error is too high and we aren't using the smallest step, and we haven't hit the max number of attempts.
478501
// So let's adapt the step size.
479502
self.details.attempts += 1;
480-
let proposed_step = 0.9
481-
* step_size
503+
let proposed_step_s = 0.9
504+
* step_size_s
482505
* (self.prop.opts.tolerance / self.details.error)
483506
.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() {
485509
self.prop.opts.min_step.to_seconds()
486510
} else {
487-
proposed_step
511+
proposed_step_s
488512
};
489513
// Note that we don't set self.step_size, that will be updated right before we return
490514
}

src/propagators/mod.rs

+3
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616
along with this program. If not, see <https://www.gnu.org/licenses/>.
1717
*/
1818

19+
use anise::errors::MathError;
1920
use snafu::prelude::*;
2021
use std::fmt;
2122

@@ -66,4 +67,6 @@ pub enum PropagationError {
6667
NthEventError { nth: usize, found: usize },
6768
#[snafu(display("propagation failed because {source}"))]
6869
PropConfigError { source: ConfigError },
70+
#[snafu(display("propagation encountered a math error {source}"))]
71+
PropMathError { source: MathError },
6972
}

tests/orbit_determination/spacecraft.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -329,7 +329,7 @@ fn od_val_sc_srp_estimation(
329329
let setup = Propagator::dp78(
330330
sc_dynamics,
331331
IntegratorOptions::builder()
332-
.max_step(1.minutes())
332+
.max_step(30.seconds())
333333
.error_ctrl(ErrorControl::RSSCartesianStep)
334334
.build(),
335335
);

0 commit comments

Comments
 (0)