|  | #!/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 | 
|  |  | 
|  | # fix_include_guard.py script takes any number of C or C++ header files as | 
|  | # input, and formats their include guards to conform to the style mandated | 
|  | # by Google C++. See | 
|  | # https://google.github.io/styleguide/cppguide.html#The__define_Guard | 
|  | # | 
|  | # clang-format does not, unfortunately, have support for automatically | 
|  | # generating the correct include guards; hence the existence of this script. | 
|  | # | 
|  | # This script will write the names of all affected files to stdout; if no such | 
|  | # output is present, all files have the correct guards. This script is | 
|  | # idempotent. | 
|  |  | 
|  | import argparse | 
|  | import os | 
|  | import sys | 
|  | import re | 
|  |  | 
|  | from pathlib import Path | 
|  |  | 
|  | PROJECT_NAME = "OPENTITAN" | 
|  |  | 
|  | # This file is $REPO_TOP/util/fix_include_guard.py, so it takes two parent() | 
|  | # calls to get back to the top. | 
|  | REPO_TOP = Path(__file__).resolve().parent.parent | 
|  |  | 
|  |  | 
|  | def main(): | 
|  | parser = argparse.ArgumentParser() | 
|  | parser.add_argument( | 
|  | '--dry-run', | 
|  | action='store_true', | 
|  | help='report writes which would have happened') | 
|  | parser.add_argument( | 
|  | 'headers', | 
|  | type=str, | 
|  | nargs='+', | 
|  | help='headers to fix guards for') | 
|  | args = parser.parse_args() | 
|  |  | 
|  | total_fixes = 0 | 
|  | for header_path in args.headers: | 
|  | header = Path(header_path).resolve().relative_to(REPO_TOP) | 
|  | if header.suffix != '.h' or 'vendor' in header.parts: | 
|  | continue | 
|  |  | 
|  | uppercase_dir = re.sub(r'[^\w]', '_', str(header.parent)).upper() | 
|  | uppercase_stem = re.sub(r'[^\w]', '_', str(header.stem)).upper() | 
|  | guard = '%s_%s_%s_H_' % (PROJECT_NAME, uppercase_dir, uppercase_stem) | 
|  |  | 
|  | header_text = header.read_text() | 
|  | header_original = header_text | 
|  |  | 
|  | # Find the old guard name, which will be the first #ifndef in the file. | 
|  | old_guard = re.search(r'#ifndef +(\w+)', header_text).group(1) | 
|  |  | 
|  | # Fix the guards at the top, which are guaranteed to be there. | 
|  | header_text = re.sub('#(ifndef|define) +%s' % (old_guard, ), | 
|  | r'#\1 %s' % (guard, ), header_text) | 
|  |  | 
|  | # Fix up the endif. Since this is the last thing in the file, and it | 
|  | # might be missing the comment, we just truncate the file, and add on | 
|  | # the required guard end. | 
|  | header_text = header_text[:header_text.rindex('#endif')] | 
|  | header_text += "#endif  // %s\n" % (guard, ) | 
|  |  | 
|  | if header_text != header_original: | 
|  | print('Fixing header: "%s"' % (header, ), file=sys.stdout) | 
|  | total_fixes += 1 | 
|  | if not args.dry_run: | 
|  | header.write_text(header_text) | 
|  |  | 
|  | print('Fixed %d files.' % (total_fixes, ), file=sys.stderr) | 
|  |  | 
|  |  | 
|  | if __name__ == '__main__': | 
|  | main() |