-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathemit-program.sml
executable file
·57 lines (51 loc) · 1.94 KB
/
emit-program.sml
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
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
structure EmitProgram :> EMIT_PROGRAM =
struct
structure GS = GameState
datatype status =
Progress of real
| Paused of int
| Done
fun emit turns =
let
val icr = 1.0 / real (length turns + 1)
val progress = ref 0.0
val status = ref (Progress 0.0)
fun preview _ = ()
val turnsleft = ref turns
fun taketurn dos =
(case !turnsleft of
(* Weird; should only happen for the empty program. *)
nil => (status := Done;
DOS.kill (DOS.getpid dos);
DOS.Can'tRun)
| (t :: turns) =>
let val gs = DOS.gamestate dos
val slot = case t of
LTG.LeftApply (_, i) => i
| LTG.RightApply (i, _) => i
in
if LTG.slotisdead (GS.myside gs) slot
then (status := Paused slot;
(* Someone might still revive this slot,
but we'll be blocked until then. *)
DOS.Can'tRun)
else (progress := !progress + icr;
(* Immediately be finished if that was the
last instruction. *)
if List.null turns
then (status := Done; DOS.kill (DOS.getpid dos))
else status := Progress (!progress);
turnsleft := turns;
DOS.Turn t)
end)
in
(status, { preview = preview, taketurn = taketurn })
end
fun emitspawn dos turns =
let val (status, dom) = emit turns
val pid = DOS.spawn (SOME (DOS.getpid dos))
("EP", DOS.getpriority dos, dom)
in
(status, pid)
end
end