| #!/usr/bin/env python3 |
| """ |
| Generate a Makefile .d fragment for the Verilog includes. |
| """ |
| |
| import argparse |
| import os |
| import re |
| import sys |
| |
| from io import StringIO |
| |
| from lib.deps import add_dependency |
| from lib.deps import write_deps |
| |
| parser = argparse.ArgumentParser() |
| parser.add_argument( |
| "inputfile", type=argparse.FileType('r'), help="Input Verilog file" |
| ) |
| parser.add_argument( |
| "--file_per_line", |
| action='store_true', |
| help="Output dependencies file per line, rather than Make .d format." |
| ) |
| |
| v_include = re.compile(r'`include[ ]*"([^"]*)"|\$readmem[bh]\("(.*)",(.*)\)') |
| |
| |
| def read_dependencies(inputfile): |
| """Read the dependencies out of a verilog file. |
| |
| >>> list(read_dependencies(StringIO(''' |
| ... `include "a.h" |
| ... `include "b.h" /* Cmt |
| ... `include "c.h" |
| ... */ |
| ... `include "d.h" // Cmt |
| ... // `include "e.h" |
| ... `include "f.h" |
| ... '''))) |
| ['a.h', 'b.h', 'd.h', 'f.h'] |
| |
| """ |
| data = inputfile.read() |
| # Strip out the /* */ comments |
| data = re.sub('/\\*.*\\*/', '', data, flags=re.DOTALL) |
| # Strip out the // comments |
| data = re.sub(r'//[^\n]*', '', data) |
| |
| matches = v_include.findall(data) |
| for includefile in matches: |
| yield includefile[0] + includefile[1] |
| |
| |
| def main(argv): |
| args = parser.parse_args(argv[1:]) |
| inputpath = os.path.abspath(args.inputfile.name) |
| inputdir = os.path.dirname(inputpath) |
| |
| if args.file_per_line: |
| for dep in read_dependencies(args.inputfile): |
| print(os.path.abspath(os.path.join(inputdir, dep))) |
| else: |
| data = StringIO() |
| for includefile in read_dependencies(args.inputfile): |
| add_dependency( |
| data, inputpath, |
| os.path.abspath(os.path.join(inputdir, includefile)) |
| ) |
| |
| write_deps(args.inputfile.name, data) |
| |
| |
| if __name__ == "__main__": |
| if len(sys.argv) == 1: |
| import doctest |
| failure_count, test_count = doctest.testmod() |
| assert test_count > 0 |
| assert failure_count == 0, "Doctests failed!" |
| sys.exit(main(sys.argv)) |