From fbbbe20111a092551e60f59baadc7b68b89269b2 Mon Sep 17 00:00:00 2001 From: milanofthe Date: Fri, 15 Nov 2024 00:06:39 +0100 Subject: [PATCH] improved error contron for GEAR methods --- examples/example_nested_subsystems.py | 2 +- examples/example_pendulum.py | 2 +- examples/example_stickslip.py | 2 +- examples/example_vanderpol.py | 2 +- pathsim/solvers/gear.py | 40 +++++++++------------------ 5 files changed, 17 insertions(+), 31 deletions(-) diff --git a/examples/example_nested_subsystems.py b/examples/example_nested_subsystems.py index 0a23ee8..8773702 100644 --- a/examples/example_nested_subsystems.py +++ b/examples/example_nested_subsystems.py @@ -86,7 +86,7 @@ ] #initialize simulation with the blocks, connections, timestep and logging enabled -Sim = Simulation(blocks, connections, dt=dt, log=True, Solver=GEAR52A, tolerance_lte_abs=1e-5, tolerance_lte_rel=1e-3) +Sim = Simulation(blocks, connections, dt=dt, log=True, Solver=GEAR52A, tolerance_lte_abs=1e-5, tolerance_lte_rel=1e-4) #run simulation for some number of seconds Sim.run(3*mu) diff --git a/examples/example_pendulum.py b/examples/example_pendulum.py index 2b4eb46..e1490bf 100644 --- a/examples/example_pendulum.py +++ b/examples/example_pendulum.py @@ -19,7 +19,7 @@ Scope ) -from pathsim.solvers import RKCK54, GEAR52A +from pathsim.solvers import RKCK54 # MATHEMATICAL PENDULUM ================================================================= diff --git a/examples/example_stickslip.py b/examples/example_stickslip.py index 5c168d4..01d05f4 100644 --- a/examples/example_stickslip.py +++ b/examples/example_stickslip.py @@ -46,7 +46,7 @@ def _f(x, u, t): ] #initialize simulation with the blocks, connections, timestep and logging enabled -Sim = Simulation(blocks, connections, dt=0.01, log=True, Solver=GEAR52A, tolerance_lte_abs=1e-6, tolerance_lte_rel=1e-3) +Sim = Simulation(blocks, connections, dt=0.01, log=True, Solver=GEAR52A, tolerance_lte_abs=1e-5, tolerance_lte_rel=1e-4) #run the simulation for some time Sim.run(2*T) diff --git a/examples/example_vanderpol.py b/examples/example_vanderpol.py index ec012f7..70fae7a 100644 --- a/examples/example_vanderpol.py +++ b/examples/example_vanderpol.py @@ -42,7 +42,7 @@ def jac(x, u, t): ] #initialize simulation with the blocks, connections, timestep and logging enabled -Sim = Simulation(blocks, connections, dt=5, log=True, Solver=GEAR52A, tolerance_lte_abs=1e-5, tolerance_lte_rel=1e-3) +Sim = Simulation(blocks, connections, dt=1, log=True, Solver=GEAR52A, tolerance_lte_abs=1e-6, tolerance_lte_rel=1e-4) Sim.run(3*mu) diff --git a/pathsim/solvers/gear.py b/pathsim/solvers/gear.py index 90223bd..b737c1c 100644 --- a/pathsim/solvers/gear.py +++ b/pathsim/solvers/gear.py @@ -19,8 +19,7 @@ def compute_bdf_coefficients(order, timesteps): """ Computes the coefficients for backward differentiation formulas for a given order. - The timesteps can be specified if the coefficients are for variable timestep BDF - methods. + The timesteps can be specified for variable timestep BDF methods. For m-th order BDF we have for the n-th timestep: sum(alpha_i * x_i; i=n-m,...,n) = h_n * f_n(x_n, t_n) @@ -190,8 +189,8 @@ def error_controller(self, x_m): #compute timestep scale factor using accuracy order of truncation error timestep_rescale = self.beta / error_norm ** (1/self.n) - #clip the rescale factor to a reasonable range - timestep_rescale = np.clip(timestep_rescale, 0.1, 10.0) + # #clip the rescale factor to a reasonable range + # timestep_rescale = np.clip(timestep_rescale, 0.1, 10.0) return success, error_norm, timestep_rescale @@ -391,36 +390,23 @@ def error_controller(self, x_m, x_p): #compute the error norm and clip it error_norm_m = np.clip(float(np.max(scaled_error_m)), 1e-18, None) - error_norm_p = np.clip(float(np.max(scaled_error_p)), 1e-18, None) + error_norm_p = np.clip(float(np.max(scaled_error_p)), 1e-18, None) - #decrease the order if smaller order is more accurate - if error_norm_m < error_norm_p: - - #success metric - success = error_norm_m <= 1.0 + #success metric (use lower order estimate) + success = error_norm_m <= 1.0 - #compute timestep scale factor using accuracy order of truncation error - timestep_rescale = self.beta / error_norm_m ** (1/self.n) - timestep_rescale = np.clip(timestep_rescale, 0.1, 10.0) + #compute timestep scale factor using accuracy order of truncation error + timestep_rescale = self.beta / error_norm_m ** (1/self.n) - #decrease method order by one + #decrease the order if smaller order is more accurate (stability) + if error_norm_m < error_norm_p: self.n = max(self.n-1, self.n_min) - - return success, error_norm_m, timestep_rescale - + + #increase the order if larger order is more accurate (accuracy -> larger steps) else: - - #success metric - success = error_norm_p <= 1.0 - - #compute timestep scale factor using accuracy order of truncation error - timestep_rescale = self.beta / error_norm_p ** (1/(self.n + 1)) - timestep_rescale = np.clip(timestep_rescale, 0.1, 10.0) - - #increase method order by one self.n = min(self.n+1, self.n_max) - return success, error_norm_p, timestep_rescale + return success, error_norm_p, timestep_rescale # methods for timestepping ---------------------------------------------------------