| #!/usr/bin/env python3 | 
 | # Copyright lowRISC contributors. | 
 | # Licensed under the Apache License, Version 2.0, see LICENSE for details. | 
 | # SPDX-License-Identifier: Apache-2.0 | 
 | r"""Parses cdc report and dump filtered messages in hjson format. | 
 | """ | 
 | import argparse | 
 | import logging as log | 
 | import re | 
 | import sys | 
 | import os | 
 | import hjson | 
 |  | 
 | from pathlib import Path | 
 | from LintParser import LintParser | 
 |  | 
 |  | 
 | def extract_rule_patterns(file_path: Path): | 
 |     ''' | 
 |     This parses the CDC summary table to get the message totals, | 
 |     rule names and corresponding severities. | 
 |     ''' | 
 |  | 
 |     rule_patterns = [] | 
 |     full_file = '' | 
 |     try: | 
 |         with Path(file_path).open() as f: | 
 |             full_file = f.read() | 
 |     except IOError: | 
 |         # We will attempt read this file again in a second pass to parse out | 
 |         # the details, this error will get caught and reported. | 
 |         pass | 
 |  | 
 |     category = '' | 
 |     severity = '' | 
 |     known_rule_names = {} | 
 |     total_msgs = 0 | 
 |     # extract the summary table | 
 |     m = re.findall( | 
 |         r'^Summary of Policy: NEW((?:.|\n|\r\n)*)Rule Details of Policy: NEW', | 
 |         full_file, flags=re.MULTILINE) | 
 |     if m: | 
 |         # step through the table and identify rule names and their | 
 |         # category and severity | 
 |         for line in m[0].split('\n'): | 
 |             if re.match(r'^POLICY\s+NEW', line): | 
 |                 total = re.findall(r'^POLICY\s+NEW\s+([0-9]+)', line) | 
 |                 total_msgs = int(total[0]) | 
 |             elif re.match(r'^ GROUP\s+SDC_ENV_LINT', line): | 
 |                 category = 'sdc' | 
 |             elif re.match(r'^ GROUP\s+VCDC_SETUP_CHECKS', line): | 
 |                 category = 'setup' | 
 |             elif re.match(r'^ GROUP\s+VCDC_ANALYSIS_CHECKS', line): | 
 |                 category = 'cdc' | 
 |             elif re.match(r'^ GROUP\s+ERROR', line): | 
 |                 severity = 'error' | 
 |             elif re.match(r'^ GROUP\s+WARNING', line): | 
 |                 severity = 'warning' | 
 |             elif re.match(r'^ GROUP\s+INFO', line): | 
 |                 severity = 'info' | 
 |             elif re.match(r'^ GROUP\s+REVIEW', line): | 
 |                 severity = 'review' | 
 |             elif re.match(r'^  INSTANCE', line): | 
 |                 # we've found a new rule. convert it to a known rule pattern | 
 |                 # with the correct category and severity | 
 |                 rule = re.findall( | 
 |                     r'^  INSTANCE\s+([a-zA-Z0-9\_]+)\s+([0-9\_]+)', line) | 
 |                 name = rule[0][0] | 
 |                 count = int(rule[0][1]) | 
 |                 # a few rules produce messages with different severities but | 
 |                 # the same rule labels. for simplicity, we promote messages | 
 |                 # from lower severity buckets to the severity bucket where | 
 |                 # this rule name has first been encountered. since higher | 
 |                 # severity messages are listed first in this summary table, it | 
 |                 # is straightforward to check whether the rule name has | 
 |                 # already appeared in a higher severity bucket. | 
 |                 if name in known_rule_names: | 
 |                     msg_group = known_rule_names[name] | 
 |                     log.warning('Rule {} is reported in multiple severity ' | 
 |                                 'classes. All messages of this rule are ' | 
 |                                 'promoted to {}'.format(name, msg_group)) | 
 |  | 
 |                 else: | 
 |                     msg_group = category + '_' + severity | 
 |                     known_rule_names.update({name: msg_group}) | 
 |                     rule_patterns.append((msg_group, r'^{}:.*'.format(name))) | 
 |  | 
 |     return rule_patterns | 
 |  | 
 |  | 
 | # Reuse the lint parser, but add more buckets. | 
 | class CdcParser(LintParser): | 
 |  | 
 |     def __init__(self) -> None: | 
 |         self.buckets = { | 
 |             'flow_info': [], | 
 |             'flow_warning': [], | 
 |             'flow_error': [], | 
 |             'sdc_info': [], | 
 |             'sdc_review': [], | 
 |             'sdc_warning': [], | 
 |             'sdc_error': [], | 
 |             'setup_info': [], | 
 |             'setup_review': [], | 
 |             'setup_warning': [], | 
 |             'setup_error': [], | 
 |             'cdc_info': [], | 
 |             'cdc_review': [], | 
 |             'cdc_warning': [], | 
 |             'cdc_error': [], | 
 |             # this bucket is temporary and will be removed at the end of the | 
 |             # parsing pass. | 
 |             'fusesoc-error': [] | 
 |         } | 
 |         self.severities = { | 
 |             'flow_info': 'info', | 
 |             'flow_warning': 'warning', | 
 |             'flow_error': 'error', | 
 |             'sdc_info': 'info', | 
 |             'sdc_review': 'warning', | 
 |             'sdc_warning': 'warning', | 
 |             'sdc_error': 'error', | 
 |             'setup_info': 'info', | 
 |             'setup_review': 'warning', | 
 |             'setup_warning': 'warning', | 
 |             'setup_error': 'error', | 
 |             'cdc_info': 'info', | 
 |             'cdc_review': 'warning', | 
 |             'cdc_warning': 'warning', | 
 |             'cdc_error': 'error' | 
 |         } | 
 |  | 
 |  | 
 | # TODO(#9079): this script will be removed long term once the | 
 | # parser has been merged with the Dvsim core code. | 
 | def main(): | 
 |     parser = argparse.ArgumentParser( | 
 |         description="""This script parses AscentLint log and report files from | 
 |         a lint run, filters the messages and creates an aggregated result | 
 |         .hjson file with lint messages and their severities. | 
 |  | 
 |         The script returns nonzero status if any warnings or errors are | 
 |         present. | 
 |         """) | 
 |     parser.add_argument('--repdir', | 
 |                         type=lambda p: Path(p).resolve(), | 
 |                         default="./", | 
 |                         help="""The script searches the 'vcdc.log' and | 
 |                         'vcdc.rpt' files in this directory. | 
 |                         Defaults to './'""") | 
 |  | 
 |     parser.add_argument('--outfile', | 
 |                         type=lambda p: Path(p).resolve(), | 
 |                         default="./results.hjson", | 
 |                         help="""Path to the results Hjson file. | 
 |                         Defaults to './results.hjson'""") | 
 |  | 
 |     args = parser.parse_args() | 
 |  | 
 |     # Define warning/error patterns for each logfile | 
 |     parser_args = {} | 
 |  | 
 |     # Patterns for lint.log | 
 |     parser_args.update({ | 
 |         args.repdir.joinpath('build.log'): [ | 
 |         # If lint warnings have been found, the lint tool will exit | 
 |         # with a nonzero status code and fusesoc will always spit out | 
 |         # an error like | 
 |         # | 
 |         #    ERROR: Failed to build ip:core:name:0.1 : 'make' exited with an error code | 
 |         # | 
 |         # If we found any other warnings or errors, there's no point in | 
 |         # listing this too. BUT we want to make sure we *do* see this | 
 |         # error if there are no other errors or warnings, since that | 
 |         # shows something has come unstuck. (Probably the lint tool | 
 |         # spat out a warning that we don't understand) | 
 |         ("fusesoc-error", | 
 |          r"^ERROR: Failed to build .* : 'make' exited with an error code") | 
 |         ] | 
 |     }) | 
 |  | 
 |     # Patterns for vcdc.log | 
 |     parser_args.update({ | 
 |         args.repdir.joinpath('syn-icarus/vcdc.log'): [ | 
 |             ("flow_error", r"^FlexNet Licensing error.*"), | 
 |             ("flow_error", r"^Error: .*"), | 
 |             ("flow_error", r"^ERROR.*"), | 
 |             ("flow_error", r"^  ERR .*"), | 
 |             ("flow_warning", r"^Warning: .*"), | 
 |             # We ignore several messages here: | 
 |             # #25010: unused signals | 
 |             # #25011: unused signals | 
 |             # #25012: unused port | 
 |             # #25013: unused signals | 
 |             # #26038: unused or RTL constant | 
 |             # #39035: parameter becomes local | 
 |             # #39122: non-positive repeat | 
 |             # #39491: parameter in package | 
 |             ("flow_warning", r"^  " | 
 |                              "(?!WARN \[#25010\])" | 
 |                              "(?!WARN \[#25011\])" | 
 |                              "(?!WARN \[#25012\])" | 
 |                              "(?!WARN \[#25013\])" | 
 |                              "(?!WARN \[#26038\])" | 
 |                              "(?!WARN \[#39035\])" | 
 |                              "(?!WARN \[#39122\])" | 
 |                              "(?!WARN \[#39491\])" | 
 |                              "WARN .*"), | 
 |             ("flow_info", r"^  INFO .*") | 
 |         ] | 
 |     }) | 
 |  | 
 |     # The CDC messages are a bit more involved to parse out, since we | 
 |     # need to know the names and associated severities to do this. | 
 |     # The tool prints out an overview table in the report, which we are | 
 |     # going to parse first in order to get this information. | 
 |     # This is then used to construct the regex patterns to look for | 
 |     # in a second pass to get the actual CDC messages. | 
 |     cdc_rule_patterns = extract_rule_patterns( | 
 |         args.repdir.joinpath('REPORT/vcdc.new.rpt')) | 
 |  | 
 |     # Patterns for vcdc.rpt | 
 |     parser_args.update({ | 
 |         args.repdir.joinpath('REPORT/vcdc.new.rpt'): cdc_rule_patterns | 
 |     }) | 
 |  | 
 |     # Parse logs | 
 |     parser = CdcParser() | 
 |     num_messages = parser.get_results(parser_args) | 
 |  | 
 |     # Write out results file | 
 |     parser.write_results_as_hjson(args.outfile) | 
 |  | 
 |     # return nonzero status if any warnings or errors are present | 
 |     # lint infos do not count as failures | 
 |     if num_messages['error'] > 0 or num_messages['warning'] > 0: | 
 |         log.info("Found %d lint errors and %d lint warnings", | 
 |                  num_messages['error'], | 
 |                  num_messages['warning']) | 
 |         sys.exit(1) | 
 |  | 
 |     log.info("Lint logfile parsed succesfully") | 
 |     sys.exit(0) | 
 |  | 
 |  | 
 | if __name__ == "__main__": | 
 |     main() |