| #!/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'(?<!CPU_METHOD_)(IRQCHIP|OF)_(DECLARE|MATCH)(_DRIVER)?\(.*?\)' |
| else: |
| re_macros = r'(?<!CPU_METHOD_)(IRQCHIP|OF)_(DECLARE|MATCH)\(.*?\)' |
| for m in re.finditer(re_macros, data): |
| try: |
| compat = re.search(r'"(.*?)"', m[0])[1] |
| except: |
| # Fails on compatible strings in #define, so just skip |
| continue |
| compat_list += [compat] |
| |
| return compat_list |
| |
| |
| def parse_of_device_id(data, match_table_list=None): |
| """ Find all compatible strings in of_device_id structs """ |
| compat_list = [] |
| for m in re.finditer(r'of_device_id(\s+\S+)?\s+(\S+)\[\](\s+\S+)?\s*=\s*({.*?);', data): |
| if match_table_list is not None and m[2] not in match_table_list: |
| continue |
| compat_list += re.findall(r'\.compatible\s+=\s+"(\S+)"', m[4]) |
| |
| return compat_list |
| |
| |
| def parse_of_match_table(data): |
| """ Find all driver's of_match_table """ |
| match_table_list = [] |
| for m in re.finditer(r'\.of_match_table\s+=\s+(of_match_ptr\()?([a-zA-Z0-9_-]+)', data): |
| match_table_list.append(m[2]) |
| |
| return match_table_list |
| |
| |
| def parse_of_functions(data, func_name): |
| """ Find all compatibles in the last argument of a given function """ |
| compat_list = [] |
| for m in re.finditer(rf'{func_name}\(([a-zA-Z0-9_>\(\)"\-]+,\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) |