Skip to content

Commit d1c7848

Browse files
jc2870akpm00
authored andcommitted
scripts: add macro_checker script to check unused parameters in macros
Recently, I saw a patch[1] on the ext4 mailing list regarding the correction of a macro definition error. Jan mentioned that "The bug in the macro is a really nasty trap...". Because existing compilers are unable to detect unused parameters in macro definitions. This inspired me to write a script to check for unused parameters in macro definitions and to run it. Surprisingly, the script uncovered numerous issues across various subsystems, including filesystems, drivers, and sound etc. Some of these issues involved parameters that were accepted but never used, for example: #define XFS_DAENTER_DBS(mp,w) \ (XFS_DA_NODE_MAXDEPTH + (((w) == XFS_DATA_FORK) ? 2 : 0)) where mp was unused. While others are actual bugs. For example: #define HAL_SEQ_WCSS_UMAC_CE0_SRC_REG(x) \ (ab->hw_params.regs->hal_seq_wcss_umac_ce0_src_reg) #define HAL_SEQ_WCSS_UMAC_CE0_DST_REG(x) \ (ab->hw_params.regs->hal_seq_wcss_umac_ce0_dst_reg) #define HAL_SEQ_WCSS_UMAC_CE1_SRC_REG(x) \ (ab->hw_params.regs->hal_seq_wcss_umac_ce1_src_reg) #define HAL_SEQ_WCSS_UMAC_CE1_DST_REG(x) \ (ab->hw_params.regs->hal_seq_wcss_umac_ce1_dst_reg) where x was entirely unused, and instead, a local variable ab was used. I have submitted patches[2-5] to fix some of these issues, but due to the large number, many still remain unaddressed. I believe that the kernel and matainers would benefit from this script to check for unused parameters in macro definitions. It should be noted that it may cause some false positives in conditional compilation scenarios, such as #ifdef DEBUG static int debug(arg) {}; #else #define debug(arg) #endif So the caller needs to manually verify whether it is a true issue. But this should be fine, because Maintainers should only need to review their own subsystems, which typically results in only a few reports. [1]: https://patchwork.ozlabs.org/project/linux-ext4/patch/[email protected]/ [2]: https://lore.kernel.org/linux-xfs/[email protected]/ [3]: https://lore.kernel.org/linux-bcachefs/[email protected]/ [4]: https://sourceforge.net/p/linux-f2fs/mailman/message/58797811/ [5]: https://sourceforge.net/p/linux-f2fs/mailman/message/58797812/ [[email protected]: reduce false positives] Link: https://lkml.kernel.org/r/[email protected] Link: https://lkml.kernel.org/r/[email protected] Signed-off-by: Julian Sun <[email protected]> Cc: Al Viro <[email protected]> Cc: Christian Brauner <[email protected]> Cc: Darrick J. Wong <[email protected]> Cc: Jan Kara <[email protected]> Cc: Junchao Sun <[email protected]> Cc: Kalle Valo <[email protected]> Cc: Masahiro Yamada <[email protected]> Cc: Miguel Ojeda <[email protected]> Cc: Nicolas Schier <[email protected]> Signed-off-by: Andrew Morton <[email protected]>
1 parent c6f371b commit d1c7848

File tree

1 file changed

+131
-0
lines changed

1 file changed

+131
-0
lines changed

scripts/macro_checker.py

Lines changed: 131 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,131 @@
1+
#!/usr/bin/python3
2+
# SPDX-License-Identifier: GPL-2.0
3+
# Author: Julian Sun <[email protected]>
4+
5+
""" Find macro definitions with unused parameters. """
6+
7+
import argparse
8+
import os
9+
import re
10+
11+
parser = argparse.ArgumentParser()
12+
13+
parser.add_argument("path", type=str, help="The file or dir path that needs check")
14+
parser.add_argument("-v", "--verbose", action="store_true",
15+
help="Check conditional macros, but may lead to more false positives")
16+
args = parser.parse_args()
17+
18+
macro_pattern = r"#define\s+(\w+)\(([^)]*)\)"
19+
# below vars were used to reduce false positives
20+
fp_patterns = [r"\s*do\s*\{\s*\}\s*while\s*\(\s*0\s*\)",
21+
r"\(?0\)?", r"\(?1\)?"]
22+
correct_macros = []
23+
cond_compile_mark = "#if"
24+
cond_compile_end = "#endif"
25+
26+
def check_macro(macro_line, report):
27+
match = re.match(macro_pattern, macro_line)
28+
if match:
29+
macro_def = re.sub(macro_pattern, '', macro_line)
30+
identifier = match.group(1)
31+
content = match.group(2)
32+
arguments = [item.strip() for item in content.split(',') if item.strip()]
33+
34+
macro_def = macro_def.strip()
35+
if not macro_def:
36+
return
37+
# used to reduce false positives, like #define endfor_nexthops(rt) }
38+
if len(macro_def) == 1:
39+
return
40+
41+
for fp_pattern in fp_patterns:
42+
if (re.match(fp_pattern, macro_def)):
43+
return
44+
45+
for arg in arguments:
46+
# used to reduce false positives
47+
if "..." in arg:
48+
return
49+
for arg in arguments:
50+
if not arg in macro_def and report == False:
51+
return
52+
# if there is a correct macro with the same name, do not report it.
53+
if not arg in macro_def and identifier not in correct_macros:
54+
print(f"Argument {arg} is not used in function-line macro {identifier}")
55+
return
56+
57+
correct_macros.append(identifier)
58+
59+
60+
# remove comment and whitespace
61+
def macro_strip(macro):
62+
comment_pattern1 = r"\/\/*"
63+
comment_pattern2 = r"\/\**\*\/"
64+
65+
macro = macro.strip()
66+
macro = re.sub(comment_pattern1, '', macro)
67+
macro = re.sub(comment_pattern2, '', macro)
68+
69+
return macro
70+
71+
def file_check_macro(file_path, report):
72+
# number of conditional compiling
73+
cond_compile = 0
74+
# only check .c and .h file
75+
if not file_path.endswith(".c") and not file_path.endswith(".h"):
76+
return
77+
78+
with open(file_path, "r") as f:
79+
while True:
80+
line = f.readline()
81+
if not line:
82+
break
83+
line = line.strip()
84+
if line.startswith(cond_compile_mark):
85+
cond_compile += 1
86+
continue
87+
if line.startswith(cond_compile_end):
88+
cond_compile -= 1
89+
continue
90+
91+
macro = re.match(macro_pattern, line)
92+
if macro:
93+
macro = macro_strip(macro.string)
94+
while macro[-1] == '\\':
95+
macro = macro[0:-1]
96+
macro = macro.strip()
97+
macro += f.readline()
98+
macro = macro_strip(macro)
99+
if not args.verbose:
100+
if file_path.endswith(".c") and cond_compile != 0:
101+
continue
102+
# 1 is for #ifdef xxx at the beginning of the header file
103+
if file_path.endswith(".h") and cond_compile != 1:
104+
continue
105+
check_macro(macro, report)
106+
107+
def get_correct_macros(path):
108+
file_check_macro(path, False)
109+
110+
def dir_check_macro(dir_path):
111+
112+
for dentry in os.listdir(dir_path):
113+
path = os.path.join(dir_path, dentry)
114+
if os.path.isdir(path):
115+
dir_check_macro(path)
116+
elif os.path.isfile(path):
117+
get_correct_macros(path)
118+
file_check_macro(path, True)
119+
120+
121+
def main():
122+
if os.path.isfile(args.path):
123+
get_correct_macros(args.path)
124+
file_check_macro(args.path, True)
125+
elif os.path.isdir(args.path):
126+
dir_check_macro(args.path)
127+
else:
128+
print(f"{args.path} doesn't exit or is neither a file nor a dir")
129+
130+
if __name__ == "__main__":
131+
main()

0 commit comments

Comments
 (0)