forked from coq-community/coq-dpdgraph
-
Notifications
You must be signed in to change notification settings - Fork 0
/
dpd_compute.ml
177 lines (150 loc) · 5.86 KB
/
dpd_compute.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
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
(*~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~*)
(* This file is part of the DpdGraph tools. *)
(* Copyright (C) 2009-2015 Anne Pacalet (Anne.Pacalet@free.fr) *)
(* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ *)
(* This file is distributed under the terms of the *)
(* GNU Lesser General Public License Version 2.1 *)
(* (see the enclosed LICENSE file for mode details) *)
(*~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~*)
let debug_flag = ref false
let with_defs = ref true
let reduce_trans = ref true
let pp intro format = Format.printf "%s" intro ; Format.printf format
let debug format =
if !debug_flag then pp "(debug): " format
else Format.ifprintf Format.std_formatter format
let error format = pp "Error: " format
let warning format = pp "Warning: " format
let feedback format = pp "Info: " format
type error =
| OpenFileError of string
| LexicalError of Lexing.position * string
| UnterminatedComment of Lexing.position option
| ParsingError of Lexing.position * Lexing.position
| EdgeWithoutNode of int
| NodeWithSameId of int * string * string
exception Error of error
let pp_lex_pos fmt p = Format.fprintf fmt "(line:%d, character:%d)"
p.Lexing.pos_lnum
(p.Lexing.pos_cnum - p.Lexing.pos_bol)
let pp_lex_inter fmt (p1, p2) =
let l1 = p1.Lexing.pos_lnum in
let c1 = p1.Lexing.pos_cnum - p1.Lexing.pos_bol in
let l2 = p2.Lexing.pos_lnum in
let c2 = p2.Lexing.pos_cnum - p2.Lexing.pos_bol in
if l1 = l2 then
if c1 = c2 then
pp_lex_pos fmt p1
else
Format.fprintf fmt "(line:%d, character:%d-%d)" l1 c1 c2
else
Format.fprintf fmt "between %a and %a" pp_lex_pos p1 pp_lex_pos p2
let pp_error = function
| OpenFileError msg ->
error "%s.@." msg
| LexicalError (pos, str) ->
error "%a: illegal character '%s'.@." pp_lex_pos pos str
| UnterminatedComment (Some pos) ->
error "unterminated comment (started near %a).@." pp_lex_pos pos
| UnterminatedComment None ->
error "unterminated comment.@."
| ParsingError (p1, p2) ->
error "parsing error %a.@." pp_lex_inter (p1, p2)
| EdgeWithoutNode node_id ->
error "no node with number %d: cannot build edge.@." node_id
| NodeWithSameId (node_id, old_name, name) ->
error "a node named '%s' already has the number %d. \
Cannot create new node named '%s' with the same number.@."
old_name node_id name
let get_attrib a attribs =
try Some (List.assoc a attribs) with Not_found -> None
let bool_attrib a attribs = match get_attrib a attribs with
| Some "yes" -> Some true
| Some "no" -> Some false
| Some _ (* TODO : warning ? *)
| None -> None
module Node = struct
type t = int * string * (string * string) list
let id (id, _, _) = id
let name (_, name, _) = name
let attribs (_, _, attribs) = attribs
let get_attrib a n = get_attrib a (attribs n)
let bool_attrib a n = bool_attrib a (attribs n)
let hash n = id n
let equal n1 n2 = id n1 = id n2
let compare n1 n2 = Pervasives.compare (id n1) (id n2)
end
module Edge = struct
type t = (string * string) list
let get_attrib a e = get_attrib a e
let bool_attrib a e = bool_attrib a e
let compare e1 e2 = Pervasives.compare e1 e2
let default = []
end
module G = Graph.Imperative.Digraph.ConcreteLabeled(Node)(Edge)
type t_obj = N of Node.t | E of (int * int * (string * string) list)
let build_graph lobj =
let g = G.create () in
let node_tbl = Hashtbl.create 10 in
let get_node id =
try Hashtbl.find node_tbl id
with Not_found -> raise (Error (EdgeWithoutNode id))
in
let add_obj o = match o with
| N ((id, _, _) as n) ->
begin
try
let old_n = Hashtbl.find node_tbl id in
raise (Error (NodeWithSameId (id, Node.name old_n, Node.name n)))
with Not_found ->
Hashtbl.add node_tbl id n;
let n = G.V.create n in G.add_vertex g n
end
| E (id1, id2, attribs) ->
let e = G.E.create (get_node id1) attribs (get_node id2) in
G.add_edge_e g e
in List.iter add_obj lobj;
g
(** remove edge (n1 -> n2) iff n2 is indirectly reachable by n1,
* or if n1 and n2 are the same *)
let reduce_graph g =
(* a table in which each node is mapped to the set of indirected accessible
* nodes *)
let module Vset = Set.Make (G.V) in
let reach_tbl = Hashtbl.create (G.nb_vertex g) in
let rec reachable v =
try Hashtbl.find reach_tbl v (* already done *)
with Not_found ->
let nb_succ_before = List.length (G.succ g v) in
let add_succ_reachable acc s =
let acc = (* add [s] successors *)
List.fold_left (fun set x -> Vset.add x set) acc (G.succ g s)
in (Vset.union acc (if Node.equal v s then Vset.empty else reachable s))
in
let acc = List.fold_left add_succ_reachable Vset.empty (G.succ g v) in
(* try to remove edges *)
let rm_edge sv = if Vset.mem sv acc then G.remove_edge g v sv in
List.iter rm_edge (G.succ g v);
let nb_succ_after = List.length (G.succ g v) in
debug "Reduce for %s : %d -> %d@." (Node.name v)
nb_succ_before nb_succ_after;
Hashtbl.add reach_tbl v acc;
acc
in
G.iter_vertex (fun v -> ignore (reachable v)) g
let remove_node g n =
let transfer_edges p =
G.remove_edge g p n;
List.iter (fun s -> G.add_edge g p s) (G.succ g n)
in
List.iter transfer_edges (G.pred g n);
G.remove_vertex g n (* also remove edges n -> s *)
let remove_some_nodes g =
let do_v v = match Node.bool_attrib "prop" v with
| None | Some false -> remove_node g v
| Some true -> ()
in
G.iter_vertex do_v g
let simplify_graph g =
if not !with_defs then remove_some_nodes g;
if !reduce_trans then reduce_graph g;