-
Notifications
You must be signed in to change notification settings - Fork 12
/
dot.ml
151 lines (137 loc) · 4.95 KB
/
dot.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
(*! Graphviz backend !*)
open Common
open Cuttlebone
open Cuttlebone.Graphs
open Printf
let bits1_name =
let open Extr.PrimTyped in
function
| Not _sz -> "Not"
| SExt (_, _) -> "SExt"
| ZExtL (_sz, _width) -> "ZExtL"
| ZExtR (_sz, _width) -> "ZExtR"
| Repeat (_, _) -> "Repeat"
| Slice (_logsz, _offset, _width) -> "Slice"
| Lowered (IgnoreBits _) -> "IgnoreBits"
| Lowered (DisplayBits _) -> "DisplayBits"
let comparison_name (cmp: Cuttlebone.Extr.bits_comparison)=
match cmp with
| CLt -> "lt"
| CGt -> "gt"
| CLe -> "le"
| CGe -> "ge"
let bits2_name =
let open Extr.PrimTyped in
function
| Sel _logsz -> "Sel"
| SliceSubst (_logsz, _offset, _width) -> "SliceSubst"
| IndexedSlice (_logsz, _width) -> "IndexedSlice"
| And _sz -> "And"
| Or _sz -> "Or"
| Xor _sz -> "Xor"
| Lsl (_sz, _places) -> "Lsl"
| Lsr (_sz, _places) -> "Lsr"
| Asr (_sz, _places) -> "Asr"
| Concat (_sz1, _sz2) -> "Concat"
| Plus _sz -> "Plus"
| Minus _sz -> "Minus"
| Mul (_sz1, _sz2) -> "Mul"
| EqBits (_sz, false) -> "Eq"
| EqBits (_sz, true) -> "NEq"
| Compare (signed, op, _sz) ->
sprintf "%s %s" (if signed then "signed" else "unsigned")
(comparison_name op)
let rec label_ptrs tag_to_parents = function
| CMux (sz, s, c1, c2) -> Some (sprintf "Mux@%d" sz, [s; c1; c2], [])
| CUnop (fn, c) -> Some (bits1_name fn, [c], [])
| CBinop (fn, c1, c2) -> Some (bits2_name fn, [c1; c2], [])
| CExternal { f; arg; _ } -> Some (f.ffi_name, [arg], [])
| CReadRegister r -> Some (sprintf "Reg %s" r.reg_name, [], [])
| CBundle (_, _) -> None (* FIXME *)
| CBundleRef (_, _, _) -> None (* FIXME *)
| CAnnot (_sz, annot, c) ->
let nparents = List.length (Hashtbl.find tag_to_parents c.tag) in
if nparents = 0 then
failwith "Unreachable node"
else if nparents > 1 then
(* More than one node pointing to this annotation: don't merge *)
Some (annot, [c], [])
else
(* Exactly one node pointing to this annotation: try to merge it with the
node it qualifies *)
(match label_ptrs tag_to_parents c.node with
| Some (sublabel, subcircuits, removed) ->
Some (sublabel ^ "[" ^ annot ^ "]", subcircuits, c :: removed)
| None -> Some (annot, [c], []))
| CConst _ -> None
type ptr_or_label =
| Ptr of int
| Label of string
let field_label i pl =
sprintf "<f%d> %s" i (match pl with
| Label lbl -> lbl
| Ptr _ -> "·")
let field_ptr_or_label (c: circuit) =
match c.node with
(* | CQuestionMark n -> Label (sprintf "?'%d" n) *)
| CConst bs -> Label (Util.string_of_bits bs)
| _ -> Ptr c.tag
let dot_record_label head args =
let fields = List.mapi field_label args in
let head = (if head = "constant_value_from_source" then "" else head) in
String.concat "|" (sprintf "<hd> %s" head :: fields)
let subcircuits = function
| CMux (_sz, s, c1, c2) -> [s; c1; c2]
| CUnop (_, c) -> [c]
| CBinop (_, c1, c2) -> [c1; c2]
| CExternal { arg; _ } -> [arg]
| CReadRegister _r -> []
| CBundle (_) -> [] (* FIXME *)
| CBundleRef (_, _, _) -> [] (* FIXME *)
| CAnnot (_sz, _annot, c) -> [c]
| CConst _ -> []
let hashtbl_update tbl k v_dflt v_fn =
Hashtbl.replace tbl k
(v_fn (match Hashtbl.find_opt tbl k with
| Some v -> v
| None -> v_dflt))
let compute_parents (circuits: circuit list) =
let tag_to_parents = Hashtbl.create 50 in
List.iter (fun c ->
List.iter (fun (child: circuit) ->
hashtbl_update tag_to_parents child.tag []
(fun children -> child :: children))
(subcircuits c.Hashcons.node))
circuits;
tag_to_parents
let print_dot_file out Graphs.{ graph_roots = roots; graph_nodes = circuits } =
let tag_to_parents = compute_parents circuits in
let tag_to_graph_data = Hashtbl.create 50 in
let orphaned = ref [] in
List.iter (fun Hashcons.{ node; tag; _ } ->
match label_ptrs tag_to_parents node with
| None -> ()
| Some (lbl, children, removed) ->
orphaned := List.rev_append removed !orphaned;
Hashtbl.add tag_to_graph_data tag (lbl, children))
circuits;
List.iter (fun c -> Hashtbl.remove tag_to_graph_data c.Hashcons.tag) !orphaned;
fprintf out "digraph {\n";
(* fprintf out "rankdir=BT\n"; *)
Hashtbl.iter (fun ptr (lbl, children) ->
let args_or_ptrs = List.map field_ptr_or_label children in
let lbl = dot_record_label lbl args_or_ptrs in
fprintf out "N%d [label=\"%s\", shape=\"record\"]\n" ptr lbl;
List.iteri (fun i pl ->
match pl with
| Ptr ptr' -> fprintf out "N%d:hd -> N%d:f%d\n" ptr' ptr i
| _ -> ())
args_or_ptrs)
tag_to_graph_data;
List.iteri (fun i { root_reg; root_circuit } ->
fprintf out "R%d [label=\"Register %s\", shape=\"record\"]\n" i root_reg.reg_name;
fprintf out "N%d:hd -> R%d\n" root_circuit.tag i)
roots;
fprintf out "}\n"
let main fpath g =
with_output_to_file fpath print_dot_file g