-
Notifications
You must be signed in to change notification settings - Fork 6
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add an actor model implementation example
- Loading branch information
Showing
3 changed files
with
200 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
(library | ||
(name hoot) | ||
(libraries backoff multicore-magic picos)) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,155 @@ | ||
open Picos | ||
|
||
let[@inline never] impossible () = failwith "impossible" | ||
let[@inline never] not_in_a_hoot () = raise (Sys_error "Not in a hoot") | ||
|
||
module Message = struct | ||
type t = .. | ||
end | ||
|
||
type _ tdt = | ||
| Nil : [> `Nil ] tdt | ||
| Message : { | ||
message : Message.t; | ||
next : [ `Nil | `Message ] tdt; | ||
} | ||
-> [> `Message ] tdt | ||
| Wait : Trigger.t -> [> `Wait ] tdt | ||
| Pid : { | ||
computation : unit Computation.t; | ||
terminated : unit Computation.t; | ||
incoming : incoming Atomic.t; | ||
mutable received : [ `Nil | `Message ] tdt; | ||
} | ||
-> [> `Pid ] tdt | ||
|
||
and incoming = In : [< `Nil | `Message | `Wait ] tdt -> incoming [@@unboxed] | ||
|
||
module Pid = struct | ||
type t = [ `Pid ] tdt | ||
|
||
let key : [ `Nil | `Pid ] tdt Fiber.FLS.t = Fiber.FLS.create () | ||
end | ||
|
||
let self_of fiber = | ||
match Fiber.FLS.get_exn fiber Pid.key with | ||
| Pid _ as t -> t | ||
| Nil | (exception Fiber.FLS.Not_set) -> not_in_a_hoot () | ||
|
||
let self () : Pid.t = self_of @@ Fiber.current () | ||
|
||
let run main = | ||
let fiber = Fiber.current () in | ||
let before = Fiber.FLS.get fiber Pid.key ~default:Nil in | ||
let computation = Computation.create ~mode:`LIFO () in | ||
let inner = Computation.Packed computation in | ||
let (Pid p as t) : Pid.t = | ||
let terminated = Computation.create ~mode:`LIFO () | ||
and incoming = Atomic.make (In Nil) |> Multicore_magic.copy_as_padded in | ||
Pid { computation; terminated; incoming; received = Nil } | ||
in | ||
Fiber.FLS.set fiber Pid.key t; | ||
let (Packed parent as outer) = Fiber.get_computation fiber in | ||
let canceler = Computation.attach_canceler ~from:parent ~into:computation in | ||
Fiber.set_computation fiber inner; | ||
Computation.capture p.terminated main (); | ||
Computation.finish p.computation; | ||
Fiber.set_computation fiber outer; | ||
Computation.detach parent canceler; | ||
Fiber.FLS.set fiber Pid.key before | ||
|
||
let wait (Pid p : Pid.t) = Computation.await p.terminated | ||
|
||
let spawn main = | ||
let (Pid p as t) : Pid.t = | ||
let computation = Computation.create ~mode:`LIFO () | ||
and terminated = Computation.create ~mode:`LIFO () | ||
and incoming = Atomic.make (In Nil) in | ||
Pid { computation; terminated; incoming; received = Nil } | ||
in | ||
let fiber = Fiber.create ~forbid:false p.computation in | ||
Fiber.FLS.set fiber Pid.key t; | ||
begin | ||
Fiber.spawn fiber @@ fun fiber -> | ||
let (Pid p) : Pid.t = self_of fiber in | ||
Computation.capture p.terminated main (); | ||
Computation.finish p.computation | ||
end; | ||
t | ||
|
||
let rec rev_to (Message _ as ms : [ `Message ] tdt) : | ||
[ `Nil | `Message ] tdt -> _ = function | ||
| Nil -> ms | ||
| Message r -> rev_to (Message { message = r.message; next = ms }) r.next | ||
|
||
let rev (Message r : [ `Message ] tdt) = | ||
rev_to (Message { message = r.message; next = Nil }) r.next | ||
|
||
let rec receive (Pid p as t : Pid.t) = | ||
match Atomic.get p.incoming with | ||
| In Nil as before -> | ||
let trigger = Trigger.create () in | ||
let after = In (Wait trigger) in | ||
if Atomic.compare_and_set p.incoming before after then begin | ||
match Trigger.await trigger with | ||
| None -> () | ||
| Some (exn, bt) -> Printexc.raise_with_backtrace exn bt | ||
end; | ||
receive t | ||
| _ -> begin | ||
match Atomic.exchange p.incoming (In Nil) with | ||
| In (Wait _ | Nil) -> impossible () | ||
| In (Message _ as ms) -> | ||
let (Message r : [ `Message ] tdt) = rev ms in | ||
p.received <- r.next; | ||
r.message | ||
end | ||
|
||
let receive () = | ||
let (Pid p as t) = self () in | ||
match p.received with | ||
| Message r -> | ||
p.received <- r.next; | ||
r.message | ||
| Nil -> receive t | ||
|
||
let rec send (Pid p as t : Pid.t) message backoff = | ||
match Atomic.get p.incoming with | ||
| In ((Nil | Message _) as before) -> | ||
let after = Message { message; next = before } in | ||
if not (Atomic.compare_and_set p.incoming (In before) (In after)) then | ||
send t message (Backoff.once backoff) | ||
| In (Wait trigger as before) -> | ||
let after = Message { message; next = Nil } in | ||
if Atomic.compare_and_set p.incoming (In before) (In after) then | ||
Trigger.signal trigger | ||
else send t message (Backoff.once backoff) | ||
|
||
let send t message = send t message Backoff.default | ||
|
||
type Message.t += Terminated of Pid.t | ||
|
||
let monitor ~at ~the:(Pid the_p as the : Pid.t) = | ||
let[@alert "-handler"] trigger = | ||
Trigger.from_action at the @@ fun _ at the -> send at (Terminated the) | ||
in | ||
if not (Computation.try_attach the_p.terminated trigger) then | ||
send at (Terminated the) | ||
|
||
exception Terminate | ||
|
||
let empty_bt = Printexc.get_callstack 0 | ||
|
||
let link (Pid p1 as t1 : Pid.t) (Pid p2 as t2 : Pid.t) = | ||
let[@alert "-handler"] trigger = | ||
Trigger.from_action t1 t2 @@ fun _ (Pid p1 : Pid.t) (Pid p2 : Pid.t) -> | ||
Computation.cancel p1.computation Terminate empty_bt; | ||
Computation.cancel p2.computation Terminate empty_bt | ||
in | ||
if | ||
(not (Computation.try_attach p1.terminated trigger)) | ||
|| not (Computation.try_attach p2.terminated trigger) | ||
then begin | ||
Computation.cancel p1.computation Terminate empty_bt; | ||
Computation.cancel p2.computation Terminate empty_bt | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,42 @@ | ||
val run : (unit -> unit) -> unit | ||
(** *) | ||
|
||
module Pid : sig | ||
(** *) | ||
|
||
type t | ||
(** *) | ||
end | ||
|
||
val spawn : (unit -> unit) -> Pid.t | ||
(** *) | ||
|
||
val self : unit -> Pid.t | ||
(** *) | ||
|
||
val wait : Pid.t -> unit | ||
(** *) | ||
|
||
module Message : sig | ||
(** *) | ||
|
||
type t = .. | ||
(** *) | ||
end | ||
|
||
val receive : unit -> Message.t | ||
(** *) | ||
|
||
val send : Pid.t -> Message.t -> unit | ||
(** *) | ||
|
||
type Message.t += Terminated of Pid.t (** *) | ||
|
||
val monitor : at:Pid.t -> the:Pid.t -> unit | ||
(** *) | ||
|
||
exception Terminate | ||
(** *) | ||
|
||
val link : Pid.t -> Pid.t -> unit | ||
(** *) |