From e5eadebc2b74093774df2fd0869d9be8ba0f3aea Mon Sep 17 00:00:00 2001 From: Dorival Pedroso Date: Sat, 28 Sep 2024 17:42:57 +1000 Subject: [PATCH] [Important] Use a tiny constant in ODE Output to (try to) improve the uniform spacing calculation --- russell_ode/src/output.rs | 13 ++++++++++++- 1 file changed, 12 insertions(+), 1 deletion(-) diff --git a/russell_ode/src/output.rs b/russell_ode/src/output.rs index 116f0c35..b67458a1 100644 --- a/russell_ode/src/output.rs +++ b/russell_ode/src/output.rs @@ -8,6 +8,11 @@ use std::io::BufReader; use std::path::Path; use std::sync::Arc; +/// Holds a tiny number (epsilon) to improve the rounding of x1 - x0 in with_dense_output +/// +/// This tolerance helps in making x1 = 0.49999999999999839 behave as x1 = 0.5, for example. +const EPS_X1_H_OUT: f64 = 1e-13; + /// Holds the data generated at an accepted step or during the dense output #[derive(Clone, Debug, Deserialize)] pub struct OutData { @@ -389,7 +394,7 @@ impl<'a, A> Output<'a, A> { if self.with_dense_output() { if let Some(h_out) = self.dense_h_out { // uniform spacing - let n = usize::max(2, ((x1 - x0) / h_out) as usize + 1); // at least 2 (first and last) are required + let n = usize::max(2, ((x1 + EPS_X1_H_OUT - x0) / h_out) as usize + 1); // at least 2 (first and last) are required if self.dense_x.len() != n { self.dense_x.resize(n, 0.0); } @@ -793,6 +798,12 @@ mod tests { 1e-15, ); + // with x1 nearly equal to 0.5 + let x1 = 0.4999999999999; + out.set_dense_h_out(0.1).unwrap(); + out.initialize(0.0, x1, false).unwrap(); + array_approx_eq(&out.dense_x, &[0.0, 0.1, 0.2, 0.3, 0.4, x1], 1e-15); + // with empty x_out out.set_dense_x_out(&[]).unwrap(); out.initialize(3.0, 4.0, false).unwrap();