Source file single_waiter.ml

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
(* Allows a single fiber to wait to be notified by another fiber in the same domain.
   If multiple fibers need to wait at once, or the notification comes from another domain,
   this can't be used. *)

type 'a t = {
  mutable wake : ('a, exn) result -> unit;
}

let create () = { wake = ignore }

let wake t v = t.wake v

let await t id =
  let x =
    Suspend.enter @@ fun ctx enqueue ->
    Cancel.Fiber_context.set_cancel_fn ctx (fun ex ->
        t.wake <- ignore;
        enqueue (Error ex)
      );
    t.wake <- (fun x ->
        Cancel.Fiber_context.clear_cancel_fn ctx;
        t.wake <- ignore;
        enqueue x
      )
  in
  Trace.read id;
  x