forked from rescript-lang/rescript
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathext_scc.ml
101 lines (92 loc) · 3.66 KB
/
ext_scc.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
(* Copyright (C) 2015-2016 Bloomberg Finance L.P.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* In addition to the permissions granted to you by the LGPL, you may combine
* or link a "work that uses the Library" with a publicly distributed version
* of this file to produce a combined library or application, then distribute
* that combined work under the terms of your choosing, with no requirement
* to comply with the obligations normally placed on you by section 4 of the
* LGPL version 3 (or the corresponding section of a later version of the LGPL
* should you choose to use a later version).
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *)
type node = Vec_int.t
(**
[int] as data for this algorithm
Pros:
1. Easy to eoncode algorithm (especially given that the capacity of node is known)
2. Algorithms itself are much more efficient
3. Node comparison semantics is clear
4. Easy to print output
Cons:
1. post processing input data
*)
let min_int (x : int) y = if x < y then x else y
let graph e =
let index = ref 0 in
let s = Vec_int.empty () in
let output = Int_vec_vec.empty () in
(* collect output *)
let node_numes = Array.length e in
let on_stack_array = Array.make node_numes false in
let index_array = Array.make node_numes (-1) in
let lowlink_array = Array.make node_numes (-1) in
let rec scc v_data =
let new_index = !index + 1 in
index := new_index;
Vec_int.push s v_data;
index_array.(v_data) <- new_index;
lowlink_array.(v_data) <- new_index;
on_stack_array.(v_data) <- true;
let v = e.(v_data) in
Vec_int.iter v (fun w_data ->
if Array.unsafe_get index_array w_data < 0 then (
(* not processed *)
scc w_data;
Array.unsafe_set lowlink_array v_data
(min_int
(Array.unsafe_get lowlink_array v_data)
(Array.unsafe_get lowlink_array w_data)))
else if Array.unsafe_get on_stack_array w_data then
(* successor is in stack and hence in current scc *)
Array.unsafe_set lowlink_array v_data
(min_int
(Array.unsafe_get lowlink_array v_data)
(Array.unsafe_get lowlink_array w_data)));
if
Array.unsafe_get lowlink_array v_data
= Array.unsafe_get index_array v_data
then (
(* start a new scc *)
let s_len = Vec_int.length s in
let last_index = ref (s_len - 1) in
let u = ref (Vec_int.unsafe_get s !last_index) in
while !u <> v_data do
Array.unsafe_set on_stack_array !u false;
last_index := !last_index - 1;
u := Vec_int.unsafe_get s !last_index
done;
on_stack_array.(v_data) <- false;
(* necessary *)
Int_vec_vec.push output
(Vec_int.get_and_delete_range s !last_index (s_len - !last_index)))
in
for i = 0 to node_numes - 1 do
if Array.unsafe_get index_array i < 0 then scc i
done;
output
let graph_check v =
let v = graph v in
( Int_vec_vec.length v,
Int_vec_vec.fold_left (fun acc x -> Vec_int.length x :: acc) [] v )