Release coccinelle-0.2.3rc4
[bpt/coccinelle.git] / commons / ograph_simple.ml
CommitLineData
978fd7e5
C
1open Common
2
3open Ocollection
4open Oset
5open Oassoc
6(* open Ograph *)
7
8open Oassocb
9open Osetb
10
11
12(* Difference with ograph_extended ? why not share code ? could, but
13 * in ograph_extended we dont force the user to have a key and we
14 * generate those keys as he add nodes. Here we assume the user already
15 * have an idea of what kind of key he wants to use (a string, a
ae4735db 16 * filename, a, int, whatever)
978fd7e5
C
17*)
18
19class ['key, 'a,'b] ograph_mutable =
20 let build_assoc () = new oassocb [] in
21 let build_set () = new osetb Setb.empty in
22
23object(o)
ae4735db 24
978fd7e5
C
25
26 val mutable succ = build_assoc()
27 val mutable pred = build_assoc()
28 val mutable nods = (build_assoc() : ('key, 'a) Oassocb.oassocb)
29
ae4735db
C
30 method add_node i (e: 'a) =
31 nods <- nods#add (i, e);
978fd7e5
C
32 pred <- pred#add (i, build_set() );
33 succ <- succ#add (i, build_set() );
34
ae4735db
C
35 method del_node (i) =
36 (* check: e is effectively the index associated with e,
978fd7e5
C
37 and check that already in *)
38
39 (* todo: assert that have no pred and succ, otherwise
ae4735db 40 * will have some dangling pointers
978fd7e5 41 *)
ae4735db 42 nods <- nods#delkey i;
978fd7e5
C
43 pred <- pred#delkey i;
44 succ <- succ#delkey i;
45
ae4735db 46 method del_leaf_node_and_its_edges (i) =
978fd7e5
C
47 let succ = o#successors i in
48 if not (succ#null)
49 then failwith "del_leaf_node_and_its_edges: have some successors"
50 else begin
51 let pred = o#predecessors i in
ae4735db 52 pred#tolist +> List.iter (fun (k, edge) ->
978fd7e5
C
53 o#del_arc (k,i) edge;
54 );
55 o#del_node i
56 end
ae4735db
C
57
58 method leaf_nodes () =
978fd7e5 59 let (set : 'key Oset.oset) = build_set () in
ae4735db 60 o#nodes#tolist +> List.fold_left (fun acc (k,v) ->
978fd7e5
C
61 if (o#successors k)#null
62 then acc#add k
63 else acc
64 ) set
978fd7e5 65
ae4735db
C
66
67 method replace_node i (e: 'a) =
978fd7e5
C
68 assert (nods#haskey i);
69 nods <- nods#replkey (i, e);
ae4735db
C
70
71 method add_arc (a,b) (v: 'b) =
978fd7e5
C
72 succ <- succ#replkey (a, (succ#find a)#add (b, v));
73 pred <- pred#replkey (b, (pred#find b)#add (a, v));
74 method del_arc (a,b) v =
75 succ <- succ#replkey (a, (succ#find a)#del (b,v));
76 pred <- pred#replkey (b, (pred#find b)#del (a,v));
77
78 method successors e = succ#find e
79 method predecessors e = pred#find e
80
81 method nodes = nods
82 method allsuccessors = succ
83
84 (* detect if no loop ? *)
ae4735db 85 method ancestors k =
978fd7e5
C
86 let empty_set = build_set() in
87
ae4735db
C
88
89 let rec aux acc x =
978fd7e5 90 if acc#mem x
ae4735db
C
91 then
92 (* bugfix: have_loop := true; ? not, not necessarally.
978fd7e5
C
93 * if you got a diamon, seeing a second time the same
94 * x does not mean we are in a loop
95 *)
96 acc
97 else
98 let acc = acc#add x in
99 let prefs = o#predecessors x in
100 let prefs = prefs#tolist +> List.map fst in
101 prefs +> List.fold_left (fun acc x -> aux acc x) acc
102 in
103 let set = aux empty_set k in
104 let set = set#del k in
105 set
106
107
108
109
ae4735db 110end
978fd7e5
C
111
112
113let print_ograph_generic ~str_of_key ~str_of_node filename g =
114 Common.with_open_outfile filename (fun (pr,_) ->
115 pr "digraph misc {\n" ;
116 pr "size = \"10,10\";\n" ;
117
118 let nodes = g#nodes in
ae4735db 119 nodes#iter (fun (k,node) ->
978fd7e5
C
120 pr (spf "%s [label=\"%s\"];\n" (str_of_key k) (str_of_node k node))
121 );
ae4735db 122 nodes#iter (fun (k,node) ->
978fd7e5
C
123 let succ = g#successors k in
124 succ#iter (fun (j,edge) ->
125 pr (spf "%s -> %s;\n" (str_of_key k) (str_of_key j));
126 );
127 );
128 pr "}\n" ;
129 );
130 ()