Skip to content

Commit

Permalink
CP-49158: Use exponential backoff for delay between recursive calls
Browse files Browse the repository at this point in the history
This delay was right after we waited for a new event, delaying all event
responses by 50ms (including task completions).
Eliminate the first delay, so that if we find the event we're looking after the
DB update, then we can return immediately.

On spurious wakeups (e.g. not the event we subscribed for) the delay is still
useful, so keep it for recursive calls after the first one, and exponentially
increase it up to the configured maximum.

No feature flag, this is a relatively small change, and we use exponential backoffs
elsewhere in XAPI already.

Signed-off-by: Edwin Török <[email protected]>
  • Loading branch information
edwintorok committed Dec 10, 2024
1 parent ab75aea commit b3bd1d8
Show file tree
Hide file tree
Showing 3 changed files with 37 additions and 16 deletions.
29 changes: 22 additions & 7 deletions ocaml/xapi-aux/throttle.ml
Original file line number Diff line number Diff line change
Expand Up @@ -41,9 +41,20 @@ module Make (Size : SIZE) = struct
end

module Batching = struct
type t = {delay_before: Mtime.span; delay_between: Mtime.span}
type t = {
delay_initial: Mtime.span
; delay_before: Mtime.span
; delay_between: Mtime.span
}

let make ~delay_before ~delay_between = {delay_before; delay_between}
let make ~delay_before ~delay_between =
(* we are dividing, cannot overflow *)
let delay_initial =
Mtime.Span.to_float_ns delay_between /. 16.
|> Mtime.Span.of_float_ns
|> Option.get
in
{delay_initial; delay_before; delay_between}

(** [perform_delay delay] calls {!val:Thread.delay} when [delay] is non-zero.
Expand All @@ -55,11 +66,15 @@ module Batching = struct
if Mtime.Span.is_longer delay ~than:Mtime.Span.min_span then
Thread.delay (Clock.Timer.span_to_s delay)

let with_recursive_loop config f arg =
let rec self arg =
perform_delay config.delay_between ;
(f [@tailcall]) self arg
let span_min a b = if Mtime.Span.is_shorter a ~than:b then a else b

let with_recursive_loop config f =
let rec self arg input =
let arg = span_min config.delay_between Mtime.Span.(2 * arg) in
perform_delay arg ;
(f [@tailcall]) (self arg) input
in
let self0 arg input = (f [@tailcall]) (self arg) input in
perform_delay config.delay_before ;
f self arg
f (self0 config.delay_initial)
end
16 changes: 11 additions & 5 deletions ocaml/xapi-aux/throttle.mli
Original file line number Diff line number Diff line change
Expand Up @@ -34,21 +34,27 @@ module Batching : sig
*)

val with_recursive_loop : t -> (('a -> 'b) -> 'a -> 'b) -> 'a -> 'b
(** [with_recursive_loop config f arg] calls [f self arg], where [self] can be used
(** [with_recursive config f arg] calls [f self arg], where [self] can be used
for recursive calls.
A [delay_before] amount of seconds is inserted once, and [delay_between] is inserted between recursive calls:
[arg] is an argument that the implementation of [f] can change between recursive calls for its own purposes,
otherwise [()] can be used.
A [delay_before] amount of seconds is inserted once, and [delay_between/8] is inserted between recursive calls,
except the first one, and delays increase exponentially until [delay_between] is reached
{v
delay_before
f ...
(self[@tailcall]) ...
delay_between
f ...
(self[@tailcall]) ...
delay_between
delay_between/8
f ...
(self[@tailcall]) ...
delay_between/4
f ...
v}
The delays are determined by [config]
The delays are determined by [config], and [delay_between] uses an exponential backoff, up to [config.delay_between] delay.
*)
end
8 changes: 4 additions & 4 deletions ocaml/xapi/xapi_event.ml
Original file line number Diff line number Diff line change
Expand Up @@ -497,11 +497,11 @@ let rec next ~__context =
in
(* Like grab_range () only guarantees to return a non-empty range by blocking if necessary *)
let grab_nonempty_range =
Throttle.Batching.with_recursive_loop batching @@ fun self () ->
Throttle.Batching.with_recursive_loop batching @@ fun self arg ->
let last_id, end_id = grab_range () in
if last_id = end_id then
let (_ : int64) = wait subscription end_id in
(self [@tailcall]) ()
(self [@tailcall]) arg
else
(last_id, end_id)
in
Expand Down Expand Up @@ -608,7 +608,7 @@ let from_inner __context session subs from from_t timer batching =
let msg_gen, messages, tableset, (creates, mods, deletes, last) =
with_call session subs (fun sub ->
let grab_nonempty_range =
Throttle.Batching.with_recursive_loop batching @@ fun self () ->
Throttle.Batching.with_recursive_loop batching @@ fun self arg ->
let ( (msg_gen, messages, _tableset, (creates, mods, deletes, last))
as result
) =
Expand All @@ -627,7 +627,7 @@ let from_inner __context session subs from from_t timer batching =
(* last id the client got is equivalent to the current one *)
last_msg_gen := msg_gen ;
wait2 sub last timer ;
(self [@tailcall]) ()
(self [@tailcall]) arg
) else
result
in
Expand Down

0 comments on commit b3bd1d8

Please sign in to comment.