#!/usr/bin/env python3 # SPDX-License-Identifier: GPL-2.0-only import fnmatch import os import re import argparse def parse_of_declare_macros(data, include_driver_macros=True): """ Find all compatible strings in OF_DECLARE() style macros """ compat_list = [] # CPU_METHOD_OF_DECLARE does not have a compatible string if include_driver_macros: re_macros = r'(?\(\)"\-]+,\s)*"([a-zA-Z0-9_,-]+)"\)', data): compat_list.append(m[2]) return compat_list def parse_compatibles(file, compat_ignore_list): with open(file, 'r', encoding='utf-8') as f: data = f.read().replace('\n', '') if compat_ignore_list is not None: # For a compatible in the DT to be matched to a driver it needs to show # up in a driver's of_match_table match_table_list = parse_of_match_table(data) compat_list = parse_of_device_id(data, match_table_list) compat_list = [compat for compat in compat_list if compat not in compat_ignore_list] else: compat_list = parse_of_declare_macros(data) compat_list += parse_of_device_id(data) compat_list += parse_of_functions(data, "_is_compatible") compat_list += parse_of_functions(data, "of_find_compatible_node") compat_list += parse_of_functions(data, "for_each_compatible_node") compat_list += parse_of_functions(data, "of_get_compatible_child") return compat_list def parse_compatibles_to_ignore(file): with open(file, 'r', encoding='utf-8') as f: data = f.read().replace('\n', '') # Compatibles that show up in OF_DECLARE macros can't be expected to # match a driver, except for the _DRIVER ones. return parse_of_declare_macros(data, include_driver_macros=False) def print_compat(filename, compatibles): if not compatibles: return if show_filename: compat_str = ' '.join(compatibles) print(filename + ": compatible(s): " + compat_str) else: print(*compatibles, sep='\n') def glob_without_symlinks(root, glob): for path, dirs, files in os.walk(root): # Ignore hidden directories for d in dirs: if fnmatch.fnmatch(d, ".*"): dirs.remove(d) for f in files: if fnmatch.fnmatch(f, glob): yield os.path.join(path, f) def files_to_parse(path_args): for f in path_args: if os.path.isdir(f): for filename in glob_without_symlinks(f, "*.c"): yield filename else: yield f show_filename = False if __name__ == "__main__": ap = argparse.ArgumentParser() ap.add_argument("cfile", type=str, nargs='*', help="C source files or directories to parse") ap.add_argument('-H', '--with-filename', help="Print filename with compatibles", action="store_true") ap.add_argument('-d', '--driver-match', help="Only print compatibles that should match to a driver", action="store_true") args = ap.parse_args() show_filename = args.with_filename compat_ignore_list = None if args.driver_match: compat_ignore_list = [] for f in files_to_parse(args.cfile): compat_ignore_list.extend(parse_compatibles_to_ignore(f)) for f in files_to_parse(args.cfile): compat_list = parse_compatibles(f, compat_ignore_list) print_compat(f, compat_list)