153 lines
5.6 KiB
Python
Executable File
153 lines
5.6 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
# DESCRIPTION: Verilator: Verilog Test driver/expect definition
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify it
|
|
# under the terms of either the GNU Lesser General Public License Version 3
|
|
# or the Perl Artistic License Version 2.0.
|
|
# SPDX-FileCopyrightText: 2024 Wilson Snyder
|
|
# SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
|
|
|
|
import vltest_bootstrap
|
|
import collections
|
|
|
|
test.scenarios('dist')
|
|
|
|
Tabs_Exempt_Re = r'(\.out$)|(/fstcpp)|(Makefile)|(\.mk$)|(\.mk\.in$)|test_regress/t/t_preproc\.v|install-sh'
|
|
|
|
Unicode_Exempt_Re = r'(Changes$|CONTRIBUTORS$|LICENSES?|contributors.rst$|spelling.txt$|ci-rtlmeter-report.py)'
|
|
|
|
|
|
def get_source_files():
|
|
git_files = test.run_capture("cd " + test.root + " && git ls-files")
|
|
if test.verbose:
|
|
print("MF " + git_files)
|
|
files = {}
|
|
for filename in git_files.split():
|
|
if filename == '':
|
|
continue
|
|
files[filename] = True
|
|
return files
|
|
|
|
|
|
def check_verilog_indent(filename, contents):
|
|
# Check for proper spacing
|
|
# This isn't automated as part of CI action auto-format because .out files
|
|
# might break as a result
|
|
indents = collections.defaultdict(int)
|
|
is_on = True
|
|
for line in contents.splitlines():
|
|
if re.search(r'// verilog_format: off', line):
|
|
is_on = False
|
|
if re.search(r'// verilog_format: on', line):
|
|
is_on = True
|
|
if not is_on:
|
|
continue
|
|
# verilog-format puts input/output as double levels, ignore
|
|
if re.match(r' *(input|output|inout) ', line):
|
|
continue
|
|
|
|
m = re.match(r'^ *', line)
|
|
num_spaces = m.end() if m else 0
|
|
if num_spaces > 0:
|
|
indents[num_spaces] += 1
|
|
|
|
# pprint(indents)
|
|
spacing = 2
|
|
if (indents[3] + indents[6] + indents[9]) > (indents[2] + indents[4] + indents[6]) * 2:
|
|
spacing = 3
|
|
elif indents[2] == 0 and indents[4] > 1: # Hard because two levels of 2-space indents gives 4
|
|
spacing = 4
|
|
if spacing != 2:
|
|
warns[filename] = "Indents should be 2 spaces per level, not what seems to be " + str(
|
|
spacing) + "/level in: " + filename
|
|
|
|
|
|
if not os.path.exists(test.root + "/.git"):
|
|
test.skip("Not in a git repository")
|
|
|
|
files = get_source_files()
|
|
|
|
warns = {}
|
|
fcount = 0
|
|
for filename in sorted(files.keys()):
|
|
filename = os.path.join(test.root, filename)
|
|
if not os.path.exists(filename): # git file might be deleted but not yet staged
|
|
continue
|
|
contents = test.file_contents(filename)
|
|
if re.search(r'(\.out|\.dat)$', filename):
|
|
continue # Ignore golden files
|
|
if re.search(r'[\001\002\003\004\005\006]', contents):
|
|
continue # Ignore binary files
|
|
if contents != "" and contents[-1] != "\n":
|
|
contents += "\n"
|
|
warns[filename] = "Missing trailing newline (add one) in: " + filename
|
|
if "\r" in contents:
|
|
contents = re.sub(r'\r', '', contents)
|
|
warns[filename] = "Carriage returns (remove them) in: " + filename
|
|
if "\f" in contents:
|
|
contents = re.sub(r'\f', '', contents)
|
|
warns[filename] = "Form-feeds (remove them) in: " + filename
|
|
if "\t" in contents and not re.search(Tabs_Exempt_Re, filename):
|
|
warns[filename] = "Tabs (cannot --gold fix) in " + filename
|
|
|
|
if (re.search(r'[ \t]\n', contents)
|
|
or re.search(r'\n\n+$', contents)): # Regexp repeated below
|
|
eol_ws_exempt = ('spelling.txt' in filename or '/gtkwave/' in filename)
|
|
if eol_ws_exempt:
|
|
continue
|
|
if 'HARNESS_UPDATE_GOLDEN' in os.environ:
|
|
changes = False
|
|
(contents, n) = re.subn(r'[ \t]+\n', '\n', contents)
|
|
if n:
|
|
changes = True
|
|
if not eol_ws_exempt:
|
|
(contents, n) = re.subn(r'\n\n+$', '\n', contents)
|
|
if n:
|
|
changes = True
|
|
if not changes:
|
|
continue
|
|
warns[filename] = "Updated whitespace at " + filename
|
|
test.write_wholefile(filename, contents)
|
|
continue
|
|
|
|
lineno = 0
|
|
for line in contents.splitlines():
|
|
lineno += 1
|
|
if re.search(r'\s$', line):
|
|
warns[filename] = "Trailing whitespace at " + filename + ":" + str(lineno)
|
|
warns[filename] += " (last character is ASCII " + str(ord(line[-1])) + ")"
|
|
|
|
if not eol_ws_exempt and re.search(r'\n\n+$', contents): # Regexp repeated above
|
|
warns[filename] = "Trailing newlines at EOF in: " + filename
|
|
|
|
# Unicode checker; should this be done in another file?
|
|
# No way to auto-fix.
|
|
unicode_exempt = re.search(Unicode_Exempt_Re, filename)
|
|
m = re.search(r'(([^ \t\r\n\x20-\x7e]).*)', contents)
|
|
if not unicode_exempt and m:
|
|
warns[filename] = "Warning: non-ASCII contents '" + m.group(2) + "' at '" + m.group(
|
|
1) + "' in: " + filename
|
|
|
|
if re.search(r'\.s?vh?$', filename):
|
|
check_verilog_indent(filename, contents)
|
|
|
|
fcount += 1
|
|
|
|
if fcount < 50:
|
|
test.error("Too few source files found")
|
|
|
|
if len(warns):
|
|
# First warning lists everything as that's shown in the driver summary
|
|
msg = ""
|
|
if 'HARNESS_UPDATE_GOLDEN' in os.environ:
|
|
msg += "Updated files with whitespace errors: " + ' '.join(sorted(warns.keys())) + "\n"
|
|
else:
|
|
msg += "Files have whitespace errors: " + ' '.join(sorted(warns.keys())) + "\n"
|
|
msg += "To auto-fix (some): HARNESS_UPDATE_GOLDEN=1 {command} or --golden\n"
|
|
msg += "If change any Verilog then remember to update .out files too (with --golden)\n"
|
|
for filename in sorted(warns.keys()):
|
|
msg += warns[filename] + "\n"
|
|
test.error(msg)
|
|
|
|
test.passes()
|