-
Notifications
You must be signed in to change notification settings - Fork 1
/
eff_gen.ml
55 lines (47 loc) · 1.14 KB
/
eff_gen.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
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
let n = try int_of_string (Sys.argv.(1)) with _ -> 25
module MkGen (S :sig
type 'a t
val iter : ('a -> unit) -> 'a t -> unit
end) : sig
val gen : 'a S.t -> (unit -> 'a option)
end = struct
let gen : type a. a S.t -> (unit -> a option) = fun l ->
let module M = struct effect Yield : a -> unit end in
let open M in
let rec step = ref (fun () ->
match S.iter (fun v -> perform (Yield v)) l with
| () -> None
| effect (Yield v) k ->
step := (fun () -> continue k ());
Some v)
in
fun () -> !step ()
end
(* A generator for a list *)
module L = MkGen(struct
type 'a t = 'a list
let iter = List.iter
end)
type 'a tree =
| Leaf
| Node of 'a tree * 'a * 'a tree
let rec make = function
| 0 -> Leaf
| n -> let t = make (n-1) in Node (t,n,t)
let rec iter f = function
| Leaf -> ()
| Node (l, x, r) -> iter f l; f x; iter f r
(* A generator for a tree *)
module T = MkGen(struct
type 'a t = 'a tree
let iter = iter
end)
let main () =
let next = T.gen (make n) in
let rec consume () =
match next () with
| None -> ()
| Some _ -> consume ()
in
consume ()
let _ = main ()