vppinfra: preserve total_cpu_time across clock freq adjustment

If clib_time_verify_frequency() adjusts the clock frequency, transform
total_cpu_time to the new time coordinate space. Otherwise, we break
comparisons with previous clib_time_now() value.

Without this correction, time jumps in one direction or the other
depending on the sign of the frequency change. Reasonably harmless in
most cases, but under perfect storm conditions the wheels fall off.

Type: fix

Signed-off-by: Dave Barach <dave@barachs.net>
Change-Id: I21802c2630e2c87ff817cd732b7d78bc022cd2d7
This commit is contained in:
Dave Barach
2019-12-01 08:59:03 -05:00
committed by Florin Coras
parent 5ef9ca6c0d
commit e52d8d880a
2 changed files with 17 additions and 4 deletions

View File

@@ -198,6 +198,7 @@ clib_time_verify_frequency (clib_time_t * c)
f64 dtr_max; f64 dtr_max;
u64 dtc = c->last_cpu_time - c->last_verify_cpu_time; u64 dtc = c->last_cpu_time - c->last_verify_cpu_time;
f64 new_clocks_per_second, delta; f64 new_clocks_per_second, delta;
f64 save_total_cpu_time_in_seconds;
c->last_verify_cpu_time = c->last_cpu_time; c->last_verify_cpu_time = c->last_cpu_time;
c->last_verify_reference_time = now_reference; c->last_verify_reference_time = now_reference;
@@ -239,10 +240,12 @@ clib_time_verify_frequency (clib_time_t * c)
flt_round_nearest ((f64) dtc / (dtr * c->round_to_units)) flt_round_nearest ((f64) dtc / (dtr * c->round_to_units))
* c->round_to_units; * c->round_to_units;
/* Compute abs(rate change) */
delta = new_clocks_per_second - c->clocks_per_second; delta = new_clocks_per_second - c->clocks_per_second;
if (delta < 0.0) if (delta < 0.0)
delta = -delta; delta = -delta;
/* If rate change > 1%, reject it and try again */
if (PREDICT_FALSE ((delta / c->clocks_per_second) > .01)) if (PREDICT_FALSE ((delta / c->clocks_per_second) > .01))
{ {
clib_warning ("Rejecting large frequency change of %.2f%%", clib_warning ("Rejecting large frequency change of %.2f%%",
@@ -251,11 +254,19 @@ clib_time_verify_frequency (clib_time_t * c)
return; return;
} }
c->clocks_per_second = /* Save total cpu time in seconds */
flt_round_nearest ((f64) dtc / (dtr * c->round_to_units)) save_total_cpu_time_in_seconds = c->total_cpu_time * c->seconds_per_clock;
* c->round_to_units;
/* Recalculate clock rate */
c->clocks_per_second = new_clocks_per_second;
c->seconds_per_clock = 1 / c->clocks_per_second; c->seconds_per_clock = 1 / c->clocks_per_second;
/*
* Restore total cpu time in seconds. Otherwise, if c->clocks_per_second
* has decreased, time may appear to flow backwards.
*/
c->total_cpu_time = save_total_cpu_time_in_seconds * c->clocks_per_second;
/* Double time between verifies; max at 64 secs ~ 1 minute. */ /* Double time between verifies; max at 64 secs ~ 1 minute. */
if (c->log2_clocks_per_frequency_verify < c->log2_clocks_per_second + 6) if (c->log2_clocks_per_frequency_verify < c->log2_clocks_per_second + 6)
c->log2_clocks_per_frequency_verify += 1; c->log2_clocks_per_frequency_verify += 1;

View File

@@ -205,14 +205,16 @@ clib_time_now_internal (clib_time_t * c, u64 n)
{ {
u64 l = c->last_cpu_time; u64 l = c->last_cpu_time;
u64 t = c->total_cpu_time; u64 t = c->total_cpu_time;
f64 rv;
t += n - l; t += n - l;
c->total_cpu_time = t; c->total_cpu_time = t;
c->last_cpu_time = n; c->last_cpu_time = n;
rv = t * c->seconds_per_clock;
if (PREDICT_FALSE if (PREDICT_FALSE
((c->last_cpu_time - ((c->last_cpu_time -
c->last_verify_cpu_time) >> c->log2_clocks_per_frequency_verify)) c->last_verify_cpu_time) >> c->log2_clocks_per_frequency_verify))
clib_time_verify_frequency (c); clib_time_verify_frequency (c);
return t * c->seconds_per_clock; return rv;
} }
/* Maximum f64 value as max clib_time */ /* Maximum f64 value as max clib_time */