2019-11-02 20:35:50 +00:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
# -*- Python -*- See copyright, etc below
|
2022-12-12 02:58:02 +00:00
|
|
|
# pylint: disable=C0103,C0114,C0115,C0115,C0116,C0209,R0914
|
2019-11-02 20:35:50 +00:00
|
|
|
######################################################################
|
|
|
|
|
|
|
|
import argparse
|
2024-02-09 22:50:09 +00:00
|
|
|
import json
|
2019-11-02 20:35:50 +00:00
|
|
|
import os
|
|
|
|
import subprocess
|
2024-02-09 22:50:09 +00:00
|
|
|
from tempfile import NamedTemporaryFile
|
2021-03-06 02:59:00 +00:00
|
|
|
# from pprint import pprint, pformat
|
2019-11-02 20:35:50 +00:00
|
|
|
|
|
|
|
#######################################################################
|
|
|
|
|
2020-12-19 03:34:14 +00:00
|
|
|
|
2019-11-02 20:35:50 +00:00
|
|
|
class VlHierGraph:
|
2022-09-28 01:47:45 +00:00
|
|
|
|
2020-12-19 03:34:14 +00:00
|
|
|
def __init__(
|
2022-09-28 01:47:45 +00:00
|
|
|
self,
|
|
|
|
verilator_args, # presently all verilator options are passed-thru
|
2022-12-12 02:58:02 +00:00
|
|
|
debug=0,
|
2022-09-28 01:47:45 +00:00
|
|
|
output_filename='graph.dot'): # output filename
|
2019-11-02 20:35:50 +00:00
|
|
|
self.debug = debug
|
|
|
|
self.next_vertex_number = 0
|
2024-02-09 22:50:09 +00:00
|
|
|
self.addr_to_number = {}
|
2019-11-02 20:35:50 +00:00
|
|
|
|
2024-08-27 01:43:34 +00:00
|
|
|
with NamedTemporaryFile() as tree_temp, NamedTemporaryFile() as meta_temp:
|
2022-12-12 02:58:02 +00:00
|
|
|
vargs = [
|
2024-02-09 22:50:09 +00:00
|
|
|
'--json-only-output',
|
|
|
|
tree_temp.name,
|
|
|
|
'--json-only-meta-output',
|
|
|
|
meta_temp.name,
|
2022-12-12 02:58:02 +00:00
|
|
|
'--bbox-sys', # Parse some stuff can't translate
|
|
|
|
'--bbox-unsup',
|
2024-02-09 22:50:09 +00:00
|
|
|
'--prefix vljson'
|
|
|
|
]
|
2022-12-12 02:58:02 +00:00
|
|
|
vargs += verilator_args
|
|
|
|
self.run_verilator(vargs)
|
2024-02-09 22:50:09 +00:00
|
|
|
self.tree = json.load(tree_temp)
|
|
|
|
self.meta = json.load(meta_temp)
|
2022-12-12 02:58:02 +00:00
|
|
|
|
|
|
|
with open(output_filename, "w", encoding="utf8") as fh:
|
2019-11-02 20:35:50 +00:00
|
|
|
# For more serious purposes, use the python graphviz package instead
|
|
|
|
fh.write("digraph {\n")
|
|
|
|
fh.write(" dpi=300;\n")
|
|
|
|
fh.write(" order=LR;\n")
|
2024-08-27 01:43:34 +00:00
|
|
|
fh.write(" node [fontsize=8 shape=\"box\" margin=0.01 width=0 height=0]")
|
2019-11-02 20:35:50 +00:00
|
|
|
fh.write(" edge [fontsize=6]")
|
|
|
|
# Find cells
|
2024-02-09 22:50:09 +00:00
|
|
|
modules = self.flatten(self.tree, lambda n: n['type'] == "MODULE")
|
|
|
|
top_module = True
|
|
|
|
for mod in modules:
|
2019-11-02 20:35:50 +00:00
|
|
|
# origNames are before parameterization, name if after
|
2024-02-09 22:50:09 +00:00
|
|
|
mod_number = self.addr_to_vertex_number(mod['addr'])
|
|
|
|
fh.write(" n%d [label=\"%s\"" % (mod_number, mod['name']))
|
|
|
|
if top_module:
|
2019-11-02 20:35:50 +00:00
|
|
|
fh.write(" color=\"red\" rank=1")
|
2024-02-09 22:50:09 +00:00
|
|
|
top_module = False
|
2019-11-02 20:35:50 +00:00
|
|
|
fh.write("];\n")
|
|
|
|
|
2024-02-09 22:50:09 +00:00
|
|
|
cells = self.flatten(mod, lambda n: n['type'] == "CELL")
|
|
|
|
for cell in cells:
|
|
|
|
def_number = self.addr_to_vertex_number(cell['modp'])
|
2020-12-19 03:34:14 +00:00
|
|
|
fh.write(" n%d->n%d [label=\"%s\"];\n" %
|
2024-02-09 22:50:09 +00:00
|
|
|
(mod_number, def_number, cell['name']))
|
2019-11-02 20:35:50 +00:00
|
|
|
|
|
|
|
fh.write("}\n")
|
|
|
|
|
2024-02-09 22:50:09 +00:00
|
|
|
def flatten(self, node, accept_cb=lambda n: True):
|
|
|
|
"""Flatten tree to list using DFS.
|
|
|
|
accept_cb(node) should return True for nodes you want to save"""
|
|
|
|
arr = []
|
|
|
|
self.dfs(node, lambda n: accept_cb(n) and arr.append(n))
|
|
|
|
return arr
|
|
|
|
|
|
|
|
def dfs(self, node, node_cb):
|
|
|
|
"""Traverse given tree using DFS and apply node_cb(node) on each one"""
|
|
|
|
node_cb(node)
|
|
|
|
for _, v in node.items():
|
|
|
|
if isinstance(v, list): # childlist
|
|
|
|
for child in v:
|
|
|
|
self.dfs(child, node_cb)
|
|
|
|
|
|
|
|
def addr_to_vertex_number(self, name):
|
|
|
|
if name not in self.addr_to_number:
|
2019-11-02 20:35:50 +00:00
|
|
|
self.next_vertex_number += 1
|
2024-02-09 22:50:09 +00:00
|
|
|
self.addr_to_number[name] = self.next_vertex_number
|
|
|
|
return self.addr_to_number[name]
|
2019-11-02 20:35:50 +00:00
|
|
|
|
2021-03-06 03:52:39 +00:00
|
|
|
def run_verilator(self, vargs):
|
2019-11-02 20:35:50 +00:00
|
|
|
"""Run Verilator command, check errors"""
|
|
|
|
if os.getenv("VERILATOR_ROOT"):
|
|
|
|
command = os.getenv("VERILATOR_ROOT") + "/bin/verilator"
|
|
|
|
else:
|
|
|
|
command = "verilator"
|
2021-03-06 03:52:39 +00:00
|
|
|
command += ' ' + ' '.join(vargs)
|
2019-11-02 20:35:50 +00:00
|
|
|
if self.debug:
|
|
|
|
print("\t%s " % command)
|
|
|
|
status = subprocess.call(command, shell=True)
|
|
|
|
if status != 0:
|
2024-08-27 01:43:34 +00:00
|
|
|
raise RuntimeError("Command failed running Verilator with '" + command + "', stopped")
|
2020-12-19 03:34:14 +00:00
|
|
|
|
2019-11-02 20:35:50 +00:00
|
|
|
|
|
|
|
#######################################################################
|
|
|
|
|
2020-12-19 03:34:14 +00:00
|
|
|
if __name__ == '__main__':
|
2019-11-02 20:35:50 +00:00
|
|
|
parser = argparse.ArgumentParser(
|
|
|
|
allow_abbrev=False,
|
|
|
|
formatter_class=argparse.RawTextHelpFormatter,
|
2024-08-27 01:43:34 +00:00
|
|
|
description="""Example of using Verilator JSON output to create a .dot file showing the
|
2019-11-02 20:35:50 +00:00
|
|
|
design module hierarchy.
|
|
|
|
|
|
|
|
Example usage:
|
|
|
|
vl_hier_graph -f input.vc top.v -o graph.dot
|
|
|
|
dot -Tpdf -o graph.pdf graph.dot
|
|
|
|
""",
|
2020-12-19 03:34:14 +00:00
|
|
|
epilog="""All other arguments are pass-thru to Verilator: e.g.:
|
2019-11-02 20:35:50 +00:00
|
|
|
|
|
|
|
+define+<var>=<value> Set preprocessor define
|
|
|
|
-F <file> Parse options from a file, relatively
|
|
|
|
-f <file> Parse options from a file
|
|
|
|
-G<name>=<value> Overwrite toplevel parameter
|
|
|
|
+incdir+<dir> Directory to search for includes
|
|
|
|
+libext+<ext>+[ext]... Extensions for finding modules
|
|
|
|
-v <filename> Verilog library
|
|
|
|
-y <dir> Directory to search for modules
|
|
|
|
|
2020-03-21 15:24:24 +00:00
|
|
|
This file ONLY is placed under the Creative Commons Public Domain, for
|
|
|
|
any use, without warranty, 2019 by Wilson Snyder.
|
|
|
|
SPDX-License-Identifier: CC0-1.0
|
2020-12-19 03:34:14 +00:00
|
|
|
""")
|
2024-08-27 01:43:34 +00:00
|
|
|
parser.add_argument('-debug', '--debug', action='store_const', const=9, help='enable debug')
|
2020-12-19 03:34:14 +00:00
|
|
|
parser.add_argument('-o',
|
|
|
|
'--o',
|
|
|
|
action='store',
|
|
|
|
metavar='filename',
|
|
|
|
required=True,
|
2019-11-02 20:35:50 +00:00
|
|
|
help='output filename')
|
|
|
|
(args, rem) = parser.parse_known_args()
|
|
|
|
|
2024-08-27 01:43:34 +00:00
|
|
|
print("NOTE: vl_hier_graph is only an example starting point for writing your own tool.")
|
2019-11-02 20:35:50 +00:00
|
|
|
# That is:
|
|
|
|
# 1. We will accept basic patches
|
|
|
|
# 2. We are not expecting to make this globally useful. (e.g. we don't cleanup obj_dir)
|
|
|
|
# 3. "make install" will not install this.
|
|
|
|
# 4. This has not had production-worthy validation.
|
|
|
|
|
2024-08-27 01:43:34 +00:00
|
|
|
fc = VlHierGraph(output_filename=args.o, debug=args.debug, verilator_args=rem)
|
2019-11-02 20:35:50 +00:00
|
|
|
|
|
|
|
######################################################################
|
2021-03-06 02:59:00 +00:00
|
|
|
# Local Variables:
|
|
|
|
# compile-command: "./vl_hier_graph -h ; VERILATOR_ROOT=$V4 ./vl_hier_graph +define+thru top.v"
|
|
|
|
# End:
|