Source file top_closure.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
module Make (Keys : Top_closure_intf.Keys) (Monad : Monad_intf.S1) = struct
open Monad
let top_closure ~key ~deps elements =
let visited = ref Keys.empty in
let res = ref [] in
let rec loop elt ~temporarily_marked =
let key = key elt in
if Keys.mem temporarily_marked key then
return (Error [ elt ])
else if not (Keys.mem !visited key) then (
visited := Keys.add !visited key;
let temporarily_marked = Keys.add temporarily_marked key in
deps elt >>= iter_elts ~temporarily_marked >>= function
| Ok () ->
res := elt :: !res;
return (Ok ())
| Error l -> return (Error (elt :: l))
) else
return (Ok ())
and iter_elts elts ~temporarily_marked =
return elts >>= function
| [] -> return (Ok ())
| elt :: elts -> (
loop elt ~temporarily_marked >>= function
| Error _ as result -> return result
| Ok () -> iter_elts elts ~temporarily_marked)
in
iter_elts elements ~temporarily_marked:Keys.empty >>= function
| Ok () -> return (Ok (List.rev !res))
| Error elts -> return (Error elts)
end
[@@inlined always]
module Int = Make (Int.Set) (Monad.Id)
module String = Make (String.Set) (Monad.Id)