forked from mit-plv/koika
-
Notifications
You must be signed in to change notification settings - Fork 2
/
rtl.ml
623 lines (553 loc) · 24.5 KB
/
rtl.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
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
(*! Generic RTL backend !*)
open Common
open Cuttlebone
open Cuttlebone.Util
open Cuttlebone.Graphs
open Printf
(** Whether to print the circuits being compiled to Verilog. *)
let debug = false
(** By default, circuit annotations are used to generate signal names; if this
flag is set to [true] they are also used to generate comments. *)
let add_debug_annotations = false
(** Whether the design should have an always block resetting the registers to
their initial value if the RST_N signal is low. *)
let include_reset_line = true
(** Whether external rules should be implemented as internally-instantiated
modules or exposed as wires of the current module. *)
let compile_bundles_internally = false
(** Whether to add a /* verilator public */ declaration to all registers; this
is in line with what Cuttlesim does. The performance penalty seems to be in
the order of a few percents. *)
let verilator_public_registers = true
type node_metadata =
{ shared: bool; circuit: circuit' }
type metadata_map = (int, node_metadata) Hashtbl.t
type expr =
| EPtr of node * string option
| EName of string
| EConst of Common.bits_value
| EUnop of Extr.PrimTyped.fbits1 * node
| EBinop of Extr.PrimTyped.fbits2 * node * node
| EMux of node * node * node
and entity =
| Expr of expr
| Module of { name: string;
internal: bool;
inputs: input_pin list;
outputs: output_pin list }
and node =
{ tag: int;
size: int;
entity: entity;
annots: string list;
mutable kind: node_kind }
and node_kind =
| Unique
| Shared
| Printed of { name: string }
and input_pin =
{ ip_name: string; ip_node: node }
and output_pin =
{ op_name: string; op_wire: string; op_sz: int }
type node_map = (int, node) Hashtbl.t
type pin_set = (string, [`Input | `Output] * int) Hashtbl.t
let fn1_sz fn =
let fsig = Extr.PrimSignatures.coq_Sigma1 (Bits1 fn) in
typ_sz (retSig fsig)
let fn2_sz fn =
let fsig = Extr.PrimSignatures.coq_Sigma2 (Bits2 fn) in
typ_sz (retSig fsig)
let rwdata_circuits (rwd: rwdata) =
let open Extr in
[(Rwdata_r0, rwd.read0); (Rwdata_r1, rwd.read1);
(Rwdata_w0, rwd.write0); (Rwdata_w1, rwd.write1);
(Rwdata_data0, rwd.data0); (Rwdata_data1, rwd.data1)]
let compute_circuit_metadata (metadata: metadata_map) (c: circuit) =
let rec iter c =
let open Hashcons in
match Hashtbl.find_opt metadata c.tag with
| Some _ ->
Hashtbl.replace metadata c.tag { shared = true; circuit = c.node }
| None ->
iter' c.node;
assert (not (Hashtbl.mem metadata c.tag)); (* No cycles *)
Hashtbl.replace metadata c.tag { shared = false; circuit = c.node }
and iter' (node: circuit') =
match node with
| CMux (_, s, c1, c2) -> iter s; iter c1; iter c2
| CConst _ -> ()
| CReadRegister _ -> ()
| CUnop (_, c1) -> iter c1
| CBinop (_, c1, c2) -> iter c1; iter c2
| CExternal { arg; _ } -> iter arg
| CBundle (_, ios) ->
List.iter (fun (_, rwd) ->
List.iter (fun (_, c) -> iter c) (rwdata_circuits rwd))
ios
| CBundleRef (_, c, _) -> iter c
| CAnnot (_, _, c) -> iter c in
ignore (iter c)
let rwdata_field_to_string (field: Extr.rwdata_field) =
match field with
| Rwdata_r0 -> "read0"
| Rwdata_r1 -> "read1"
| Rwdata_w0 -> "write0"
| Rwdata_w1 -> "write1"
| Rwdata_data0 -> "data0"
| Rwdata_data1 -> "data1"
let rwcircuit_to_string (rwc: rwcircuit) =
match rwc with
| Rwcircuit_canfire -> "_canfire"
| Rwcircuit_rwdata (reg, field) ->
sprintf "%s_%s" reg.reg_name (rwdata_field_to_string field)
(* Our state *before* calling the external rule is an output of this module, and
the resulting state produced by the rule is an input of this module. *)
let before_prefix, after_prefix = "output", "input"
let field_name instance field =
sprintf "%s_%s" instance field
let node_of_circuit (metadata: metadata_map) (cache: node_map) (ios: pin_set) (c: circuit) =
let fresh_node size entity =
{ tag = -1; size; annots = []; entity; kind = Unique } in
let rec lift ({ tag; node = c'; _ }: circuit) =
match Hashtbl.find_opt cache tag with
| Some node -> node
| None ->
let size, annots, entity = lift' c' in
let shared = (Hashtbl.find metadata tag).shared in
let kind = if shared then Shared else Unique in
let n = { tag; size; entity; annots; kind } in
Hashtbl.replace cache n.tag n;
n
and lift' (c: circuit') =
match c with
| CMux (sz, s, c1, c2) -> sz, [], Expr (EMux (lift s, lift c1, lift c2))
| CConst c -> Array.length c, [], Expr (EConst c)
| CReadRegister r -> typ_sz (reg_type r), [], Expr (EName r.reg_name)
| CUnop (f, c1) -> fn1_sz f, [], Expr (EUnop (f, lift c1))
| CBinop (f, c1, c2) -> fn2_sz f, [], Expr (EBinop (f, lift c1, lift c2))
| CExternal { f; internal; arg } ->
let name = f.ffi_name in
let size = typ_sz f.ffi_rettype in
let inputs = [{ ip_name = "arg"; ip_node = lift arg }] in
let outputs = [{ op_name = "out"; op_wire = "out"; op_sz = typ_sz f.ffi_rettype }] in
let mod_ent = Module { name; internal; inputs; outputs } in
if not internal then register_ios name inputs outputs;
size, [], Expr (EPtr (fresh_node size mod_ent, Some "out"))
| CBundle (name, bundle_ios) ->
let name = "rule_" ^ name in
let reg_ios = List.map inputs_outputs_of_module_io bundle_ios in
let inputs, outputs = reg_ios |> List.concat |> List.split in
let cf_name = rwcircuit_to_string Rwcircuit_canfire in
let outputs = { op_name = field_name after_prefix cf_name; op_wire = cf_name; op_sz = 1 } :: outputs in
register_ios name inputs outputs;
0, [], Module { name; internal = compile_bundles_internally; inputs; outputs }
| CBundleRef (sz, c, field) ->
sz, [], Expr (EPtr (lift c, Some (field_name after_prefix (rwcircuit_to_string field))))
| CAnnot (sz, annot, c) ->
let ann = if annot = "" then [] else [annot] in
match lift c with
| { kind = Unique; annots; entity; _ } -> sz, ann @ annots, entity
| n -> sz, ann, Expr (EPtr (n, None))
and register_ios name inputs outputs =
(* The module's outputs are our inputs, and vice-versa *)
List.iter (fun op -> Hashtbl.replace ios (field_name name op.op_name) (`Input, op.op_sz)) outputs;
List.iter (fun ip -> Hashtbl.replace ios (field_name name ip.ip_name) (`Output, ip.ip_node.size)) inputs
and inputs_outputs_of_module_io (reg, rwd) =
List.map (fun (field, c) ->
let ip_node = lift c in
let nm = rwcircuit_to_string (Rwcircuit_rwdata (reg, field)) in
({ ip_name = field_name before_prefix nm; ip_node },
{ op_name = field_name after_prefix nm; op_wire = nm; op_sz = ip_node.size }))
(rwdata_circuits rwd) in
lift c
module Debug = struct
let expr_to_str = function
| EPtr (n, Some field) -> sprintf "EPtr (%d, %s)" n.tag field
| EPtr (n, None) -> sprintf "EPtr (%d, None)" n.tag
| EName n -> sprintf "EName \"%s\"" n
| EConst cst -> sprintf "EConst %s" (string_of_bits cst)
| EMux (s, c1, c2) -> sprintf "EMux (%d, %d, %d)" s.tag c1.tag c2.tag
| EUnop (f, c) -> sprintf "EUnop (%s, %d)" (unop_to_str f) c.tag
| EBinop (f, c1, c2) -> sprintf "EBinop (%s, %d, %d)" (binop_to_str f) c1.tag c2.tag
let entity_to_str = function
| Expr e -> expr_to_str e
| Module { name; internal; inputs; outputs } ->
sprintf "Module (%s, internal=%b, [%s], [%s])" name internal
(String.concat "; "
(List.map (fun { ip_name; ip_node } ->
sprintf "(%s, %d)" ip_name ip_node.tag) inputs))
(String.concat "; "
(List.map (fun { op_name; _ } ->
sprintf "%s" op_name) outputs))
let node_kind_to_str = function
| Unique -> "Unique"
| Shared -> "Shared"
| Printed { name } -> sprintf "Printed { name = \"%s\" }" name
let node_to_str { tag; entity; annots; kind; _ } =
sprintf "{ tag=%d; kind=%s; entity=%s; annots=[%s] }"
tag (node_kind_to_str kind) (entity_to_str entity) (String.concat "; " annots)
let iter_ordered f tbl =
List.iter (fun (k, v) -> f k v)
(List.sort poly_cmp (List.of_seq (Hashtbl.to_seq tbl)))
let print_node_cache cache =
if debug then
iter_ordered (fun tag node ->
eprintf "%d → %s\n" tag (node_to_str node))
cache
let print_metadata metadata =
if debug then
iter_ordered (fun tag { shared; circuit } ->
eprintf "%d → { shared=%b; circuit=%s }\n"
tag shared (circuit_to_str circuit))
metadata
let annotate annots s =
if add_debug_annotations then
match annots with
| [] -> s
| ans -> sprintf "/*<%s>*/%s" (String.concat "; " ans) s
else s
end
let compute_roots_metadata graph_roots =
let metadata = Hashtbl.create 500 in
List.iter (fun c -> compute_circuit_metadata metadata c.root_circuit) graph_roots;
Debug.print_metadata metadata;
metadata
let translate_roots metadata graph_roots =
let ios = Hashtbl.create 50 in
let cache = Hashtbl.create 500 in
let translate_root { root_reg; root_circuit } =
(root_reg.reg_name,
bits_of_value root_reg.reg_init,
node_of_circuit metadata cache ios root_circuit) in
let roots = List.map translate_root graph_roots in
Debug.print_node_cache cache;
roots, ios
let gensym_next, gensym_reset =
make_gensym ""
let rec last = function
| [] -> None
| [lst] -> Some lst
| _ :: tl -> last tl
let name_node (n: node) =
gensym_next ("_" ^ match last n.annots with Some a -> a | None -> "")
let unlowered () =
failwith "sext, zextl, zextr, etc. must be elaborated away by the compiler."
module type RTLBackend = sig
val p_module : modname:string -> out_channel -> circuit_graph -> unit
end
module Verilog : RTLBackend = struct
let may_share = function
| Module _ | Expr (EUnop _ | EBinop _ | EMux _) -> true
| _ -> false
let p_stmt out indent fmt =
fprintf out "%s" (String.make indent '\t');
kfprintf (fun out -> fprintf out ";\n") out fmt
let p_assign out name expr =
p_stmt out 1 "assign %s = %s" name expr
let sp_type kind sz =
(* if sz = 1 then kind else *) (* Printing wires as [0:0] allows slicing into them *)
sprintf "%s[%d:0]" kind (pred sz)
let p_decl out kind sz ?expr name =
let typ = sp_type kind sz in
match expr with
| None -> p_stmt out 1 "%s %s" typ name
| Some expr -> p_stmt out 1 "%s %s = %s" typ name expr
let precedence = function
| EName _ | EConst _ -> -1
| EPtr _ -> -2
| EUnop (Slice _, _) | EBinop ((Sel _ | IndexedSlice _), _, _) -> -3
| EUnop (Not _, _) | EUnop (Rev _, _) -> -4
| EBinop (Concat _, _, _) -> -5
| EUnop (Repeat _, _) -> -6
| EBinop (Mul _, _, _) -> -7
| EBinop ((Plus _ | Minus _), _, _) -> -8
| EBinop ((Lsl _ | Lsr _ | Asr _), _, _) -> -9
| EBinop (Compare (_, _, _), _, _) -> -10
| EBinop (EqBits _, _, _) -> -11
| EBinop (And _, _, _) -> -12
| EBinop (Xor _, _, _) -> -13
| EBinop (Or _, _, _) -> -14
| EMux (_, _, _) -> -15
| EUnop ((SExt _ | ZExtL _ | ZExtR _ | Lowered _), _)
| EBinop (SliceSubst _, _, _) -> unlowered ()
let complex_expr = function
| EPtr _ | EName _ -> false
(* Yosys doesn't support chained selects (x[4 +: 32][3 +: 2]) *)
| EUnop (Slice _, _) | EBinop ((Sel _ | IndexedSlice _), _, _) -> true
| EUnop _ | EBinop _ | EMux _ | EConst _ -> true
let sp_cast signed p () x =
sprintf (if signed then "$signed(%a)" else "$unsigned(%a)") p x
let reserved =
StringSet.of_list
["accept_on"; "alias"; "always"; "always_comb"; "always_ff";
"always_latch"; "and"; "assert"; "assign"; "assume"; "automatic";
"before"; "begin"; "bind"; "bins"; "binsof"; "bit"; "break"; "buf";
"bufif0"; "bufif1"; "byte"; "case"; "casex"; "casez"; "cell"; "chandle";
"checker"; "class"; "clocking"; "cmos"; "config"; "const"; "constraint";
"context"; "continue"; "cover"; "covergroup"; "coverpoint"; "cross";
"deassign"; "default"; "defparam"; "disable"; "dist"; "do"; "edge";
"else"; "end"; "endcase"; "endchecker"; "endclass"; "endclocking";
"endconfig"; "endfunction"; "endgenerate"; "endgroup"; "endinterface";
"endmodule"; "endpackage"; "endprimitive"; "endprogram"; "endproperty";
"endsequence"; "endspecify"; "endtable"; "endtask"; "enum"; "event";
"eventually"; "expect"; "export"; "extends"; "extern"; "final";
"first_match"; "for"; "force"; "foreach"; "forever"; "fork"; "forkjoin";
"function"; "generate"; "genvar"; "global"; "highz0"; "highz1"; "if";
"iff"; "ifnone"; "ignore_bins"; "illegal_bins"; "implies"; "import";
"incdir"; "include"; "initial"; "inout"; "input"; "inside"; "instance";
"int"; "integer"; "interface"; "intersect"; "join"; "join_any";
"join_none"; "large"; "let"; "liblist"; "library"; "local"; "localparam";
"logic"; "longint"; "macromodule"; "matches"; "medium"; "modport";
"module"; "nand"; "negedge"; "new"; "nexttime"; "nmos"; "nor";
"noshowcancelled"; "not"; "notif0"; "notif1"; "null"; "or"; "output";
"package"; "packed"; "parameter"; "pmos"; "posedge"; "primitive";
"priority"; "program"; "property"; "protected"; "pull0"; "pull1";
"pulldown"; "pullup"; "pulsestyle_ondetect"; "pulsestyle_onevent";
"pure"; "rand"; "randc"; "randcase"; "randsequence"; "rcmos"; "real";
"realtime"; "ref"; "reg"; "reject_on"; "release"; "repeat"; "restrict";
"return"; "rnmos"; "rpmos"; "rtran"; "rtranif0"; "rtranif1"; "s_always";
"s_eventually"; "s_nexttime"; "s_until"; "s_until_with"; "scalared";
"sequence"; "shortint"; "shortreal"; "showcancelled"; "signed"; "small";
"solve"; "specify"; "specparam"; "static"; "string"; "strong"; "strong0";
"strong1"; "struct"; "super"; "supply0"; "supply1"; "sync_accept_on";
"sync_reject_on"; "table"; "tagged"; "task"; "this"; "throughout";
"time"; "timeprecision"; "timeunit"; "tran"; "tranif0"; "tranif1"; "tri";
"tri0"; "tri1"; "triand"; "trior"; "trireg"; "type"; "typedef"; "union";
"unique"; "unique0"; "unsigned"; "until"; "until_with"; "untypted";
"use"; "var"; "vectored"; "virtual"; "void"; "wait"; "wait_order";
"wand"; "weak"; "weak0"; "weak1"; "while"; "wildcard"; "wire"; "with";
"within"; "wor"; "xnor"; "xor"]
let special_prefix = "_"
let mangling_prefix = "renamed_"
let specials_re =
Str.regexp (sprintf "^\\(%s\\|%s\\)" special_prefix mangling_prefix)
let sp_reg_name nm =
if StringSet.mem nm reserved || Str.string_match specials_re nm 0 then
mangling_prefix ^ nm
else nm
let rec p_expr out (e: expr) =
let lvl = precedence e in
let sp_str () s = s in
let p_unparens ?restricted_ctx n = p_node ?restricted_ctx out min_int n in
let p () n = snd (p_node out lvl n) in
let p0 () n = snd (p_unparens n) in
let pr () n = snd (p_unparens ~restricted_ctx:true n) in
let sp fmt = ksprintf (fun s -> (lvl, s)) fmt in
match e with
| EPtr (n, None) -> p_unparens n (* Parentheses added by caller *)
| EPtr (n, Some field) -> let nm = p () n in (lvl, field_name nm field)
| EName name -> (lvl, sp_reg_name name)
| EConst cst -> (lvl, string_of_bits cst)
| EUnop (f, e) ->
(match f with
| Not _ -> sp "~%a" p e
| Rev sz ->
let rec loop n = if n = 0
then sprintf "%a[%d]" p e n
else sprintf "%a[%d], " p e n ^ loop (n - 1) in
let rhs = if sz = 0 || sz = 1
then sprintf "%a" p e (* is that legal for a bits 0 vector? *)
else "{" ^ loop (sz-1) ^ "}" in
(lvl, rhs)
| Repeat (_, times) -> sp "{%d{%a}}" times p0 e
| Slice (_, offset, slice_sz) -> sp "%a[%d +: %d]" pr e offset slice_sz
| SExt _ | ZExtL _ | ZExtR _ | Lowered _ -> unlowered ())
| EBinop (f, e1, e2) ->
(match f with
| Mul _ -> sp "%a * %a" p e1 p e2
| Plus _ -> sp "%a + %a" p e1 p e2
| Minus _ -> sp "%a - %a" p e1 p e2
| Compare(s, CLt, _) -> sp "%a < %a" (sp_cast s p) e1 (sp_cast s p) e2
| Compare(s, CGt, _) -> sp "%a > %a" (sp_cast s p) e1 (sp_cast s p) e2
| Compare(s, CLe, _) -> sp "%a <= %a" (sp_cast s p) e1 (sp_cast s p) e2
| Compare(s, CGe, _) -> sp "%a >= %a" (sp_cast s p) e1 (sp_cast s p) e2
| Sel _ -> sp "%a[%a]" pr e1 p0 e2
| IndexedSlice (_, slice_sz) -> sp "%a[%a +: %d]" pr e1 p0 e2 slice_sz
| And 1 -> sp "%a && %a" p e1 p e2
| And _ -> sp "%a & %a" p e1 p e2
| Or 1 -> sp "%a || %a" p e1 p e2
| Or _ -> sp "%a | %a" p e1 p e2
| Xor _ -> sp "%a ^ %a" p e1 p e2
| Lsl (_, _) -> sp "%a << %a" p e1 p e2
| Lsr (_, _) -> sp "%a >> %a" p e1 p e2
| Asr (_, _) ->
(* Arithmetic shifts need an explicit cast:
reg [2:0] a = $signed(3'b100) >>> 1; // 3'b110
reg [2:0] mux_a = 1'b1 ? $signed(3'b100) >>> 1 : 3'b000; // 3'b010
reg [2:0] b = $unsigned($signed(3'b100) >>> 1); // 3'b110
reg [2:0] mux_b = 1'b1 ? $unsigned($signed(3'b100) >>> 1) : 3'b000; // 3'b110
(see https://stackoverflow.com/questions/60345469/) *)
sp "%a" (sp_cast false sp_str) (sprintf "%a >>> %a" (sp_cast true p) e1 p e2)
| EqBits (_, true) -> sp "%a != %a" p e1 p e2
| EqBits (_, false) -> sp "%a == %a" p e1 p e2
| Concat (_, _) -> sp "{%a, %a}" p0 e1 p0 e2
| SliceSubst _ -> unlowered ())
| EMux (s, e1, e2) -> sp "%a ? %a : %a" p s p e1 p e2
and p_shared_entity out n =
let p0 () n = snd (p_node out min_int n) in
match n.entity with
| Module { name; internal = true; inputs; outputs } ->
let instance_name = gensym_next ("mod_" ^ name) in
let outputs = List.map (fun op -> { op with op_wire = field_name instance_name op.op_wire }) outputs in
let in_args = List.map (fun { ip_name; ip_node } -> sprintf ".%s(%a)" ip_name p0 ip_node) inputs in
let out_args = List.map (fun { op_name; op_wire; _ } -> sprintf ".%s(%s)" op_name op_wire) outputs in
let args = String.concat ", " (".CLK(CLK)" :: ".RST_N(RST_N)" :: in_args @ out_args) in
List.iter (fun { op_wire; op_sz; _ } -> p_decl out "wire" op_sz op_wire) outputs;
p_stmt out 1 "%s %s(%s)" name instance_name args;
instance_name
| Module { name; internal = false; inputs; _ } ->
List.iter (fun { ip_name; ip_node } ->
p_assign out (field_name name ip_name) (p0 () ip_node))
inputs;
name
| Expr e ->
let _, s = p_wrapped_expr out min_int n.annots e in
let name = name_node n in
p_decl out "wire" n.size name ~expr:s;
name
and p_wrapped_expr out ctx_lvl annots expr =
let lvl, s = p_expr out expr in
let s = Debug.annotate annots s in
if lvl <= ctx_lvl then (0, sprintf "(%s)" s) else (lvl, s)
and p_shared_node out (n: node) =
let name = p_shared_entity out n in
n.kind <- Printed { name };
(0, name)
and p_node ?(restricted_ctx=false) out ctx_lvl (n: node) =
match n.kind with
| Shared when may_share n.entity -> p_shared_node out n
| Printed { name } -> (0, name)
| Unique | Shared ->
match n.entity with
| Module _ -> p_shared_node out n
| Expr e -> if restricted_ctx && complex_expr e then p_shared_node out n
else p_wrapped_expr out ctx_lvl n.annots e
let p_delayed out name expr =
p_stmt out 3 "%s <= %s" name expr
let p_root_init out roots =
List.iter (fun (name, init, _) -> p_delayed out name (string_of_bits init)) roots
let p_root_net out roots =
List.iter (fun (name, _, net) -> p_delayed out name net) roots
let p_always_block out roots =
fprintf out " always @(posedge CLK) begin\n";
if not include_reset_line then
p_root_net out roots
else begin
fprintf out " if (!RST_N) begin\n";
p_root_init out roots;
fprintf out " end else begin\n";
p_root_net out roots;
fprintf out " end\n";
end;
fprintf out " end\n"
let p_root_network out (name, init, node) =
(name, init, snd (p_node out min_int node))
let p_root_decl out (name, init, _) =
let name =
if verilator_public_registers then
name ^ " /* verilator public */"
else name in
p_decl out "reg" (Array.length init) name ~expr:(string_of_bits init)
let sp_pin (name, (direction, sz)) =
let typ = sp_type (match direction with
| `Input -> "input wire"
| `Output -> "output wire") sz in
sprintf "%s %s" typ name
let p_header out modname ios =
fprintf out "// -*- mode: verilog -*-\n";
fprintf out "// Generated by cuttlec from a Kôika module\n";
let pins = ("CLK", (`Input, 1)) :: ("RST_N", (`Input, 1)) :: ios in
fprintf out "module %s(%s);\n" modname (String.concat ", " (List.map sp_pin pins))
let p_module ~modname out { graph_roots; _ } =
let metadata = compute_roots_metadata graph_roots in
let roots, ios = translate_roots metadata graph_roots in
let roots = List.map (fun (nm, init, n) -> (sp_reg_name nm, init, n)) roots in
p_header out modname (List.of_seq (Hashtbl.to_seq ios));
List.iter (p_root_decl out) roots;
fprintf out "\n";
let root_exprs = List.map (p_root_network out) roots in
fprintf out "\n";
p_always_block out root_exprs;
fprintf out "endmodule\n"
end
module Dot = struct
let hd = "hd"
let p_decl out decl params =
fprintf out "%s%s\n" decl
(match params with
| [] -> ""
| params ->
let sp_param (key, value) = sprintf "%s=\"%s\"" key value in
sprintf " [%s]" (String.concat ", " (List.map sp_param params)))
let sp_annots annots =
match last annots with
| None -> ""
| Some a -> a
let p_edge out annots (n, f) (n', f') =
p_decl out (sprintf "%s:%s -> %s:%s" n f n' f')
(if annots = [] then [] else [("label", sp_annots annots)])
let name_of_tag tag =
sprintf "N%d" tag
type result =
| Named of string
| Inlined of string
let mark_visited node = function
| Inlined _ -> ()
| Named name -> node.kind <- Printed { name }
let sp_label annots label =
if annots = [] then label
else sprintf "%s (%s)" label (sp_annots annots)
let interesting_annot_re = Str.regexp "^var_"
let interesting_annot annot =
Str.string_match interesting_annot_re annot 0
let rec p_structure out annots name label args =
let p_arg_label idx (n, n_port) =
let port = sprintf "f%d" idx in
let label_str = match p_node out n with
| Named n_name -> p_edge out [] (n_name, n_port) (name, port); "."
| Inlined s -> s in
sprintf "<%s> %s" port label_str in
let hd_label = sprintf "<hd> %s" (sp_label annots label) in
let arg_labels = List.mapi p_arg_label args in
let label = String.concat "|" (hd_label :: arg_labels) in
p_decl out name [("label", label); ("shape", "record")];
name
and p_expr out node e =
let annots =
List.filter interesting_annot node.annots in
let p_vertex label args =
let name = name_of_tag node.tag in
Named (p_structure out annots name label args) in
match e with
| EPtr (n, None) when node.kind = Shared && annots <> [] -> p_vertex "ptr" [(n, hd)]
| EPtr (n, None) -> p_node out n
| EPtr (n, Some s) -> p_vertex "ptr" [(n, s)]
| EName nm -> p_vertex nm []
| EConst cst -> Inlined (string_of_bits cst)
| EUnop (f, n1) -> p_vertex (unop_to_str f) [(n1, hd)]
| EBinop (f, n1, n2) -> p_vertex (binop_to_str f) [(n1, hd); (n2, hd)]
| EMux (s, n1, n2) -> p_vertex "mux" [(s, hd); (n1, hd); (n2, hd)]
and p_node out (n: node) =
match n.kind with
| Printed { name } -> Named name
| Unique | Shared ->
let res = match n.entity with
| Expr e -> p_expr out n e
| Module _ -> Named "FIXME" in
mark_visited n res;
res
let p_root out (name, _, node) =
ignore (p_structure out [] name name [(node, hd)])
let p_module ~modname out { graph_roots; _ } =
let metadata = compute_roots_metadata graph_roots in
let roots, _ios = translate_roots metadata graph_roots in
fprintf out "digraph %s {\n" modname;
List.iter (p_root out) roots;
fprintf out "}\n"
let main target_dpath (modname: string) (c: circuit_graph) =
with_output_to_file (Filename.concat target_dpath (modname ^ ".dot"))
(p_module ~modname) c
end
let main target_dpath (modname: string) (c: circuit_graph) =
with_output_to_file (Filename.concat target_dpath (modname ^ ".v"))
(Verilog.p_module ~modname) c