blob: d1b902dd9627df18f6a984a1143ebc51e4cd75ff [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001# SPDX-License-Identifier: GPL-2.0+
Simon Glass0d24de92012-01-14 15:12:45 +00002# Copyright (c) 2011 The Chromium OS Authors.
3#
Simon Glass0d24de92012-01-14 15:12:45 +00004
Simon Glassd29fe6e2013-03-26 13:09:39 +00005import collections
Simon Glass0d24de92012-01-14 15:12:45 +00006import os
7import re
Vadim Bendebury99adf6e2013-01-09 16:00:10 +00008import sys
Simon Glassbf776672020-04-17 18:09:04 -06009
10from patman import command
11from patman import gitutil
12from patman import terminal
Evan Bennec6db6c2021-04-01 13:49:30 +110013
14EMACS_PREFIX = r'(?:[0-9]{4}.*\.patch:[0-9]+: )?'
15TYPE_NAME = r'([A-Z_]+:)?'
16RE_ERROR = re.compile(r'ERROR:%s (.*)' % TYPE_NAME)
17RE_WARNING = re.compile(EMACS_PREFIX + r'WARNING:%s (.*)' % TYPE_NAME)
18RE_CHECK = re.compile(r'CHECK:%s (.*)' % TYPE_NAME)
19RE_FILE = re.compile(r'#(\d+): (FILE: ([^:]*):(\d+):)?')
20RE_NOTE = re.compile(r'NOTE: (.*)')
21
Simon Glass0d24de92012-01-14 15:12:45 +000022
Simon Glassae5e9262022-01-29 14:14:06 -070023def find_check_patch():
Simon Glass0157b182022-01-29 14:14:11 -070024 top_level = gitutil.get_top_level()
Simon Glass0d24de92012-01-14 15:12:45 +000025 try_list = [
26 os.getcwd(),
27 os.path.join(os.getcwd(), '..', '..'),
Doug Andersond96ef372012-11-26 15:23:23 +000028 os.path.join(top_level, 'tools'),
29 os.path.join(top_level, 'scripts'),
Simon Glass0d24de92012-01-14 15:12:45 +000030 '%s/bin' % os.getenv('HOME'),
31 ]
32 # Look in current dir
33 for path in try_list:
34 fname = os.path.join(path, 'checkpatch.pl')
35 if os.path.isfile(fname):
36 return fname
37
38 # Look upwwards for a Chrome OS tree
39 while not os.path.ismount(path):
40 fname = os.path.join(path, 'src', 'third_party', 'kernel', 'files',
41 'scripts', 'checkpatch.pl')
42 if os.path.isfile(fname):
43 return fname
44 path = os.path.dirname(path)
Vadim Bendebury99adf6e2013-01-09 16:00:10 +000045
Masahiro Yamada31e21412014-08-16 00:59:26 +090046 sys.exit('Cannot find checkpatch.pl - please put it in your ' +
47 '~/bin directory or use --no-check')
Simon Glass0d24de92012-01-14 15:12:45 +000048
Evan Bennec6db6c2021-04-01 13:49:30 +110049
Simon Glassae5e9262022-01-29 14:14:06 -070050def check_patch_parse_one_message(message):
Evan Bennec6db6c2021-04-01 13:49:30 +110051 """Parse one checkpatch message
52
53 Args:
54 message: string to parse
55
56 Returns:
57 dict:
58 'type'; error or warning
59 'msg': text message
60 'file' : filename
61 'line': line number
62 """
63
64 if RE_NOTE.match(message):
65 return {}
66
67 item = {}
68
69 err_match = RE_ERROR.match(message)
70 warn_match = RE_WARNING.match(message)
71 check_match = RE_CHECK.match(message)
72 if err_match:
73 item['cptype'] = err_match.group(1)
74 item['msg'] = err_match.group(2)
75 item['type'] = 'error'
76 elif warn_match:
77 item['cptype'] = warn_match.group(1)
78 item['msg'] = warn_match.group(2)
79 item['type'] = 'warning'
80 elif check_match:
81 item['cptype'] = check_match.group(1)
82 item['msg'] = check_match.group(2)
83 item['type'] = 'check'
84 else:
85 message_indent = ' '
86 print('patman: failed to parse checkpatch message:\n%s' %
87 (message_indent + message.replace('\n', '\n' + message_indent)),
88 file=sys.stderr)
89 return {}
90
91 file_match = RE_FILE.search(message)
92 # some messages have no file, catch those here
93 no_file_match = any(s in message for s in [
94 '\nSubject:', 'Missing Signed-off-by: line(s)',
95 'does MAINTAINERS need updating'
96 ])
97
98 if file_match:
99 err_fname = file_match.group(3)
100 if err_fname:
101 item['file'] = err_fname
102 item['line'] = int(file_match.group(4))
103 else:
104 item['file'] = '<patch>'
105 item['line'] = int(file_match.group(1))
106 elif no_file_match:
107 item['file'] = '<patch>'
108 else:
109 message_indent = ' '
110 print('patman: failed to find file / line information:\n%s' %
111 (message_indent + message.replace('\n', '\n' + message_indent)),
112 file=sys.stderr)
113
114 return item
115
116
Simon Glassae5e9262022-01-29 14:14:06 -0700117def check_patch_parse(checkpatch_output, verbose=False):
Evan Bennec6db6c2021-04-01 13:49:30 +1100118 """Parse checkpatch.pl output
119
120 Args:
121 checkpatch_output: string to parse
122 verbose: True to print out every line of the checkpatch output as it is
123 parsed
124
125 Returns:
126 namedtuple containing:
127 ok: False=failure, True=ok
Simon Glass32cc6ae2022-02-11 13:23:18 -0700128 problems (list of problems): each a dict:
Evan Bennec6db6c2021-04-01 13:49:30 +1100129 'type'; error or warning
130 'msg': text message
131 'file' : filename
132 'line': line number
133 errors: Number of errors
134 warnings: Number of warnings
135 checks: Number of checks
136 lines: Number of lines
137 stdout: checkpatch_output
138 """
139 fields = ['ok', 'problems', 'errors', 'warnings', 'checks', 'lines',
140 'stdout']
141 result = collections.namedtuple('CheckPatchResult', fields)
142 result.stdout = checkpatch_output
143 result.ok = False
144 result.errors, result.warnings, result.checks = 0, 0, 0
145 result.lines = 0
146 result.problems = []
147
148 # total: 0 errors, 0 warnings, 159 lines checked
149 # or:
150 # total: 0 errors, 2 warnings, 7 checks, 473 lines checked
151 emacs_stats = r'(?:[0-9]{4}.*\.patch )?'
152 re_stats = re.compile(emacs_stats +
153 r'total: (\d+) errors, (\d+) warnings, (\d+)')
154 re_stats_full = re.compile(emacs_stats +
155 r'total: (\d+) errors, (\d+) warnings, (\d+)'
156 r' checks, (\d+)')
157 re_ok = re.compile(r'.*has no obvious style problems')
158 re_bad = re.compile(r'.*has style problems, please review')
159
160 # A blank line indicates the end of a message
161 for message in result.stdout.split('\n\n'):
162 if verbose:
163 print(message)
164
165 # either find stats, the verdict, or delegate
166 match = re_stats_full.match(message)
167 if not match:
168 match = re_stats.match(message)
169 if match:
170 result.errors = int(match.group(1))
171 result.warnings = int(match.group(2))
172 if len(match.groups()) == 4:
173 result.checks = int(match.group(3))
174 result.lines = int(match.group(4))
175 else:
176 result.lines = int(match.group(3))
177 elif re_ok.match(message):
178 result.ok = True
179 elif re_bad.match(message):
180 result.ok = False
181 else:
Simon Glassae5e9262022-01-29 14:14:06 -0700182 problem = check_patch_parse_one_message(message)
Evan Bennec6db6c2021-04-01 13:49:30 +1100183 if problem:
184 result.problems.append(problem)
185
186 return result
187
188
Douglas Andersondce43222022-07-19 14:56:27 -0700189def check_patch(fname, verbose=False, show_types=False, use_tree=False):
Evan Bennec6db6c2021-04-01 13:49:30 +1100190 """Run checkpatch.pl on a file and parse the results.
Simon Glass0d24de92012-01-14 15:12:45 +0000191
Simon Glass7d5b04e2020-07-05 21:41:49 -0600192 Args:
193 fname: Filename to check
194 verbose: True to print out every line of the checkpatch output as it is
195 parsed
196 show_types: Tell checkpatch to show the type (number) of each message
Douglas Andersondce43222022-07-19 14:56:27 -0700197 use_tree (bool): If False we'll pass '--no-tree' to checkpatch.
Simon Glass7d5b04e2020-07-05 21:41:49 -0600198
Simon Glass0d24de92012-01-14 15:12:45 +0000199 Returns:
Simon Glassd29fe6e2013-03-26 13:09:39 +0000200 namedtuple containing:
201 ok: False=failure, True=ok
Simon Glass0d24de92012-01-14 15:12:45 +0000202 problems: List of problems, each a dict:
203 'type'; error or warning
204 'msg': text message
205 'file' : filename
206 'line': line number
Simon Glassd29fe6e2013-03-26 13:09:39 +0000207 errors: Number of errors
208 warnings: Number of warnings
209 checks: Number of checks
Simon Glass0d24de92012-01-14 15:12:45 +0000210 lines: Number of lines
Simon Glassd29fe6e2013-03-26 13:09:39 +0000211 stdout: Full output of checkpatch
Simon Glass0d24de92012-01-14 15:12:45 +0000212 """
Simon Glassae5e9262022-01-29 14:14:06 -0700213 chk = find_check_patch()
Maxim Cournoyerda413b52023-01-13 08:50:49 -0500214 args = [chk]
Douglas Andersondce43222022-07-19 14:56:27 -0700215 if not use_tree:
216 args.append('--no-tree')
Simon Glass89fb8b72020-06-14 10:54:06 -0600217 if show_types:
218 args.append('--show-types')
Simon Glassd9800692022-01-29 14:14:05 -0700219 output = command.output(*args, fname, raise_on_error=False)
Simon Glass0d24de92012-01-14 15:12:45 +0000220
Simon Glassae5e9262022-01-29 14:14:06 -0700221 return check_patch_parse(output, verbose)
Simon Glass0d24de92012-01-14 15:12:45 +0000222
Simon Glass0d24de92012-01-14 15:12:45 +0000223
Simon Glassae5e9262022-01-29 14:14:06 -0700224def get_warning_msg(col, msg_type, fname, line, msg):
Simon Glass0d24de92012-01-14 15:12:45 +0000225 '''Create a message for a given file/line
226
227 Args:
228 msg_type: Message type ('error' or 'warning')
229 fname: Filename which reports the problem
230 line: Line number where it was noticed
231 msg: Message to report
232 '''
233 if msg_type == 'warning':
Simon Glass252ac582022-01-29 14:14:17 -0700234 msg_type = col.build(col.YELLOW, msg_type)
Simon Glass0d24de92012-01-14 15:12:45 +0000235 elif msg_type == 'error':
Simon Glass252ac582022-01-29 14:14:17 -0700236 msg_type = col.build(col.RED, msg_type)
Simon Glassd29fe6e2013-03-26 13:09:39 +0000237 elif msg_type == 'check':
Simon Glass252ac582022-01-29 14:14:17 -0700238 msg_type = col.build(col.MAGENTA, msg_type)
Simon Glass37f3bb52020-05-06 16:29:08 -0600239 line_str = '' if line is None else '%d' % line
240 return '%s:%s: %s: %s\n' % (fname, line_str, msg_type, msg)
Simon Glass0d24de92012-01-14 15:12:45 +0000241
Douglas Andersondce43222022-07-19 14:56:27 -0700242def check_patches(verbose, args, use_tree):
Simon Glass0d24de92012-01-14 15:12:45 +0000243 '''Run the checkpatch.pl script on each patch'''
Simon Glassd29fe6e2013-03-26 13:09:39 +0000244 error_count, warning_count, check_count = 0, 0, 0
Simon Glass0d24de92012-01-14 15:12:45 +0000245 col = terminal.Color()
246
247 for fname in args:
Douglas Andersondce43222022-07-19 14:56:27 -0700248 result = check_patch(fname, verbose, use_tree=use_tree)
Simon Glassd29fe6e2013-03-26 13:09:39 +0000249 if not result.ok:
250 error_count += result.errors
251 warning_count += result.warnings
252 check_count += result.checks
Paul Burtona920a172016-09-27 16:03:50 +0100253 print('%d errors, %d warnings, %d checks for %s:' % (result.errors,
Simon Glass252ac582022-01-29 14:14:17 -0700254 result.warnings, result.checks, col.build(col.BLUE, fname)))
Simon Glassd29fe6e2013-03-26 13:09:39 +0000255 if (len(result.problems) != result.errors + result.warnings +
256 result.checks):
Paul Burtona920a172016-09-27 16:03:50 +0100257 print("Internal error: some problems lost")
Simon Glass32cc6ae2022-02-11 13:23:18 -0700258 # Python seems to get confused by this
259 # pylint: disable=E1133
Simon Glassd29fe6e2013-03-26 13:09:39 +0000260 for item in result.problems:
Simon Glass8aa41362017-01-17 16:52:23 -0700261 sys.stderr.write(
Simon Glassae5e9262022-01-29 14:14:06 -0700262 get_warning_msg(col, item.get('type', '<unknown>'),
Simon Glassafb9bf52012-09-27 15:33:46 +0000263 item.get('file', '<unknown>'),
Paul Burtona920a172016-09-27 16:03:50 +0100264 item.get('line', 0), item.get('msg', 'message')))
Simon Glassd29fe6e2013-03-26 13:09:39 +0000265 print
Paul Burtona920a172016-09-27 16:03:50 +0100266 #print(stdout)
Simon Glassd29fe6e2013-03-26 13:09:39 +0000267 if error_count or warning_count or check_count:
268 str = 'checkpatch.pl found %d error(s), %d warning(s), %d checks(s)'
Simon Glass0d24de92012-01-14 15:12:45 +0000269 color = col.GREEN
270 if warning_count:
271 color = col.YELLOW
272 if error_count:
273 color = col.RED
Simon Glass252ac582022-01-29 14:14:17 -0700274 print(col.build(color, str % (error_count, warning_count, check_count)))
Simon Glass0d24de92012-01-14 15:12:45 +0000275 return False
276 return True