#!/usr/bin/env python3 # -*- Python -*- See copyright, etc below # pylint: disable=C0114,C0115,C0209,R0903 ###################################################################### import argparse import os import re import subprocess import tempfile import xml.etree.ElementTree as ET from shutil import copy2 # from pprint import pprint, pformat ####################################################################### class VlFileCopy: def __init__( self, verilator_args, # presently all verilator options are passed-thru debug=0, output_dir='copied'): # directory name we output file uses self.debug = debug with tempfile.NamedTemporaryFile() as xml_temp: vargs = [ '--xml-output', xml_temp.name, '--bbox-sys', # Parse some stuff can't translate '--bbox-unsup', '--prefix vlxml' ] # So we know name of .xml output vargs += verilator_args self.run_verilator(vargs) self.tree = ET.parse(xml_temp.name) os.makedirs(output_dir, 0o777, True) # Find and copy the files root = self.tree.getroot() xfiles = root.find('files') for xfile in xfiles.findall('file'): filename = xfile.get('filename') if not re.match('^<', filename): # e.g. if self.debug: print("\tcp %s %s" % (filename, output_dir)) copy2(filename, output_dir) def run_verilator(self, vargs): """Run Verilator command, check errors""" if os.getenv("VERILATOR_ROOT"): command = os.getenv("VERILATOR_ROOT") + "/bin/verilator" else: command = "verilator" command += ' ' + ' '.join(vargs) if self.debug: print("\t%s " % command) status = subprocess.call(command, shell=True) if status != 0: raise Exception("Command failed running Verilator with '" + command + "', stopped") ####################################################################### if __name__ == '__main__': parser = argparse.ArgumentParser( allow_abbrev=False, formatter_class=argparse.RawTextHelpFormatter, description= """Example of using Verilator XML output to copy a list of files to an output directory (-odir, defaults to 'copied'), e.g. to easily create a tarball of the design to pass to others. Example usage: vl_file_copy -f input.vc top.v -odir mycopy # This will make at least mycopy/top.v """, epilog="""All other arguments are pass-thru to Verilator: e.g.: +define+= Set preprocessor define -F Parse options from a file, relatively -f Parse options from a file -G= Overwrite toplevel parameter +incdir+ Directory to search for includes +libext++[ext]... Extensions for finding modules -v Verilog library -y Directory to search for modules 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 """) parser.add_argument('-debug', '--debug', action='store_const', const=9, help='enable debug') parser.add_argument('-odir', '--odir', action='store', metavar='directory', required=True, help='target output directory') (args, rem) = parser.parse_known_args() print( "NOTE: vl_file_copy is only an example starting point for writing your own tool." ) # 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. fc = VlFileCopy(output_dir=args.odir, debug=args.debug, verilator_args=rem) ###################################################################### # Local Variables: # compile-command: "./vl_file_copy -h ; VERILATOR_ROOT=$V4 ./vl_file_copy +define+thru top.v" # End: