-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathutil_late_init.ml
102 lines (85 loc) · 2.04 KB
/
util_late_init.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
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
(*
One-time initialization.
Useful to create mutual dependencies between modules.
Usage:
(* Module A *)
let init_foo, foo = create "foo"
let f () =
...
foo x y (* <-- must take place after initialization *)
...
(* Module B
Function foo is initialized in this module.
*)
let foo x y =
...
let () = A.init_foo foo (* <-- initialization *)
*)
let uninitialized = Hashtbl.create 10
let create_id =
let n = ref 0 in
fun () ->
let id = !n in
assert (id >= 0);
incr n;
id
let create name =
let r = ref None in
let id = create_id () in
Hashtbl.add uninitialized id name;
let init f =
match !r with
| None ->
Hashtbl.remove uninitialized id;
r := Some f
| Some _ ->
failwith (name ^ " was already initialized")
in
let call x =
match !r with
| None ->
failwith ("Uninitialized function " ^ name)
| Some f ->
f x
in
init, call
(*
Return the list of uninitialized functions, sorted by creation date.
*)
let get_all_uninitialized () =
let l =
Hashtbl.fold (fun id name acc -> (id, name) :: acc) uninitialized []
in
let sorted = List.sort (fun (a, _) (b, _) -> compare a b) l in
List.map snd sorted
(*
Print an error and exit the process if anything is uninitialized.
*)
let check_all () =
match get_all_uninitialized () with
| [] -> ()
| l ->
Printf.eprintf
"The following functions were not initialized: %s\n%!"
(String.concat ", " l);
exit 1
let test_late_initialization () =
let n0 = List.length (get_all_uninitialized ()) in
let init_foo, foo = create "foo" in
assert (List.length (get_all_uninitialized ()) = n0 + 1);
(try
ignore (foo 1 2);
assert false
with _ -> ());
let real_foo x y = x + y in
init_foo real_foo;
assert (List.length (get_all_uninitialized ()) = n0);
(try
init_foo real_foo;
assert false
with _ -> ());
assert (foo 3 4 = 7);
true
let tests = [
"late initialization", test_late_initialization;
]